Note:

If you want to create a new page for developers, you should create it on the Moodle Developer Resource site.

Activity completion API

From MoodleDocs

Moodle 2.0


Modules do not need to be changed to support conditional availability, but they do need changing to support the completion system.

If you make no changes to a module whatsoever, it can only support 'manual' completion (where the user ticks a box).

Feature support

To support the completion system, your module must include a modulename_supports function in its lib.php. Here is an example:

/**
 * Indicates API features that the forum supports.
 *
 * @param string $feature
 * @return mixed True if yes (some features may use other values)
 */
function forum_supports($feature) {
    switch($feature) {
        case FEATURE_COMPLETION_TRACKS_VIEWS: return true;
        case FEATURE_COMPLETION_HAS_RULES: return true;
        default: return null;
    }
}

The relevant features for completion are:

  • FEATURE_COMPLETION_TRACKS_VIEWS - the module can support completion 'on view', meaning that an activity becomes marked complete as soon as a user clicks on it.
  • FEATURE_GRADE_HAS_GRADE - the module provides (or may provide, depending on settings) a grade for students. When a module supports grades, it can support completion 'on grade', meaning that an activity becomes marked complete as soon as a user is assigned a grade.
  • FEATURE_COMPLETION_HAS_RULES - the module has custom completion rules.

Completion on view

Completion on view means that, if selected, an activity is marked complete as soon as the user views it. 'View' is usually defined as seeing the module's main page; if you click on the activity, and there isn't an error, you have probably viewed it. However it is up to each module precisely how they define 'view'.

How to implement

In your module's _supports function, return true for FEATURE_COMPLETION_TRACKS_VIEWS.

Then add this code to run whenever a user successfully views the activity (for example, near the print_footer() call in view.php):

$completion=new completion_info($course);
$completion->set_module_viewed($cm);

Performance issues

Calling this method has no significant performance cost if 'on view' completion is not enabled for the activity. If it is enabled, then the performance cost is kept low because the 'viewed' state is cached; it doesn't add a database query to every request.

Completion on grade

Completion on grade means that, if selected, an activity is marked complete as soon as the user receives a grade from that activity.

How to implement

In your module's _supports function, return true for FEATURE_GRADE_HAS_GRADE. No other action is necessary.

Performance issues

When 'on grade' completion is enabled, there will be some additional database queries after a grade is assigned or changed. Unless your activity changes grades very frequently, this is unlikely to be an issue.

Custom completion rules

Custom completion rules allow for module-specific conditions. For example, the forum has custom rules so that you can make it mark a user completed when they make a certain number of posts to the forum.

It is a lot harder to implement custom completion rules than it is to use the system-provided 'view' or 'grade' conditions, but the instructions below should help make it clear.

Implementation overview

To implement custom completion rules, you need to:

  1. Return true for FEATURE_COMPLETION_HAS_RULES in your module's _supports function.
  2. Add database fields to your module's main table to store the custom completion settings.
  3. Add controls to your module's settings form so that users can select the custom rules, altering these database settings.
  4. Add a function that checks the value of these rules (if set).
  5. Add code so that whenever the value affecting a rule might change, you inform the completion system.

Database fields for completion settings

When you provide a custom completion rule for a module, that rule requires data to be stored with each module instance: whether the rule is enabled for that instance, and any options that apply to the rule.

Usually the best place to store this information is your module's main table because:

  • The information in the relevant row of this table is likely to be available in most parts of your code, so code changes are minimised.
  • You already read this row with most requests, so there is no need for additional database queries which would reduce performance.
  • The main table is used for most other module options so it is a logical place for this information.

If you are adding a basic completion condition you probably only need to add one field. To add a field to an existing module, you need to change the install.xml and then the update.php in the same way as adding any other field.

Example

Throughout this section I am using the forum as an example. The forum provides three completion options but because they all behave the same way, I am only showing one of them.

The forum adds this field to store a completion option:

  • completionposts - this may be 0 or an integer. If it's an integer, say 3, then the user needs to add 3 forum posts (either new discussions or replies) in order for the forum to count as 'completed'.

Form changes for completion settings

When you have custom completion conditions, you need to add controls to your module's settings form mod_form.php so that users can select these conditions. You can add any necessary controls.

  • Implement the add_completion_rules function which adds the form controls for your new rules.
  • Implement the completion_rule_enabled function which is called during form validation to check whether one of your module's completion rules has been selected.
  • Implement other form changes if necessary to set up the form with your data. If your data is in the form of simple text boxes or dropdowns then this is not necessary, but you might want to have a checkbox that enables the rule with a separate control to set its value. This needs form tweaks.

Example

The forum offers a checkbox with a text input box beside it. You tick the checkbox to enable the rule, then type in the desired number of posts.

First, the function that adds these controls:

function add_completion_rules() {
    $mform =& $this->_form;

    $group=array();
    $group[] =& $mform->createElement('checkbox', 'completionpostsenabled', , get_string('completionposts','forum'));
    $group[] =& $mform->createElement('text', 'completionposts', , array('size'=>3));
    $mform->setType('completionposts',PARAM_INT);
    $mform->addGroup($group, 'completionpostsgroup', get_string('completionpostsgroup','forum'), array(' '), false);
    $mform->setHelpButton('completionpostsgroup', array('completion', get_string('completionpostshelp', 'forum'), 'forum'));
    $mform->disabledIf('completionposts','completionpostsenabled','notchecked');

    return array('completionpostsgroup');
}
  • The function creates a checkbox and a text input field, which is set to accept only numbers.
  • These are grouped together so they appear on the same line, and we add a help button.
  • The text input field is disabled if the checkbox isn't ticked.
  • Note that this function must return the top-level element associated with the completion rule. (If there are multiple elements, you can return more than one.)
    • This is used so that your controls become disabled if automatic completion is not selected.

Next, a function for checking whether the user selected this option:

function completion_rule_enabled($data) {
    return (!empty($data['completionpostsenabled']) && $data['completionposts']!=0);
}
  • The custom completion rule is enabled if the 'enabled' checkbox is ticked and the text field value is something other than zero.
    • This is used to give an error if the user selects automatic completion, but fails to select any conditions.

That's all the 'required' functions, but we need to add some extra code to support the checkbox behaviour. I overrode get_data so that if there is a value in the edit field, but the checkbox is not ticked, the value counts as zero (the rule will not be enabled):

function get_data() {
    $data=parent::get_data();
    if(!$data) {
        return false;
    }
    // Turn off completion settings if the checkboxes aren't ticked
    $autocompletion=!empty($data->completion) && $data->completion==COMPLETION_TRACKING_AUTOMATIC;
    if(empty($data->completionpostsenabled) || !$autocompletion) {
        $data->completionposts=0;
    }
    return $data;
}

Finally, forum already had a data_preprocessing function but I added code to this to set up the checkboxes when the form is displayed, and to make the default value of the text fields 1 instead of 0:

function data_preprocessing(&$default_values){
    // [Existing code, not shown]

    // Set up the completion checkboxes which aren't part of standard data.
    // We also make the default value (if you turn on the checkbox) for those
    // numbers to be 1, this will not apply unless checkbox is ticked.
    $default_values['completionpostsenabled']=
        !empty($default_values['completionposts']) ? 1 : 0;
    if(empty($default_values['completionposts'])) {
        $default_values['completionposts']=1;
    }
}

Phew! That's the form done.

Completion state function

When you create completion conditions, you need to write a function module_get_completion_state that checks the value of those conditions for a particular user.

The function receives as parameters $course, $cm, and $userid - all self-explanatory, I hope - and $type. This has two values:

  • COMPLETION_AND - if multiple conditions are selected, the user must meet all of them.
  • COMPLETION_OR (not currently used) - if multiple conditions are selected, any one of them is good enough to complete the activity.

Your function should return:

  • true if your custom completion options are enabled and the user meets the conditions.
  • false if your custom completion options are enabled but the user does not yet meet the conditions.
  • $type (not false!) if none of your custom completion options are not enabled.

Example

Here's the function for forum (simplified to include only the one completion option):

/**
 * Obtains the automatic completion state for this forum based on any conditions
 * in forum settings.
 *
 * @param object $course Course
 * @param object $cm Course-module
 * @param int $userid User ID
 * @param bool $type Type of comparison (or/and; can be used as return value if no conditions)
 * @return bool True if completed, false if not, $type if conditions not set.
 */
function forum_get_completion_state($course,$cm,$userid,$type) {
    global $CFG,$DB;

    // Get forum details
    if(!($forum=$DB->get_record('forum',array('id'=>$cm->instance)))) {
        throw new Exception("Can't find forum {$cm->instance}");
    }
    // If completion option is enabled, evaluate it and return true/false 
    if($forum->completionposts) {
        return $forum->completionposts <= $DB->get_field_sql("
SELECT 
    COUNT(1) 
FROM 
    {forum_posts} fp 
    INNER JOIN {forum_discussions} fd ON fp.discussion=fd.id
WHERE
    fp.userid=:userid AND fd.forum=:forumid",
            array('userid'=>$userid,'forumid'=>$forum->id));
    } else {
        // Completion option is not enabled so just return $type
        return $type;
    }
}

Notifying the completion system