2011-09-28 23:33:21 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
// This file is part of Moodle - http://moodle.org/
|
|
|
|
//
|
|
|
|
// Moodle is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// Moodle is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Advanced grading methods support
|
|
|
|
*
|
|
|
|
* @package core
|
|
|
|
* @subpackage grading
|
|
|
|
* @copyright 2011 David Mudrak <david@moodle.com>
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
|
|
*/
|
|
|
|
|
|
|
|
defined('MOODLE_INTERNAL') || die();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Factory method returning an instance of the grading manager
|
|
|
|
*
|
2011-10-05 03:49:21 +02:00
|
|
|
* @param stdClass|int $context or $areaid
|
2011-09-28 23:33:21 +02:00
|
|
|
* @param string $component the frankenstyle name of the component
|
2011-10-02 22:24:40 +02:00
|
|
|
* @param string $area the name of the gradable area
|
2011-09-28 23:33:21 +02:00
|
|
|
* @return grading_manager
|
|
|
|
*/
|
2011-10-05 03:49:21 +02:00
|
|
|
function get_grading_manager($context_or_areaid = null, $component = null, $area = null) {
|
|
|
|
global $DB;
|
2011-09-28 23:33:21 +02:00
|
|
|
|
|
|
|
$manager = new grading_manager();
|
|
|
|
|
2011-10-05 03:49:21 +02:00
|
|
|
if (is_object($context_or_areaid)) {
|
|
|
|
$context = $context_or_areaid;
|
|
|
|
} else {
|
|
|
|
$context = null;
|
|
|
|
|
|
|
|
if (is_numeric($context_or_areaid)) {
|
|
|
|
$manager->load($context_or_areaid);
|
|
|
|
return $manager;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-28 23:33:21 +02:00
|
|
|
if (!is_null($context)) {
|
|
|
|
$manager->set_context($context);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_null($component)) {
|
|
|
|
$manager->set_component($component);
|
|
|
|
}
|
|
|
|
|
2011-10-02 22:24:40 +02:00
|
|
|
if (!is_null($area)) {
|
|
|
|
$manager->set_area($area);
|
2011-09-28 23:33:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $manager;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* General class providing access to common grading features
|
|
|
|
*
|
2011-10-03 21:57:16 +02:00
|
|
|
* Grading manager provides access to the particular grading method controller
|
|
|
|
* in that area.
|
|
|
|
*
|
2011-09-28 23:33:21 +02:00
|
|
|
* Fully initialized instance of the grading manager operates over a single
|
|
|
|
* gradable area. It is possible to work with a partially initialized manager
|
2011-10-02 22:24:40 +02:00
|
|
|
* that knows just context and component without known area, for example.
|
|
|
|
* It is also possible to change context, component and area of an existing
|
2011-09-28 23:33:21 +02:00
|
|
|
* manager. Such pattern is used when copying form definitions, for example.
|
|
|
|
*/
|
|
|
|
class grading_manager {
|
|
|
|
|
|
|
|
/** @var stdClass the context */
|
|
|
|
protected $context;
|
|
|
|
|
|
|
|
/** @var string the frankenstyle name of the component */
|
|
|
|
protected $component;
|
|
|
|
|
|
|
|
/** @var string the name of the gradable area */
|
2011-10-02 22:24:40 +02:00
|
|
|
protected $area;
|
2011-09-28 23:33:21 +02:00
|
|
|
|
2011-10-03 21:57:16 +02:00
|
|
|
/** @var stdClass|false|null the raw record from {grading_areas}, false if does not exist, null if invalidated cache */
|
|
|
|
private $areacache = null;
|
|
|
|
|
2011-09-28 23:33:21 +02:00
|
|
|
/**
|
|
|
|
* Sets the context the manager operates on
|
|
|
|
*
|
|
|
|
* @param stdClass $context
|
|
|
|
*/
|
|
|
|
public function set_context(stdClass $context) {
|
2011-10-03 21:57:16 +02:00
|
|
|
$this->areacache = null;
|
2011-09-28 23:33:21 +02:00
|
|
|
$this->context = $context;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the component the manager operates on
|
|
|
|
*
|
|
|
|
* @param string $component the frankenstyle name of the component
|
|
|
|
*/
|
|
|
|
public function set_component($component) {
|
2011-10-03 21:57:16 +02:00
|
|
|
$this->areacache = null;
|
|
|
|
list($type, $name) = normalize_component($component);
|
|
|
|
$this->component = $type.'_'.$name;
|
2011-09-28 23:33:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-10-02 22:24:40 +02:00
|
|
|
* Sets the area the manager operates on
|
2011-09-28 23:33:21 +02:00
|
|
|
*
|
2011-10-02 22:24:40 +02:00
|
|
|
* @param string $area the name of the gradable area
|
2011-09-28 23:33:21 +02:00
|
|
|
*/
|
2011-10-02 22:24:40 +02:00
|
|
|
public function set_area($area) {
|
2011-10-03 21:57:16 +02:00
|
|
|
$this->areacache = null;
|
2011-10-02 22:24:40 +02:00
|
|
|
$this->area = $area;
|
2011-09-28 23:33:21 +02:00
|
|
|
}
|
|
|
|
|
2011-10-05 03:49:21 +02:00
|
|
|
/**
|
|
|
|
* Loads the gradable area info from the database
|
|
|
|
*
|
|
|
|
* @param int $areaid
|
|
|
|
*/
|
|
|
|
public function load($areaid) {
|
|
|
|
global $DB;
|
|
|
|
|
|
|
|
$this->areacache = $DB->get_record('grading_areas', array('id' => $areaid), '*', MUST_EXIST);
|
|
|
|
$this->context = get_context_instance_by_id($this->areacache->contextid, MUST_EXIST);
|
|
|
|
$this->component = $this->areacache->component;
|
|
|
|
$this->area = $this->areacache->areaname;
|
|
|
|
}
|
|
|
|
|
2011-09-28 23:33:21 +02:00
|
|
|
/**
|
|
|
|
* Returns the list of available grading methods in the given context
|
|
|
|
*
|
|
|
|
* Basically this returns the list of installed grading plugins with an empty value
|
|
|
|
* for simple direct grading. In the future, the list of available methods may be
|
|
|
|
* controlled per-context.
|
|
|
|
*
|
|
|
|
* Requires the context property to be set in advance.
|
2011-10-02 21:05:01 +02:00
|
|
|
*
|
|
|
|
* @param bool $includenone should the 'Simple direct grading' be included
|
2011-09-28 23:33:21 +02:00
|
|
|
* @return array of the (string)name => (string)localized title of the method
|
|
|
|
*/
|
2011-10-02 21:05:01 +02:00
|
|
|
public function get_available_methods($includenone = true) {
|
2011-09-28 23:33:21 +02:00
|
|
|
|
|
|
|
$this->ensure_isset(array('context'));
|
|
|
|
|
2011-10-02 21:05:01 +02:00
|
|
|
if ($includenone) {
|
|
|
|
$list = array('' => get_string('gradingmethodnone', 'core_grading'));
|
|
|
|
} else {
|
|
|
|
$list = array();
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (get_plugin_list('gradingform') as $name => $location) {
|
|
|
|
$list[$name] = get_string('pluginname', 'gradingform_'.$name);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $list;
|
2011-09-28 23:33:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the list of gradable areas in the given context and component
|
|
|
|
*
|
|
|
|
* This performs a callback to the library of the relevant plugin to obtain
|
|
|
|
* the list of supported areas.
|
|
|
|
* @return array of (string)areacode => (string)localized title of the area
|
|
|
|
*/
|
|
|
|
public function get_available_areas() {
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
$this->ensure_isset(array('context', 'component'));
|
|
|
|
|
|
|
|
// example: if the given context+component lead to mod_assignment, this method
|
|
|
|
// will do something like
|
|
|
|
// require_once($CFG->dirroot.'/mod/assignment/lib.php');
|
|
|
|
// return assignment_gradable_area_list();
|
|
|
|
|
|
|
|
// todo - hardcoded list for now
|
2011-10-03 21:57:16 +02:00
|
|
|
return array('submission' => 'Submissions');
|
2011-09-28 23:33:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-10-02 22:24:40 +02:00
|
|
|
* Returns the currently active grading method in the gradable area
|
2011-09-28 23:33:21 +02:00
|
|
|
*
|
2011-10-02 23:31:05 +02:00
|
|
|
* @return string|null the name of the grading plugin of null if it has not been set
|
2011-09-28 23:33:21 +02:00
|
|
|
*/
|
2011-10-02 22:24:40 +02:00
|
|
|
public function get_active_method() {
|
2011-10-02 23:31:05 +02:00
|
|
|
global $DB;
|
|
|
|
|
|
|
|
$this->ensure_isset(array('context', 'component', 'area'));
|
|
|
|
|
|
|
|
// get the current grading area record if it exists
|
2011-10-03 21:57:16 +02:00
|
|
|
if (is_null($this->areacache)) {
|
|
|
|
$this->areacache = $DB->get_record('grading_areas', array(
|
|
|
|
'contextid' => $this->context->id,
|
|
|
|
'component' => $this->component,
|
|
|
|
'areaname' => $this->area),
|
|
|
|
'*', IGNORE_MISSING);
|
|
|
|
}
|
2011-10-02 23:31:05 +02:00
|
|
|
|
2011-10-03 21:57:16 +02:00
|
|
|
if ($this->areacache === false) {
|
2011-10-02 23:31:05 +02:00
|
|
|
// no area record yet
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2011-10-03 21:57:16 +02:00
|
|
|
return $this->areacache->activemethod;
|
2011-10-02 23:31:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the currently active grading method in the gradable area
|
|
|
|
*
|
|
|
|
* @param string $method the method name, eg 'rubric' (must be available)
|
|
|
|
*/
|
|
|
|
public function set_active_method($method) {
|
|
|
|
global $DB;
|
|
|
|
|
2011-10-02 22:24:40 +02:00
|
|
|
$this->ensure_isset(array('context', 'component', 'area'));
|
2011-10-02 23:31:05 +02:00
|
|
|
|
2011-10-05 03:50:12 +02:00
|
|
|
// make sure the passed method is empty or a valid plugin name
|
|
|
|
if (empty($method)) {
|
|
|
|
$method = null;
|
|
|
|
} else {
|
|
|
|
if ('gradingform_'.$method !== clean_param('gradingform_'.$method, PARAM_COMPONENT)) {
|
|
|
|
throw new moodle_exception('invalid_method_name', 'core_grading');
|
|
|
|
}
|
|
|
|
$available = $this->get_available_methods(false);
|
|
|
|
if (!array_key_exists($method, $available)) {
|
|
|
|
throw new moodle_exception('invalid_method_name', 'core_grading');
|
|
|
|
}
|
2011-10-02 23:31:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// get the current grading area record if it exists
|
2011-10-03 21:57:16 +02:00
|
|
|
if (is_null($this->areacache)) {
|
|
|
|
$this->areacache = $DB->get_record('grading_areas', array(
|
|
|
|
'contextid' => $this->context->id,
|
|
|
|
'component' => $this->component,
|
|
|
|
'areaname' => $this->area),
|
|
|
|
'*', IGNORE_MISSING);
|
|
|
|
}
|
2011-10-02 23:31:05 +02:00
|
|
|
|
2011-10-03 21:57:16 +02:00
|
|
|
if ($this->areacache === false) {
|
2011-10-02 23:31:05 +02:00
|
|
|
// no area record yet, create one with the active method set
|
|
|
|
$area = array(
|
|
|
|
'contextid' => $this->context->id,
|
|
|
|
'component' => $this->component,
|
|
|
|
'areaname' => $this->area,
|
|
|
|
'activemethod' => $method);
|
|
|
|
$DB->insert_record('grading_areas', $area);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// update the existing record if needed
|
2011-10-05 03:50:12 +02:00
|
|
|
if ($this->areacache->activemethod !== $method) {
|
2011-10-03 21:57:16 +02:00
|
|
|
$DB->set_field('grading_areas', 'activemethod', $method, array('id' => $this->areacache->id));
|
2011-10-02 23:31:05 +02:00
|
|
|
}
|
|
|
|
}
|
2011-10-03 21:57:16 +02:00
|
|
|
|
|
|
|
$this->areacache = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Extends the settings navigation with the grading settings
|
|
|
|
*
|
|
|
|
* This function is called when the context for the page is an activity module with the
|
|
|
|
* FEATURE_ADVANCED_GRADING and the user has the permission moodle/grade:managegradingforms.
|
|
|
|
*
|
|
|
|
* @param settings_navigation $settingsnav {@link settings_navigation}
|
|
|
|
* @param navigation_node $modulenode {@link navigation_node}
|
|
|
|
*/
|
|
|
|
public function extend_settings_navigation(settings_navigation $settingsnav, navigation_node $modulenode=null) {
|
|
|
|
global $PAGE, $CFG;
|
|
|
|
|
|
|
|
$this->ensure_isset(array('context', 'component'));
|
|
|
|
|
|
|
|
$areas = $this->get_available_areas();
|
|
|
|
|
|
|
|
if (empty($areas)) {
|
|
|
|
// no money, no funny
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($areas as $areaname => $areatitle) {
|
|
|
|
$this->set_area($areaname);
|
|
|
|
$method = $this->get_active_method();
|
|
|
|
|
|
|
|
if (empty($method)) {
|
|
|
|
// no grading method selected for the given area - nothing to display
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count($areas) > 1) {
|
|
|
|
// if the module supports multiple gradable areas, make a node for each of them
|
|
|
|
$node = $modulenode->add(get_string('gradinginarea', 'core_grading', $areatitle), null, settings_navigation::NODETYPE_BRANCH);
|
|
|
|
} else {
|
|
|
|
// otherwise put the items directly into the module's node
|
|
|
|
$node = $modulenode;
|
|
|
|
}
|
|
|
|
|
|
|
|
$controller = $this->get_controller($method);
|
|
|
|
$controller->extend_settings_navigation($settingsnav, $node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the given method's controller in the gradable area
|
|
|
|
*
|
|
|
|
* @param string $method the method name, eg 'rubric' (must be available)
|
|
|
|
* @return grading_controller
|
|
|
|
*/
|
|
|
|
public function get_controller($method) {
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
$this->ensure_isset(array('context', 'component', 'area'));
|
|
|
|
|
|
|
|
// make sure the passed method is a valid plugin name
|
|
|
|
if ('gradingform_'.$method !== clean_param('gradingform_'.$method, PARAM_COMPONENT)) {
|
|
|
|
throw new moodle_exception('invalid_method_name', 'core_grading');
|
|
|
|
}
|
|
|
|
$available = $this->get_available_methods(false);
|
|
|
|
if (!array_key_exists($method, $available)) {
|
|
|
|
throw new moodle_exception('invalid_method_name', 'core_grading');
|
|
|
|
}
|
|
|
|
|
|
|
|
// get the current grading area record if it exists
|
|
|
|
if (is_null($this->areacache)) {
|
|
|
|
$this->areacache = $DB->get_record('grading_areas', array(
|
|
|
|
'contextid' => $this->context->id,
|
|
|
|
'component' => $this->component,
|
|
|
|
'areaname' => $this->area),
|
|
|
|
'*', IGNORE_MISSING);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->areacache === false) {
|
|
|
|
// no area record yet, create one
|
|
|
|
$area = array(
|
|
|
|
'contextid' => $this->context->id,
|
|
|
|
'component' => $this->component,
|
|
|
|
'areaname' => $this->area);
|
|
|
|
$areaid = $DB->insert_record('grading_areas', $area);
|
|
|
|
// reload the cache
|
|
|
|
$this->areacache = $DB->get_record('grading_areas', array('id' => $areaid), '*', MUST_EXIST);
|
|
|
|
}
|
|
|
|
|
|
|
|
require_once($CFG->dirroot.'/grade/grading/form/'.$method.'/lib.php');
|
2011-10-05 03:51:41 +02:00
|
|
|
$classname = 'gradingform_'.$method.'_controller';
|
2011-10-03 21:57:16 +02:00
|
|
|
|
|
|
|
return new $classname($this->context, $this->component, $this->area, $this->areacache->id);
|
2011-09-28 23:33:21 +02:00
|
|
|
}
|
|
|
|
|
2011-10-03 21:57:16 +02:00
|
|
|
////////////////////////////////////////////////////////////////////////////
|
2011-10-02 23:31:05 +02:00
|
|
|
|
2011-09-28 23:33:21 +02:00
|
|
|
/**
|
|
|
|
* Make sure that the given properties were set to some not-null value
|
|
|
|
*
|
|
|
|
* @param array $properties the list of properties
|
|
|
|
* @throws coding_exception
|
|
|
|
*/
|
|
|
|
private function ensure_isset(array $properties) {
|
|
|
|
foreach ($properties as $property) {
|
|
|
|
if (!isset($this->$property)) {
|
2011-10-03 21:57:16 +02:00
|
|
|
throw new coding_exception('The property "'.$property.'" is not set.');
|
2011-09-28 23:33:21 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|