diff --git a/main/exercise/AnswerInOfficeDoc.php b/main/exercise/AnswerInOfficeDoc.php new file mode 100644 index 00000000000..9ab5a00fdbf --- /dev/null +++ b/main/exercise/AnswerInOfficeDoc.php @@ -0,0 +1,216 @@ +get('enable_onlyoffice_plugin')) { + throw new Exception(get_lang('OnlyOfficePluginRequired')); + } + + parent::__construct(); + $this->type = ANSWER_IN_OFFICE_DOC; + $this->isContent = $this->getIsContent(); + } + + /** + * Initialize the file path structure. + */ + public function initFile(int $sessionId, int $userId, int $exerciseId, int $exeId): void + { + $this->sessionId = $sessionId ?: 0; + $this->userId = $userId; + $this->exerciseId = $exerciseId ?: 0; + $this->exeId = $exeId; + + $this->storePath = $this->generateDirectory(); + $this->fileName = $this->generateFileName(); + $this->filePath = $this->storePath . $this->fileName; + } + + /** + * Create form for uploading an Office document. + */ + public function createAnswersForm($form): void + { + if (!empty($this->exerciseList)) { + $this->exerciseId = reset($this->exerciseList); + } + + $allowedFormats = [ + 'application/vnd.openxmlformats-officedocument.wordprocessingml.document', // .docx + 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet', // .xlsx + 'application/msword', // .doc + 'application/vnd.ms-excel' // .xls + ]; + + $form->addElement('file', 'office_file', get_lang('UploadOfficeDoc')); + $form->addRule('office_file', get_lang('ThisFieldIsRequired'), 'required'); + $form->addRule('office_file', get_lang('InvalidFileFormat'), 'mimetype', $allowedFormats); + + $allowedExtensions = implode(', ', ['.docx', '.xlsx', '.doc', '.xls']); + $form->addElement('static', 'file_hint', get_lang('AllowedFormats'), "
{$allowedExtensions}
"); + + if (!empty($this->extra)) { + $fileUrl = api_get_path(WEB_COURSE_PATH) . $this->getStoredFilePath(); + $form->addElement('static', 'current_office_file', get_lang('CurrentOfficeDoc'), "{$this->extra}"); + } + + $form->addText('weighting', get_lang('Weighting'), ['class' => 'span1']); + + global $text; + $form->addButtonSave($text, 'submitQuestion'); + + if (!empty($this->iid)) { + $form->setDefaults(['weighting' => float_format($this->weighting, 1)]); + } else { + if ($this->isContent == 1) { + $form->setDefaults(['weighting' => '10']); + } + } + } + + /** + * Process the uploaded document and save it. + */ + public function processAnswersCreation($form, $exercise): void + { + if (!empty($_FILES['office_file']['name'])) { + $extension = pathinfo($_FILES['office_file']['name'], PATHINFO_EXTENSION); + $tempFilename = "office_" . uniqid() . "." . $extension; + $tempPath = sys_get_temp_dir() . '/' . $tempFilename; + + if (!move_uploaded_file($_FILES['office_file']['tmp_name'], $tempPath)) { + return; + } + + $this->weighting = $form->getSubmitValue('weighting'); + $this->extra = ""; + $this->save($exercise); + + $this->exerciseId = $exercise->iid; + $uploadDir = $this->generateDirectory(); + + if (!is_dir($uploadDir)) { + mkdir($uploadDir, 0775, true); + } + + $filename = "office_".$this->iid.".".$extension; + $filePath = $uploadDir . $filename; + + if (!rename($tempPath, $filePath)) { + return; + } + + $this->extra = $filename; + $this->save($exercise); + } + } + + /** + * Generate the necessary directory for OnlyOffice documents. + */ + private function generateDirectory(): string + { + $exercisePath = api_get_path(SYS_COURSE_PATH).$this->course['path']."/exercises/onlyoffice/{$this->exerciseId}/{$this->iid}/"; + + if (!is_dir($exercisePath)) { + mkdir($exercisePath, 0775, true); + } + + return rtrim($exercisePath, '/') . '/'; + } + + /** + * Get the stored file path dynamically. + */ + public function getStoredFilePath(): ?string + { + if (empty($this->extra)) { + return null; + } + + return "{$this->course['path']}/exercises/onlyoffice/{$this->exerciseId}/{$this->iid}/{$this->extra}"; + } + + /** + * Get the absolute file path. Returns null if the file doesn't exist. + */ + public function getFileUrl(bool $loadFromDatabase = false): ?string + { + if ($loadFromDatabase) { + $em = Database::getManager(); + $result = $em->getRepository('ChamiloCoreBundle:TrackEAttempt')->findOneBy([ + 'exeId' => $this->exeId, + 'userId' => $this->userId, + 'questionId' => $this->iid, + 'sessionId' => $this->sessionId, + 'cId' => $this->course['real_id'], + ]); + + if (!$result || empty($result->getFilename())) { + return null; + } + + $this->fileName = $result->getFilename(); + } else { + if (empty($this->extra)) { + return null; + } + + $this->fileName = $this->extra; + } + + $filePath = $this->getStoredFilePath(); + + if (is_file(api_get_path(SYS_COURSE_PATH) . $filePath)) { + return $filePath; + } + + return null; + } + + /** + * Show the question in an exercise. + */ + public function return_header(Exercise $exercise, $counter = null, $score = []) + { + $score['revised'] = $this->isQuestionWaitingReview($score); + $header = parent::return_header($exercise, $counter, $score); + $header .= ''.get_lang("Answer").' | +
---|
'.
@@ -6463,6 +6489,24 @@ function ($answerId) use ($objAnswerTmp) {
false,
$questionDuration
);
+ } elseif ($answerType == ANSWER_IN_OFFICE_DOC) {
+ $answer = $choice;
+ $exerciseId = $this->iid;
+ $questionId = $quesId;
+ $originalFilePath = $objQuestionTmp->getFileUrl();
+ $originalExtension = !empty($originalFilePath) ? pathinfo($originalFilePath, PATHINFO_EXTENSION) : 'docx';
+ $fileName = "response_{$exeId}.{$originalExtension}";
+ Event::saveQuestionAttempt(
+ $questionScore,
+ $answer,
+ $questionId,
+ $exeId,
+ 0,
+ $exerciseId,
+ false,
+ $questionDuration,
+ $fileName
+ );
} elseif ($answerType == ORAL_EXPRESSION) {
$answer = $choice;
$absFilePath = $objQuestionTmp->getAbsoluteFilePath();
diff --git a/main/exercise/exercise_report.php b/main/exercise/exercise_report.php
index 9b027b00ef3..f87e9376e77 100755
--- a/main/exercise/exercise_report.php
+++ b/main/exercise/exercise_report.php
@@ -211,7 +211,7 @@
// From the database.
$marksFromDatabase = $questionListData[$questionId]['marks'];
- if (in_array($question->type, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION, UPLOAD_ANSWER])) {
+ if (in_array($question->type, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION, UPLOAD_ANSWER, ANSWER_IN_OFFICE_DOC])) {
// From the form.
$params['marks'] = $marks;
if ($marksFromDatabase != $marks) {
diff --git a/main/exercise/exercise_show.php b/main/exercise/exercise_show.php
index 445a2e8ebe8..f456f3c9fed 100755
--- a/main/exercise/exercise_show.php
+++ b/main/exercise/exercise_show.php
@@ -444,6 +444,7 @@ function getFCK(vals, marksid) {
case GLOBAL_MULTIPLE_ANSWER:
case FREE_ANSWER:
case UPLOAD_ANSWER:
+ case ANSWER_IN_OFFICE_DOC:
case ORAL_EXPRESSION:
case MATCHING:
case MATCHING_COMBINATION:
@@ -612,7 +613,7 @@ function getFCK(vals, marksid) {
if ($isFeedbackAllowed && $action !== 'export') {
$name = 'fckdiv'.$questionId;
$marksname = 'marksName'.$questionId;
- if (in_array($answerType, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION, UPLOAD_ANSWER])) {
+ if (in_array($answerType, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION, UPLOAD_ANSWER, ANSWER_IN_OFFICE_DOC])) {
$url_name = get_lang('EditCommentsAndMarks');
} else {
$url_name = get_lang('AddComments');
@@ -689,7 +690,7 @@ function getFCK(vals, marksid) {
}
if ($is_allowedToEdit && $isFeedbackAllowed && $action !== 'export') {
- if (in_array($answerType, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION, UPLOAD_ANSWER])) {
+ if (in_array($answerType, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION, UPLOAD_ANSWER, ANSWER_IN_OFFICE_DOC])) {
$marksname = 'marksName'.$questionId;
$arrmarks[] = $questionId;
@@ -846,7 +847,7 @@ class="exercise_mark_select"
}
}
- if (in_array($objQuestionTmp->type, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION, UPLOAD_ANSWER])) {
+ if (in_array($objQuestionTmp->type, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION, UPLOAD_ANSWER, ANSWER_IN_OFFICE_DOC])) {
$scoreToReview = [
'score' => $my_total_score,
'comments' => isset($comnt) ? $comnt : null,
diff --git a/main/exercise/question.class.php b/main/exercise/question.class.php
index 1cee986b4bc..6ecce2e3729 100755
--- a/main/exercise/question.class.php
+++ b/main/exercise/question.class.php
@@ -75,6 +75,7 @@ abstract class Question
UPLOAD_ANSWER => ['UploadAnswer.php', 'UploadAnswer'],
MULTIPLE_ANSWER_DROPDOWN => ['MultipleAnswerDropdown.php', 'MultipleAnswerDropdown'],
MULTIPLE_ANSWER_DROPDOWN_COMBINATION => ['MultipleAnswerDropdownCombination.php', 'MultipleAnswerDropdownCombination'],
+ ANSWER_IN_OFFICE_DOC => ['AnswerInOfficeDoc.php', 'AnswerInOfficeDoc'],
];
/**
@@ -110,6 +111,7 @@ public function __construct()
FILL_IN_BLANKS,
FILL_IN_BLANKS_COMBINATION,
FREE_ANSWER,
+ ANSWER_IN_OFFICE_DOC,
ORAL_EXPRESSION,
CALCULATED_ANSWER,
ANNOTATION,
@@ -1663,6 +1665,9 @@ public static function getQuestionTypeList()
self::$questionTypes[HOT_SPOT_DELINEATION] = null;
unset(self::$questionTypes[HOT_SPOT_DELINEATION]);
}
+ if ('true' !== OnlyofficePlugin::create()->get('enable_onlyoffice_plugin')) {
+ unset(self::$questionTypes[ANSWER_IN_OFFICE_DOC]);
+ }
return self::$questionTypes;
}
@@ -2248,6 +2253,7 @@ public function return_header(Exercise $exercise, $counter = null, $score = [])
case FREE_ANSWER:
case UPLOAD_ANSWER:
case ORAL_EXPRESSION:
+ case ANSWER_IN_OFFICE_DOC:
case ANNOTATION:
$score['revised'] = isset($score['revised']) ? $score['revised'] : false;
if ($score['revised'] == true) {
diff --git a/main/inc/lib/api.lib.php b/main/inc/lib/api.lib.php
index 482b7cd6d2e..22487292f5f 100755
--- a/main/inc/lib/api.lib.php
+++ b/main/inc/lib/api.lib.php
@@ -544,6 +544,7 @@
define('FILL_IN_BLANKS_COMBINATION', 27);
define('MULTIPLE_ANSWER_DROPDOWN_COMBINATION', 28);
define('MULTIPLE_ANSWER_DROPDOWN', 29);
+define('ANSWER_IN_OFFICE_DOC', 30);
define('EXERCISE_CATEGORY_RANDOM_SHUFFLED', 1);
define('EXERCISE_CATEGORY_RANDOM_ORDERED', 2);
@@ -591,6 +592,7 @@
MULTIPLE_ANSWER_TRUE_FALSE.':'.
MULTIPLE_ANSWER_COMBINATION_TRUE_FALSE.':'.
ORAL_EXPRESSION.':'.
+ ANSWER_IN_OFFICE_DOC.':'.
GLOBAL_MULTIPLE_ANSWER.':'.
MEDIA_QUESTION.':'.
CALCULATED_ANSWER.':'.
diff --git a/main/inc/lib/exercise.lib.php b/main/inc/lib/exercise.lib.php
index 32d53a14a97..e6021b41e5f 100644
--- a/main/inc/lib/exercise.lib.php
+++ b/main/inc/lib/exercise.lib.php
@@ -114,7 +114,7 @@ public static function showQuestion(
}
}
- if (in_array($answerType, [FREE_ANSWER, ORAL_EXPRESSION, UPLOAD_ANSWER]) && $freeze) {
+ if (in_array($answerType, [FREE_ANSWER, ORAL_EXPRESSION, UPLOAD_ANSWER, ANSWER_IN_OFFICE_DOC]) && $freeze) {
return '';
}
@@ -284,6 +284,22 @@ function setRemoveLink(dataContext) {
}
$s .= $multipleForm->returnForm();
break;
+ case ANSWER_IN_OFFICE_DOC:
+ if ('true' === OnlyofficePlugin::create()->get('enable_onlyoffice_plugin')) {
+ global $exe_id;
+ if (!empty($objQuestionTmp->extra)) {
+ $fileUrl = api_get_course_path()."/exercises/onlyoffice/{$exerciseId}/{$questionId}/" . $objQuestionTmp->extra;
+ $documentUrl = OnlyofficeTools::getPathToView($fileUrl, false, $exe_id, $questionId);
+ echo ' ';
+ echo "";
+ echo ' ';
+ } else {
+ echo '' . get_lang('NoOfficeDocProvided') . ' '; + } + } else { + echo '' . get_lang('OnlyOfficePluginRequired') . ' '; + } + break; case ORAL_EXPRESSION: // Add nanog if (api_get_setting('enable_record_audio') === 'true') { @@ -2576,7 +2592,7 @@ public static function get_exam_results_data( FROM $TBL_EXERCISES_REL_QUESTION terq LEFT JOIN $TBL_EXERCISES_QUESTION teq ON terq.question_id = teq.iid - WHERE teq.type in (".FREE_ANSWER.", ".ORAL_EXPRESSION.", ".ANNOTATION.", ".UPLOAD_ANSWER.") + WHERE teq.type in (".FREE_ANSWER.", ".ORAL_EXPRESSION.", ".ANNOTATION.", ".UPLOAD_ANSWER.", ".ANSWER_IN_OFFICE_DOC.") "; $resultExerciseIds = Database::query($sqlExercise); @@ -5541,7 +5557,7 @@ public static function displayQuestionListByAttempt( if ($show_results) { $score = $calculatedScore; } - if (in_array($objQuestionTmp->type, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION, UPLOAD_ANSWER])) { + if (in_array($objQuestionTmp->type, [FREE_ANSWER, ORAL_EXPRESSION, ANNOTATION, UPLOAD_ANSWER, ANSWER_IN_OFFICE_DOC])) { $reviewScore = [ 'score' => $my_total_score, 'comments' => Event::get_comments($exeId, $questionId), @@ -6419,6 +6435,7 @@ public static function getEmbeddableTypes(): array READING_COMPREHENSION, MULTIPLE_ANSWER_TRUE_FALSE_DEGREE_CERTAINTY, UPLOAD_ANSWER, + ANSWER_IN_OFFICE_DOC, MATCHING_COMBINATION, FILL_IN_BLANKS_COMBINATION, MULTIPLE_ANSWER_DROPDOWN, diff --git a/main/inc/lib/exercise_show_functions.lib.php b/main/inc/lib/exercise_show_functions.lib.php index d381836168e..4307f1bbe5f 100755 --- a/main/inc/lib/exercise_show_functions.lib.php +++ b/main/inc/lib/exercise_show_functions.lib.php @@ -1002,4 +1002,68 @@ public static function displayAnnotationAnswer( } } } + + /** + * Displays the submitted OnlyOffice document in an iframe. + * + * @param string $feedbackType The feedback type of the exercise. + * @param int $exeId The execution ID. + * @param int $userId The user ID. + * @param int $exerciseId The exercise ID. + * @param int $questionId The question ID. + * @param int $questionScore Score assigned to the response. + * @param bool $autorefresh If true, auto-refresh the iframe after a short delay (used in result view). + */ + public static function displayOnlyOfficeAnswer( + string $feedbackType, + int $exeId, + int $userId, + int $exerciseId, + int $questionId, + int $questionScore = 0, + bool $autorefresh = false + ): void { + $filePathPattern = api_get_path(SYS_COURSE_PATH).api_get_course_path()."/exercises/onlyoffice/{$exerciseId}/{$questionId}/{$userId}/response_{$exeId}.*"; + $files = glob($filePathPattern); + + if (!empty($files)) { + $fileUrl = api_get_course_path() . "/exercises/onlyoffice/{$exerciseId}/{$questionId}/{$userId}/" . basename($files[0]); + $iframeId = "onlyoffice_result_frame_{$exerciseId}_{$questionId}_{$exeId}_{$userId}"; + $loaderId = "onlyoffice_loader_{$exerciseId}_{$questionId}_{$exeId}_{$userId}"; + $iframeSrc = OnlyofficeTools::getPathToView($fileUrl, false, $exeId, $questionId, true); + $iframeSrc .= '&t=' . time(); + echo ' + |
+ ' . get_lang('SubmittedDocument') . ': '; + if ($autorefresh) { + echo ' +
+
+ ';
+ echo "";
+ } else {
+ echo '
+ ';
+ }
+ echo '' . get_lang('LoadingLatestVersion') . '... + |
' . get_lang('NoOfficeDocProvided') . ' |
' . ExerciseLib::getNotCorrectedYetText() . ' |