3 * Standard library of functions and constants for lesson
6 * @license http://www.gnu.org/copyleft/gpl.html GNU Public License
10 define("LESSON_MAX_EVENT_LENGTH", "432000"); // 5 days maximum
13 * Given an object containing all the necessary data,
14 * (defined by the form in mod_form.php) this function
15 * will create a new instance and return the id number
16 * of the new instance.
18 * @param object $lesson Lesson post data from the form
21 function lesson_add_instance($lesson) {
24 lesson_process_pre_save($lesson);
26 if (!$lesson->id
= insert_record("lesson", $lesson)) {
30 lesson_process_post_save($lesson);
32 lesson_grade_item_update(stripslashes_recursive($lesson));
38 * Given an object containing all the necessary data,
39 * (defined by the form in mod_form.php) this function
40 * will update an existing instance with new data.
42 * @param object $lesson Lesson post data from the form
45 function lesson_update_instance($lesson) {
47 $lesson->id
= $lesson->instance
;
49 lesson_process_pre_save($lesson);
51 if (!$result = update_record("lesson", $lesson)) {
52 return false; // Awe man!
55 lesson_process_post_save($lesson);
57 // update grade item definition
58 lesson_grade_item_update(stripslashes_recursive($lesson));
60 // update grades - TODO: do it only when grading style changes
61 lesson_update_grades(stripslashes_recursive($lesson), 0, false);
67 /*******************************************************************/
68 function lesson_delete_instance($id) {
69 /// Given an ID of an instance of this module,
70 /// this function will permanently delete the instance
71 /// and any data that depends on it.
73 if (! $lesson = get_record("lesson", "id", "$id")) {
79 if (! delete_records("lesson", "id", "$lesson->id")) {
82 if (! delete_records("lesson_pages", "lessonid", "$lesson->id")) {
85 if (! delete_records("lesson_answers", "lessonid", "$lesson->id")) {
88 if (! delete_records("lesson_attempts", "lessonid", "$lesson->id")) {
91 if (! delete_records("lesson_grades", "lessonid", "$lesson->id")) {
94 if (! delete_records("lesson_timer", "lessonid", "$lesson->id")) {
97 if (! delete_records("lesson_branch", "lessonid", "$lesson->id")) {
100 if (! delete_records("lesson_high_scores", "lessonid", "$lesson->id")) {
103 if ($events = get_records_select('event', "modulename = 'lesson' and instance = '$lesson->id'")) {
104 foreach($events as $event) {
105 delete_event($event->id
);
108 $pagetypes = page_import_types('mod/lesson/');
109 foreach ($pagetypes as $pagetype) {
110 if (!delete_records('block_instance', 'pageid', $lesson->id
, 'pagetype', $pagetype)) {
115 lesson_grade_item_delete($lesson);
121 * Given a course object, this function will clean up anything that
122 * would be leftover after all the instances were deleted.
124 * As of now, this function just cleans the lesson_default table
126 * @param object $course an object representing the course that is being deleted
127 * @param boolean $feedback to specify if the process must output a summary of its work
130 function lesson_delete_course($course, $feedback=true) {
132 $count = count_records('lesson_default', 'course', $course->id
);
133 delete_records('lesson_default', 'course', $course->id
);
135 //Inform about changes performed if feedback is enabled
137 notify(get_string('deletedefaults', 'lesson', $count));
143 /*******************************************************************/
144 function lesson_user_outline($course, $user, $mod, $lesson) {
145 /// Return a small object with summary information about what a
146 /// user has done with a given particular instance of this module
147 /// Used for user activity reports.
148 /// $return->time = the time they did it
149 /// $return->info = a short text description
151 if ($grades = get_records_select("lesson_grades", "lessonid = $lesson->id AND userid = $user->id",
153 foreach ($grades as $grade) {
154 $max_grade = number_format($grade->grade
* $lesson->grade
/ 100.0, 1);
157 $return->time
= $grade->completed
;
158 if ($lesson->retake
) {
159 $return->info
= get_string("gradeis", "lesson", $max_grade)." (".
160 get_string("attempt", "lesson", count($grades)).")";
162 $return->info
= get_string("gradeis", "lesson", $max_grade);
165 $return->info
= get_string("no")." ".get_string("attempts", "lesson");
170 /*******************************************************************/
171 function lesson_user_complete($course, $user, $mod, $lesson) {
172 /// Print a detailed representation of what a user has done with
173 /// a given particular instance of this module, for user activity reports.
175 if ($attempts = get_records_select("lesson_attempts", "lessonid = $lesson->id AND userid = $user->id",
176 "retry, timeseen")) {
177 print_simple_box_start();
178 $table->head
= array (get_string("attempt", "lesson"), get_string("numberofpagesviewed", "lesson"),
179 get_string("numberofcorrectanswers", "lesson"), get_string("time"));
180 $table->width
= "100%";
181 $table->align
= array ("center", "center", "center", "center");
182 $table->size
= array ("*", "*", "*", "*");
183 $table->cellpadding
= 2;
184 $table->cellspacing
= 0;
190 foreach ($attempts as $attempt) {
191 if ($attempt->retry
== $retry) {
193 if ($attempt->correct
) {
196 $timeseen = $attempt->timeseen
;
198 $table->data
[] = array($retry +
1, $npages, $ncorrect, userdate($timeseen));
201 if ($attempt->correct
) {
209 $table->data
[] = array($retry +
1, $npages, $ncorrect, userdate($timeseen));
212 print_simple_box_end();
213 // also print grade summary
214 if ($grades = get_records_select("lesson_grades", "lessonid = $lesson->id AND userid = $user->id",
216 foreach ($grades as $grade) {
217 $max_grade = number_format($grade->grade
* $lesson->grade
/ 100.0, 1);
220 if ($lesson->retake
) {
221 echo "<p>".get_string("gradeis", "lesson", $max_grade)." (".
222 get_string("attempts", "lesson").": ".count($grades).")</p>";
224 echo "<p>".get_string("gradeis", "lesson", $max_grade)."</p>";
228 echo get_string("no")." ".get_string("attempts", "lesson");
235 /*******************************************************************/
236 function lesson_print_recent_activity($course, $isteacher, $timestart) {
237 /// Given a course and a time, this module should find recent activity
238 /// that has occurred in lesson activities and print it out.
239 /// Return true if there was output, or false is there was none.
243 return false; // True if anything was printed, otherwise false
247 * Prints lesson summaries on MyMoodle Page
249 * Prints lesson name, due date and attempt information on
250 * lessons that have a deadline that has not already passed
251 * and it is available for taking.
253 * @param array $courses An array of course objects to get lesson instances from
254 * @param array $htmlarray Store overview output array( course ID => 'lesson' => HTML output )
256 function lesson_print_overview($courses, &$htmlarray) {
259 if (!$lessons = get_all_instances_in_courses('lesson', $courses)) {
263 /// Get Necessary Strings
264 $strlesson = get_string('modulename', 'lesson');
265 $strnotattempted = get_string('nolessonattempts', 'lesson');
266 $strattempted = get_string('lessonattempted', 'lesson');
269 foreach ($lessons as $lesson) {
270 if ($lesson->deadline
!= 0 // The lesson has a deadline
271 and $lesson->deadline
>= $now // And it is before the deadline has been met
272 and ($lesson->available
== 0 or $lesson->available
<= $now)) { // And the lesson is available
275 if (!$lesson->visible
) {
276 $class = ' class="dimmed"';
280 $str = print_box("$strlesson: <a$class href=\"$CFG->wwwroot/mod/lesson/view.php?id=$lesson->coursemodule\">".
281 format_string($lesson->name
).'</a>', 'name', '', true);
284 $str .= print_box(get_string('lessoncloseson', 'lesson', userdate($lesson->deadline
)), 'info', '', true);
286 // Attempt information
287 if (has_capability('mod/lesson:manage', get_context_instance(CONTEXT_MODULE
, $lesson->coursemodule
))) {
288 // Number of user attempts
289 $attempts = count_records('lesson_attempts', 'lessonid', $lesson->id
);
290 $str .= print_box(get_string('xattempts', 'lesson', $attempts), 'info', '', true);
292 // Determine if the user has attempted the lesson or not
293 if (count_records('lesson_attempts', 'lessonid', $lesson->id
, 'userid', $USER->id
)) {
294 $str .= print_box($strattempted, 'info', '', true);
296 $str .= print_box($strnotattempted, 'info', '', true);
299 $str = print_box($str, 'lesson overview', '', true);
301 if (empty($htmlarray[$lesson->course
]['lesson'])) {
302 $htmlarray[$lesson->course
]['lesson'] = $str;
304 $htmlarray[$lesson->course
]['lesson'] .= $str;
310 /*******************************************************************/
311 function lesson_cron () {
312 /// Function to be run periodically according to the moodle cron
313 /// This function searches for things that need to be done, such
314 /// as sending out mail, toggling flags etc ...
322 * Return grade for given user or all users.
324 * @param int $lessonid id of lesson
325 * @param int $userid optional user id, 0 means all users
326 * @return array array of grades, false if none
328 function lesson_get_user_grades($lesson, $userid=0) {
331 $user = $userid ?
"AND u.id = $userid" : "";
332 $fuser = $userid ?
"AND uu.id = $userid" : "";
334 if ($lesson->retake
) {
335 if ($lesson->usemaxgrade
) {
336 $sql = "SELECT u.id, u.id AS userid, MAX(g.grade) AS rawgrade
337 FROM {$CFG->prefix}user u, {$CFG->prefix}lesson_grades g
338 WHERE u.id = g.userid AND g.lessonid = $lesson->id
342 $sql = "SELECT u.id, u.id AS userid, AVG(g.grade) AS rawgrade
343 FROM {$CFG->prefix}user u, {$CFG->prefix}lesson_grades g
344 WHERE u.id = g.userid AND g.lessonid = $lesson->id
350 // use only first attempts (with lowest id in lesson_grades table)
351 $firstonly = "SELECT uu.id AS userid, MIN(gg.id) AS firstcompleted
352 FROM {$CFG->prefix}user uu, {$CFG->prefix}lesson_grades gg
353 WHERE uu.id = gg.userid AND gg.lessonid = $lesson->id
357 $sql = "SELECT u.id, u.id AS userid, g.grade AS rawgrade
358 FROM {$CFG->prefix}user u, {$CFG->prefix}lesson_grades g, ($firstonly) f
359 WHERE u.id = g.userid AND g.lessonid = $lesson->id
360 AND g.id = f.firstcompleted AND g.userid=f.userid
364 return get_records_sql($sql);
368 * Update grades in central gradebook
370 * @param object $lesson null means all lessons
371 * @param int $userid specific user only, 0 mean all
373 function lesson_update_grades($lesson=null, $userid=0, $nullifnone=true) {
375 if (!function_exists('grade_update')) { //workaround for buggy PHP versions
376 require_once($CFG->libdir
.'/gradelib.php');
379 if ($lesson != null) {
380 if ($grades = lesson_get_user_grades($lesson, $userid)) {
381 grade_update('mod/lesson', $lesson->course
, 'mod', 'lesson', $lesson->id
, 0, $grades);
383 } else if ($userid and $nullifnone) {
384 $grade = new object();
385 $grade->itemid
= $lesson->id
;
386 $grade->userid
= $userid;
387 $grade->rawgrade
= NULL;
388 grade_update('mod/lesson', $lesson->course
, 'mod', 'lesson', $lesson->id
, 0, $grade);
392 $sql = "SELECT l.*, cm.idnumber as cmidnumber, l.course as courseid
393 FROM {$CFG->prefix}lesson l, {$CFG->prefix}course_modules cm, {$CFG->prefix}modules m
394 WHERE m.name='lesson' AND m.id=cm.module AND cm.instance=l.id";
395 if ($rs = get_recordset_sql($sql)) {
396 if ($rs->RecordCount() > 0) {
397 while ($lesson = rs_fetch_next_record($rs)) {
398 lesson_grade_item_update($lesson);
399 if ($lesson->grade
!= 0) {
400 lesson_update_grades($lesson, 0, false);
410 * Create grade item for given lesson
412 * @param object $lesson object with extra cmidnumber
413 * @return int 0 if ok, error code otherwise
415 function lesson_grade_item_update($lesson) {
417 if (!function_exists('grade_update')) { //workaround for buggy PHP versions
418 require_once($CFG->libdir
.'/gradelib.php');
421 if (array_key_exists('cmidnumber', $lesson)) { //it may not be always present
422 $params = array('itemname'=>$lesson->name
, 'idnumber'=>$lesson->cmidnumber
);
424 $params = array('itemname'=>$lesson->name
);
427 if ($lesson->grade
> 0) {
428 $params['gradetype'] = GRADE_TYPE_VALUE
;
429 $params['grademax'] = 100; //means 100%
430 $params['grademin'] = 0;
431 $params['multfactor'] = $lesson->grade
/ 100.0;
434 $params['gradetype'] = GRADE_TYPE_NONE
;
435 $params['multfactor'] = 1.0;
438 return grade_update('mod/lesson', $lesson->course
, 'mod', 'lesson', $lesson->id
, 0, NULL, $params);
442 * Delete grade item for given lesson
444 * @param object $lesson object
445 * @return object lesson
447 function lesson_grade_item_delete($lesson) {
449 require_once($CFG->libdir
.'/gradelib.php');
451 return grade_update('mod/lesson', $lesson->course
, 'mod', 'lesson', $lesson->id
, 0, NULL, array('deleted'=>1));
455 /*******************************************************************/
456 function lesson_get_participants($lessonid) {
457 //Must return an array of user records (all data) who are participants
458 //for a given instance of lesson. Must include every user involved
459 //in the instance, independient of his role (student, teacher, admin...)
464 $students = get_records_sql("SELECT DISTINCT u.id, u.id
465 FROM {$CFG->prefix}user u,
466 {$CFG->prefix}lesson_attempts a
467 WHERE a.lessonid = '$lessonid' and
470 //Return students array (it contains an array of unique users)
474 function lesson_get_view_actions() {
475 return array('view','view all');
478 function lesson_get_post_actions() {
479 return array('end','start', 'update grade attempt');
483 * Runs any processes that must run before
484 * a lesson insert/update
486 * @param object $lesson Lesson form data
489 function lesson_process_pre_save(&$lesson) {
490 $lesson->timemodified
= time();
492 if (empty($lesson->timed
)) {
495 if (empty($lesson->timespent
) or !is_numeric($lesson->timespent
) or $lesson->timespent
< 0) {
496 $lesson->timespent
= 0;
498 if (!isset($lesson->completed
)) {
499 $lesson->completed
= 0;
501 if (empty($lesson->gradebetterthan
) or !is_numeric($lesson->gradebetterthan
) or $lesson->gradebetterthan
< 0) {
502 $lesson->gradebetterthan
= 0;
503 } else if ($lesson->gradebetterthan
> 100) {
504 $lesson->gradebetterthan
= 100;
507 // Conditions for dependency
508 $conditions = new stdClass
;
509 $conditions->timespent
= $lesson->timespent
;
510 $conditions->completed
= $lesson->completed
;
511 $conditions->gradebetterthan
= $lesson->gradebetterthan
;
512 $lesson->conditions
= addslashes(serialize($conditions));
513 unset($lesson->timespent
);
514 unset($lesson->completed
);
515 unset($lesson->gradebetterthan
);
517 if (!empty($lesson->password
)) {
518 $lesson->password
= md5($lesson->password
);
520 unset($lesson->password
);
523 if ($lesson->lessondefault
) {
524 $default = new stdClass
;
525 $default = clone($lesson);
526 unset($default->name
);
527 unset($default->timemodified
);
528 unset($default->available
);
529 unset($default->deadline
);
530 if ($default->id
= get_field('lesson_default', 'id', 'course', $default->course
)) {
531 update_record('lesson_default', $default);
533 insert_record('lesson_default', $default);
536 unset($lesson->lessondefault
);
540 * Runs any processes that must be run
541 * after a lesson insert/update
543 * @param object $lesson Lesson form data
546 function lesson_process_post_save(&$lesson) {
547 if ($events = get_records_select('event', "modulename = 'lesson' and instance = '$lesson->id'")) {
548 foreach($events as $event) {
549 delete_event($event->id
);
553 $event = new stdClass
;
554 $event->description
= $lesson->name
;
555 $event->courseid
= $lesson->course
;
558 $event->modulename
= 'lesson';
559 $event->instance
= $lesson->id
;
560 $event->eventtype
= 'open';
561 $event->timestart
= $lesson->available
;
562 $event->visible
= instance_is_visible('lesson', $lesson);
563 $event->timeduration
= ($lesson->deadline
- $lesson->available
);
565 if ($lesson->deadline
and $lesson->available
and $event->timeduration
<= LESSON_MAX_EVENT_LENGTH
) {
566 // Single event for the whole lesson.
567 $event->name
= $lesson->name
;
570 // Separate start and end events.
571 $event->timeduration
= 0;
572 if ($lesson->available
) {
573 $event->name
= $lesson->name
.' ('.get_string('lessonopens', 'lesson').')';
575 unset($event->id
); // So we can use the same object for the close event.
577 if ($lesson->deadline
) {
578 $event->name
= $lesson->name
.' ('.get_string('lessoncloses', 'lesson').')';
579 $event->timestart
= $lesson->deadline
;
580 $event->eventtype
= 'close';