3 // This file is part of Moodle - http://moodle.org/
5 // Moodle is free software: you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation, either version 3 of the License, or
8 // (at your option) any later version.
10 // Moodle is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 // GNU General Public License for more details.
15 // You should have received a copy of the GNU General Public License
16 // along with Moodle. If not, see <http://www.gnu.org/licenses/>.
19 * View a single (usually the own) submission, submit own work.
21 * @package mod_workshop
22 * @copyright 2009 David Mudrak <david.mudrak@gmail.com>
23 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
26 require_once(dirname(dirname(dirname(__FILE__
))).'/config.php');
27 require_once(dirname(__FILE__
).'/locallib.php');
28 require_once($CFG->dirroot
. '/repository/lib.php');
30 $cmid = required_param('cmid', PARAM_INT
); // course module id
31 $id = optional_param('id', 0, PARAM_INT
); // submission id
32 $edit = optional_param('edit', false, PARAM_BOOL
); // open for editing?
33 $assess = optional_param('assess', false, PARAM_BOOL
); // instant assessment required
35 $cm = get_coursemodule_from_id('workshop', $cmid, 0, false, MUST_EXIST
);
36 $course = $DB->get_record('course', array('id' => $cm->course
), '*', MUST_EXIST
);
38 require_login($course, false, $cm);
40 print_error('guestsarenotallowed');
43 $workshoprecord = $DB->get_record('workshop', array('id' => $cm->instance
), '*', MUST_EXIST
);
44 $workshop = new workshop($workshoprecord, $cm, $course);
46 $PAGE->set_url($workshop->submission_url(), array('cmid' => $cmid, 'id' => $id));
49 $PAGE->url
->param('edit', $edit);
52 if ($id) { // submission is specified
53 $submission = $workshop->get_submission_by_id($id);
56 'objectid' => $submission->id
,
57 'context' => $workshop->context
,
58 'courseid' => $workshop->course
->id
,
59 'relateduserid' => $submission->authorid
,
61 'workshopid' => $workshop->id
65 $event = \mod_workshop\event\submission_viewed
::create($params);
68 } else { // no submission specified
69 if (!$submission = $workshop->get_submission_by_author($USER->id
)) {
70 $submission = new stdclass();
71 $submission->id
= null;
72 $submission->authorid
= $USER->id
;
73 $submission->example
= 0;
74 $submission->grade
= null;
75 $submission->gradeover
= null;
76 $submission->published
= null;
77 $submission->feedbackauthor
= null;
78 $submission->feedbackauthorformat
= editors_get_preferred_format();
82 $ownsubmission = $submission->authorid
== $USER->id
;
83 $canviewall = has_capability('mod/workshop:viewallsubmissions', $workshop->context
);
84 $cansubmit = has_capability('mod/workshop:submit', $workshop->context
);
85 $canallocate = has_capability('mod/workshop:allocate', $workshop->context
);
86 $canpublish = has_capability('mod/workshop:publishsubmissions', $workshop->context
);
87 $canoverride = (($workshop->phase
== workshop
::PHASE_EVALUATION
) and has_capability('mod/workshop:overridegrades', $workshop->context
));
88 $userassessment = $workshop->get_assessment_of_submission_by_user($submission->id
, $USER->id
);
89 $isreviewer = !empty($userassessment);
90 $editable = ($cansubmit and $ownsubmission);
91 $ispublished = ($workshop->phase
== workshop
::PHASE_CLOSED
92 and $submission->published
== 1
93 and has_capability('mod/workshop:viewpublishedsubmissions', $workshop->context
));
95 if (empty($submission->id
) and !$workshop->creating_submission_allowed($USER->id
)) {
98 if ($submission->id
and !$workshop->modifying_submission_allowed($USER->id
)) {
103 // check this flag against the group membership yet
104 if (groups_get_activity_groupmode($workshop->cm
) == SEPARATEGROUPS
) {
105 // user must have accessallgroups or share at least one group with the submission author
106 if (!has_capability('moodle/site:accessallgroups', $workshop->context
)) {
107 $usersgroups = groups_get_activity_allowed_groups($workshop->cm
);
108 $authorsgroups = groups_get_all_groups($workshop->course
->id
, $submission->authorid
, $workshop->cm
->groupingid
, 'g.id');
109 $sharedgroups = array_intersect_key($usersgroups, $authorsgroups);
110 if (empty($sharedgroups)) {
117 if ($editable and $workshop->useexamples
and $workshop->examplesmode
== workshop
::EXAMPLES_BEFORE_SUBMISSION
118 and !has_capability('mod/workshop:manageexamples', $workshop->context
)) {
119 // check that all required examples have been assessed by the user
120 $examples = $workshop->get_examples_for_reviewer($USER->id
);
121 foreach ($examples as $exampleid => $example) {
122 if (is_null($example->grade
)) {
128 $edit = ($editable and $edit);
130 $seenaspublished = false; // is the submission seen as a published submission?
132 if ($submission->id
and ($ownsubmission or $canviewall or $isreviewer)) {
134 } elseif ($submission->id
and $ispublished) {
136 $seenaspublished = true;
137 } elseif (is_null($submission->id
) and $cansubmit) {
140 print_error('nopermissions', 'error', $workshop->view_url(), 'view or create submission');
143 if ($assess and $submission->id
and !$isreviewer and $canallocate and $workshop->assessing_allowed($USER->id
)) {
145 $assessmentid = $workshop->add_allocation($submission, $USER->id
);
146 redirect($workshop->assess_url($assessmentid));
150 require_once(dirname(__FILE__
).'/submission_form.php');
152 $maxfiles = $workshop->nattachments
;
153 $maxbytes = $workshop->maxbytes
;
154 $contentopts = array(
157 'maxfiles' => $maxfiles,
158 'maxbytes' => $maxbytes,
159 'context' => $workshop->context
,
160 'return_types' => FILE_INTERNAL | FILE_EXTERNAL
163 $attachmentopts = array('subdirs' => true, 'maxfiles' => $maxfiles, 'maxbytes' => $maxbytes, 'return_types' => FILE_INTERNAL
);
164 $submission = file_prepare_standard_editor($submission, 'content', $contentopts, $workshop->context
,
165 'mod_workshop', 'submission_content', $submission->id
);
166 $submission = file_prepare_standard_filemanager($submission, 'attachment', $attachmentopts, $workshop->context
,
167 'mod_workshop', 'submission_attachment', $submission->id
);
169 $mform = new workshop_submission_form($PAGE->url
, array('current' => $submission, 'workshop' => $workshop,
170 'contentopts' => $contentopts, 'attachmentopts' => $attachmentopts));
172 if ($mform->is_cancelled()) {
173 redirect($workshop->view_url());
175 } elseif ($cansubmit and $formdata = $mform->get_data()) {
176 if ($formdata->example
== 0) {
177 // this was used just for validation, it must be set to zero when dealing with normal submissions
178 unset($formdata->example
);
180 throw new coding_exception('Invalid submission form data value: example');
183 if (is_null($submission->id
)) {
184 $formdata->workshopid
= $workshop->id
;
185 $formdata->example
= 0;
186 $formdata->authorid
= $USER->id
;
187 $formdata->timecreated
= $timenow;
188 $formdata->feedbackauthorformat
= editors_get_preferred_format();
190 $formdata->timemodified
= $timenow;
191 $formdata->title
= trim($formdata->title
);
192 $formdata->content
= ''; // updated later
193 $formdata->contentformat
= FORMAT_HTML
; // updated later
194 $formdata->contenttrust
= 0; // updated later
195 $formdata->late
= 0x0; // bit mask
196 if (!empty($workshop->submissionend
) and ($workshop->submissionend
< time())) {
197 $formdata->late
= $formdata->late |
0x1;
199 if ($workshop->phase
== workshop
::PHASE_ASSESSMENT
) {
200 $formdata->late
= $formdata->late |
0x2;
203 // Event information.
205 'context' => $workshop->context
,
206 'courseid' => $workshop->course
->id
,
208 'submissiontitle' => $formdata->title
212 if (is_null($submission->id
)) {
213 $submission->id
= $formdata->id
= $DB->insert_record('workshop_submissions', $formdata);
214 $params['objectid'] = $submission->id
;
215 $event = \mod_workshop\event\submission_created
::create($params);
218 if (empty($formdata->id
) or empty($submission->id
) or ($formdata->id
!= $submission->id
)) {
219 throw new moodle_exception('err_submissionid', 'workshop');
222 $params['objectid'] = $submission->id
;
223 // save and relink embedded images and save attachments
224 $formdata = file_postupdate_standard_editor($formdata, 'content', $contentopts, $workshop->context
,
225 'mod_workshop', 'submission_content', $submission->id
);
226 $formdata = file_postupdate_standard_filemanager($formdata, 'attachment', $attachmentopts, $workshop->context
,
227 'mod_workshop', 'submission_attachment', $submission->id
);
228 if (empty($formdata->attachment
)) {
229 // explicit cast to zero integer
230 $formdata->attachment
= 0;
232 // store the updated values or re-save the new submission (re-saving needed because URLs are now rewritten)
233 $DB->update_record('workshop_submissions', $formdata);
234 $event = \mod_workshop\event\submission_updated
::create($params);
235 $event->add_record_snapshot('workshop', $workshoprecord);
238 // send submitted content for plagiarism detection
239 $fs = get_file_storage();
240 $files = $fs->get_area_files($workshop->context
->id
, 'mod_workshop', 'submission_attachment', $submission->id
);
242 $params['other']['content'] = $formdata->content
;
243 $params['other']['pathnamehashes'] = array_keys($files);
245 $event = \mod_workshop\event\assessable_uploaded
::create($params);
246 $event->set_legacy_logdata($logdata);
249 redirect($workshop->submission_url($formdata->id
));
253 // load the form to override grade and/or publish the submission and process the submitted data eventually
254 if (!$edit and ($canoverride or $canpublish)) {
257 'editablepublished' => $canpublish,
258 'overridablegrade' => $canoverride);
259 $feedbackform = $workshop->get_feedbackauthor_form($PAGE->url
, $submission, $options);
260 if ($data = $feedbackform->get_data()) {
261 $data = file_postupdate_standard_editor($data, 'feedbackauthor', array(), $workshop->context
);
262 $record = new stdclass();
263 $record->id
= $submission->id
;
265 $record->gradeover
= $workshop->raw_grade_value($data->gradeover
, $workshop->grade
);
266 $record->gradeoverby
= $USER->id
;
267 $record->feedbackauthor
= $data->feedbackauthor
;
268 $record->feedbackauthorformat
= $data->feedbackauthorformat
;
271 $record->published
= !empty($data->published
);
273 $DB->update_record('workshop_submissions', $record);
274 redirect($workshop->view_url());
278 $PAGE->set_title($workshop->name
);
279 $PAGE->set_heading($course->fullname
);
281 $PAGE->navbar
->add(get_string('mysubmission', 'workshop'), $workshop->submission_url(), navigation_node
::TYPE_CUSTOM
);
282 $PAGE->navbar
->add(get_string('editingsubmission', 'workshop'));
283 } elseif ($ownsubmission) {
284 $PAGE->navbar
->add(get_string('mysubmission', 'workshop'));
286 $PAGE->navbar
->add(get_string('submission', 'workshop'));
289 // Output starts here
290 $output = $PAGE->get_renderer('mod_workshop');
291 echo $output->header();
292 echo $output->heading(format_string($workshop->name
), 2);
294 // show instructions for submitting as thay may contain some list of questions and we need to know them
295 // while reading the submitted answer
296 if (trim($workshop->instructauthors
)) {
297 $instructions = file_rewrite_pluginfile_urls($workshop->instructauthors
, 'pluginfile.php', $PAGE->context
->id
,
298 'mod_workshop', 'instructauthors', 0, workshop
::instruction_editors_options($PAGE->context
));
299 print_collapsible_region_start('', 'workshop-viewlet-instructauthors', get_string('instructauthors', 'workshop'));
300 echo $output->box(format_text($instructions, $workshop->instructauthorsformat
, array('overflowdiv'=>true)), array('generalbox', 'instructions'));
301 print_collapsible_region_end();
304 // if in edit mode, display the form to edit the submission
307 if (!empty($CFG->enableplagiarism
)) {
308 require_once($CFG->libdir
.'/plagiarismlib.php');
309 echo plagiarism_print_disclosure($cm->id
);
312 echo $output->footer();
316 // else display the submission
318 if ($submission->id
) {
319 if ($seenaspublished) {
320 $showauthor = has_capability('mod/workshop:viewauthorpublished', $workshop->context
);
322 $showauthor = has_capability('mod/workshop:viewauthornames', $workshop->context
);
324 echo $output->render($workshop->prepare_submission($submission, $showauthor));
326 echo $output->box(get_string('noyoursubmission', 'workshop'));
330 if ($submission->id
) {
331 $btnurl = new moodle_url($PAGE->url
, array('edit' => 'on', 'id' => $submission->id
));
332 $btntxt = get_string('editsubmission', 'workshop');
334 $btnurl = new moodle_url($PAGE->url
, array('edit' => 'on'));
335 $btntxt = get_string('createsubmission', 'workshop');
337 echo $output->single_button($btnurl, $btntxt, 'get');
340 if ($submission->id
and !$edit and !$isreviewer and $canallocate and $workshop->assessing_allowed($USER->id
)) {
341 $url = new moodle_url($PAGE->url
, array('assess' => 1));
342 echo $output->single_button($url, get_string('assess', 'workshop'), 'post');
345 if (($workshop->phase
== workshop
::PHASE_CLOSED
) and ($ownsubmission or $canviewall)) {
346 if (!empty($submission->gradeoverby
) and strlen(trim($submission->feedbackauthor
)) > 0) {
347 echo $output->render(new workshop_feedback_author($submission));
351 // and possibly display the submission's review(s)
354 // user's own assessment
355 $strategy = $workshop->grading_strategy_instance();
356 $mform = $strategy->get_assessment_form($PAGE->url
, 'assessment', $userassessment, false);
358 'showreviewer' => true,
359 'showauthor' => $showauthor,
360 'showform' => !is_null($userassessment->grade
),
361 'showweight' => true,
363 $assessment = $workshop->prepare_assessment($userassessment, $mform, $options);
364 $assessment->title
= get_string('assessmentbyyourself', 'workshop');
366 if ($workshop->assessing_allowed($USER->id
)) {
367 if (is_null($userassessment->grade
)) {
368 $assessment->add_action($workshop->assess_url($assessment->id
), get_string('assess', 'workshop'));
370 $assessment->add_action($workshop->assess_url($assessment->id
), get_string('reassess', 'workshop'));
374 $assessment->add_action($workshop->assess_url($assessment->id
), get_string('assessmentsettings', 'workshop'));
377 echo $output->render($assessment);
379 if ($workshop->phase
== workshop
::PHASE_CLOSED
) {
380 if (strlen(trim($userassessment->feedbackreviewer
)) > 0) {
381 echo $output->render(new workshop_feedback_reviewer($userassessment));
386 if (has_capability('mod/workshop:viewallassessments', $workshop->context
) or ($ownsubmission and $workshop->assessments_available())) {
388 $strategy = $workshop->grading_strategy_instance();
389 $assessments = $workshop->get_assessments_of_submission($submission->id
);
390 $showreviewer = has_capability('mod/workshop:viewreviewernames', $workshop->context
);
391 foreach ($assessments as $assessment) {
392 if ($assessment->reviewerid
== $USER->id
) {
393 // own assessment has been displayed already
396 if (is_null($assessment->grade
) and !has_capability('mod/workshop:viewallassessments', $workshop->context
)) {
397 // students do not see peer-assessment that are not graded yet
400 $mform = $strategy->get_assessment_form($PAGE->url
, 'assessment', $assessment, false);
402 'showreviewer' => $showreviewer,
403 'showauthor' => $showauthor,
404 'showform' => !is_null($assessment->grade
),
405 'showweight' => true,
407 $displayassessment = $workshop->prepare_assessment($assessment, $mform, $options);
409 $displayassessment->add_action($workshop->assess_url($assessment->id
), get_string('assessmentsettings', 'workshop'));
411 echo $output->render($displayassessment);
413 if ($workshop->phase
== workshop
::PHASE_CLOSED
and has_capability('mod/workshop:viewallassessments', $workshop->context
)) {
414 if (strlen(trim($assessment->feedbackreviewer
)) > 0) {
415 echo $output->render(new workshop_feedback_reviewer($assessment));
421 if (!$edit and $canoverride) {
422 // display a form to override the submission grade
423 $feedbackform->display();
426 echo $output->footer();