Search moodle.org's
Developer Documentation

See Release Notes

  • Bug fixes for general core bugs in 4.3.x will end 7 October 2024 (12 months).
  • Bug fixes for security issues in 4.3.x will end 21 April 2025 (18 months).
  • PHP version: minimum PHP 8.0.0 Note: minimum PHP version has increased since Moodle 4.1. PHP 8.2.x is supported too.

Differences Between: [Versions 310 and 403] [Versions 311 and 403] [Versions 39 and 403] [Versions 400 and 403] [Versions 401 and 403]

   1  <?php
   2  // This file is part of Moodle - http://moodle.org/
   3  //
   4  // Moodle is free software: you can redistribute it and/or modify
   5  // it under the terms of the GNU General Public License as published by
   6  // the Free Software Foundation, either version 3 of the License, or
   7  // (at your option) any later version.
   8  //
   9  // Moodle is distributed in the hope that it will be useful,
  10  // but WITHOUT ANY WARRANTY; without even the implied warranty of
  11  // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  12  // GNU General Public License for more details.
  13  //
  14  // You should have received a copy of the GNU General Public License
  15  // along with Moodle.  If not, see <http://www.gnu.org/licenses/>.
  16  
  17  /**
  18   * Course completion status for a particular user/course
  19   *
  20   * @package core_completion
  21   * @category completion
  22   * @copyright 2009 Catalyst IT Ltd
  23   * @author Aaron Barnes <aaronb@catalyst.net.nz>
  24   * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
  25   */
  26  
  27  defined('MOODLE_INTERNAL') || die();
  28  require_once($CFG->dirroot.'/completion/data_object.php');
  29  
  30  /**
  31   * Course completion status for a particular user/course
  32   *
  33   * @package core_completion
  34   * @category completion
  35   * @copyright 2009 Catalyst IT Ltd
  36   * @author Aaron Barnes <aaronb@catalyst.net.nz>
  37   * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
  38   */
  39  class completion_completion extends data_object {
  40  
  41      /* @var string $table Database table name that stores completion information */
  42      public $table = 'course_completions';
  43  
  44      /* @var array $required_fields Array of required table fields, must start with 'id'. */
  45      public $required_fields = array('id', 'userid', 'course',
  46          'timeenrolled', 'timestarted', 'timecompleted', 'reaggregate');
  47  
  48      /* @var int $userid User ID */
  49      public $userid;
  50  
  51      /* @var int $course Course ID */
  52      public $course;
  53  
  54      /* @var int Time of course enrolment {@link completion_completion::mark_enrolled()} */
  55      public $timeenrolled;
  56  
  57      /**
  58       * Time the user started their course completion {@link completion_completion::mark_inprogress()}
  59       * @var int
  60       */
  61      public $timestarted;
  62  
  63      /* @var int Timestamp of course completion {@link completion_completion::mark_complete()} */
  64      public $timecompleted;
  65  
  66      /* @var int Flag to trigger cron aggregation (timestamp) */
  67      public $reaggregate;
  68  
  69      /** @var float user's course grade. */
  70      public $gradefinal;
  71  
  72      /**
  73       * Finds and returns a data_object instance based on params.
  74       *
  75       * @param array $params associative arrays varname = >value
  76       * @return data_object instance of data_object or false if none found.
  77       */
  78      public static function fetch($params) {
  79          $cache = cache::make('core', 'coursecompletion');
  80  
  81          $key = $params['userid'] . '_' . $params['course'];
  82          if ($hit = $cache->get($key)) {
  83              return $hit['value'];
  84          }
  85  
  86          $tocache = self::fetch_helper('course_completions', __CLASS__, $params);
  87          $cache->set($key, ['value' => $tocache]);
  88          return $tocache;
  89      }
  90  
  91      /**
  92       * Return status of this completion
  93       *
  94       * @return bool
  95       */
  96      public function is_complete() {
  97          return (bool) $this->timecompleted;
  98      }
  99  
 100      /**
 101       * Mark this user as started (or enrolled) in this course
 102       *
 103       * If the user is already marked as started, no change will occur
 104       *
 105       * @param integer $timeenrolled Time enrolled (optional)
 106       * @return  int|null id of completion record on successful update.
 107       */
 108      public function mark_enrolled($timeenrolled = null) {
 109  
 110          if ($this->timeenrolled === null) {
 111  
 112              if ($timeenrolled === null) {
 113                  $timeenrolled = time();
 114              }
 115  
 116              $this->timeenrolled = $timeenrolled;
 117          }
 118  
 119          return $this->_save();
 120      }
 121  
 122      /**
 123       * Mark this user as inprogress in this course
 124       *
 125       * If the user is already marked as inprogress, the time will not be changed
 126       *
 127       * @param integer $timestarted Time started (optional)
 128       * @return  int|null id of completion record on successful update.
 129       */
 130      public function mark_inprogress($timestarted = null) {
 131  
 132          $timenow = time();
 133  
 134          // Set reaggregate flag
 135          $this->reaggregate = $timenow;
 136  
 137          if (!$this->timestarted) {
 138  
 139              if (!$timestarted) {
 140                  $timestarted = $timenow;
 141              }
 142  
 143              $this->timestarted = $timestarted;
 144          }
 145  
 146          return $this->_save();
 147      }
 148  
 149      /**
 150       * Mark this user complete in this course
 151       *
 152       * This generally happens when the required completion criteria
 153       * in the course are complete.
 154       *
 155       * @param integer $timecomplete Time completed (optional)
 156       * @return  int|null id of completion record on successful update.
 157       */
 158      public function mark_complete($timecomplete = null) {
 159          global $USER;
 160  
 161          // Never change a completion time.
 162          if ($this->timecompleted) {
 163              return null;
 164          }
 165  
 166          // Use current time if nothing supplied.
 167          if (!$timecomplete) {
 168              $timecomplete = time();
 169          }
 170  
 171          // Set time complete.
 172          $this->timecompleted = $timecomplete;
 173          // Save record.
 174          if ($result = $this->_save()) {
 175              $data = $this->get_record_data();
 176              \core\event\course_completed::create_from_completion($data)->trigger();
 177          }
 178  
 179          // Notify user.
 180          $course = get_course($data->course);
 181          $messagesubject = get_string('coursecompleted', 'completion');
 182          $a = [
 183              'coursename' => get_course_display_name_for_list($course),
 184              'courselink' => (string) new moodle_url('/course/view.php', array('id' => $course->id)),
 185          ];
 186          $messagebody = get_string('coursecompletedmessage', 'completion', $a);
 187          $messageplaintext = html_to_text($messagebody);
 188  
 189          $eventdata = new \core\message\message();
 190          $eventdata->courseid          = $course->id;
 191          $eventdata->component         = 'moodle';
 192          $eventdata->name              = 'coursecompleted';
 193          $eventdata->userfrom          = core_user::get_noreply_user();
 194          $eventdata->userto            = $data->userid;
 195          $eventdata->notification      = 1;
 196          $eventdata->subject           = $messagesubject;
 197          $eventdata->fullmessage       = $messageplaintext;
 198          $eventdata->fullmessageformat = FORMAT_HTML;
 199          $eventdata->fullmessagehtml   = $messagebody;
 200          $eventdata->smallmessage      = $messageplaintext;
 201  
 202          if ($courseimage = \core_course\external\course_summary_exporter::get_course_image($course)) {
 203              $eventdata->customdata  = [
 204                  'notificationpictureurl' => $courseimage,
 205              ];
 206          }
 207          message_send($eventdata);
 208  
 209          return $result;
 210      }
 211  
 212      /**
 213       * Save course completion status
 214       *
 215       * This method creates a course_completions record if none exists
 216       * @access  private
 217       * @return  int|null id of completion record on successful update.
 218       */
 219      private function _save() {
 220          if ($this->timeenrolled === null) {
 221              $this->timeenrolled = 0;
 222          }
 223  
 224          // Save record
 225          if (isset($this->id)) {
 226              $success = $this->update();
 227          } else {
 228              // Make sure reaggregate field is not null
 229              if (!$this->reaggregate) {
 230                  $this->reaggregate = 0;
 231              }
 232  
 233              // Make sure timestarted is not null
 234              if (!$this->timestarted) {
 235                  $this->timestarted = 0;
 236              }
 237  
 238              $success = $this->insert();
 239          }
 240  
 241          if ($success) {
 242              // Update the cached record.
 243              $cache = cache::make('core', 'coursecompletion');
 244              $data = $this->get_record_data();
 245              $key = $data->userid . '_' . $data->course;
 246              $cache->set($key, ['value' => $data]);
 247              return $this->id;
 248          }
 249  
 250          return null;
 251      }
 252  }