this patch also contains improvements in handling of needsupdate, though it is not yet fully working for calculated grades
// init grade_item
$this->load_grade_item();
+ if ($this->grade_item->is_locked()) {
+ return true; // no need to recalculate locked items
+ }
+
//get used items
$useditems = $this->dependson();
}
if ($used->itemid == $this->grade_item->id) {
$final = new grade_grades($used, false); // fetching from db is not needed
+ $final->grade_item =& $this->grade_item;
}
$grades['gi'.$used->itemid] = $used->finalgrade;
}
// can not use own final grade during calculation
unset($params['gi'.$this->grade_item->id]);
-
- // do the calculation
- $this->formula->set_params($params);
- $result = $this->formula->evaluate();
-
-
- // insert final grade - will be needed anyway later
+ // insert final grade - will be needed later anyway
if (empty($final)) {
$final = new grade_grades(array('itemid'=>$this->grade_item->id, 'userid'=>$userid), false);
$final->insert();
+ $final->grade_item =& $this->grade_item;
+
+ } else if ($final->is_locked()) {
+ // no need to recalculate locked grades
+ return;
}
+
+ // do the calculation
+ $this->formula->set_params($params);
+ $result = $this->formula->evaluate();
+
// store the result
if ($result === false) {
// error during calculation
}
/**
- * In addition to update() as defined in grade_object, call flag_for_update of parent categories, if applicable.
+ * In addition to update() as defined in grade_object, call force_regrading of parent categories, if applicable.
*/
function update() {
- $qualifies = $this->qualifies_for_update();
+ $qualifies = $this->qualifies_for_regrading();
// Update the grade_item's sortorder if needed
if (!empty($this->sortorder)) {
// Use $this->path to update all parent categories
if ($result && $qualifies) {
- $this->flag_for_update();
+ $this->force_regrading();
}
return $result;
}
/**
- * If parent::delete() is successful, send flag_for_update message to parent category.
+ * If parent::delete() is successful, send force_regrading message to parent category.
* @return boolean Success or failure.
*/
function delete() {
if ($result) {
$this->load_parent_category();
if (!empty($this->parent_category)) {
- $result = $result && $this->parent_category->flag_for_update();
+ $result = $result && $this->parent_category->force_regrading();
}
// Update children's categoryid/parent field
// Notify parent category of need to update.
$this->load_parent_category();
if (!empty($this->parent_category)) {
- if (!$this->parent_category->flag_for_update()) {
+ if (!$this->parent_category->force_regrading()) {
debugging("Could not notify parent category of the need to update its final grades.");
return false;
}
* This assumes that this object has an id number and a matching record in DB. If not, it will return false.
* @return boolean
*/
- function qualifies_for_update() {
+ function qualifies_for_regrading() {
if (empty($this->id)) {
return false;
}
* thanks to the path variable, so we don't need to use recursion.
* @return boolean Success or failure
*/
- function flag_for_update() {
+ function force_regrading() {
if (empty($this->id)) {
debugging("Needsupdate requested before insering grade category.");
return true;
$wheresql = substr($wheresql, 0, strrpos($wheresql, 'OR'));
$grade_items = set_field_select('grade_items', 'needsupdate', '1', $wheresql . ' AND courseid = ' . $this->courseid);
$this->grade_item->update_from_db();
+
+ }
+
+ if (count($paths) == 1) {
+ // we are the top category - force recalculation of all formulas in course
+ $this->grade_item->force_recalculation();
}
+
return $result;
}
global $CFG;
$this->load_grade_item();
+
+ if ($this->grade_item->is_locked()) {
+ return true; // no need to recalculate locked items
+ }
+
$this->grade_item->load_scale();
+
// find grde items of immediate children (category or grade items)
$dependson = $this->grade_item->dependson();
$items = array();
}
if ($used->itemid == $this->grade_item->id) {
$final = new grade_grades($used, false);
+ $final->grade_item =& $this->grade_item;
}
$grades[$used->itemid] = $used->finalgrade;
}
// no circular references allowed
unset($grades[$this->grade_item->id]);
- // insert final grade - it will needed anyway later
+ // insert final grade - it will be needed later anyway
if (empty($final)) {
$final = new grade_grades(array('itemid'=>$this->grade_item->id, 'userid'=>$userid), false);
$final->insert();
+ $final->grade_item =& $this->grade_item;
+
+ } else if ($final->is_locked()) {
+ // no need to recalculate locked grades
+ return;
}
// if no grades calculation possible or grading not allowed clear both final and raw
* grade_item, for cases where the object type is not known.
* @return int 0, 1 or timestamp int(10)
*/
- function get_locked() {
+ function is_locked() {
$this->load_grade_item();
if (!empty($this->grade_item)) {
- return $this->grade_item->locked;
+ return $this->grade_item->is_locked();
} else {
return false;
}
* Sets the grade_item's locked variable and updates the grade_item.
* Method named after grade_item::set_locked().
* @param int $locked 0, 1 or a timestamp int(10) after which date the item will be locked.
- * @return void
+ * @return boolean success
*/
- function set_locked($locked) {
+ function set_locked($lockedstate) {
$this->load_grade_item();
+
if (!empty($this->grade_item)) {
- $this->grade_item->locked = $locked;
- return $this->grade_item->update();
+ return $this->grade_item->set_locked($lockedstate);
+
} else {
return false;
}
return $this->grade_item;
}
+ /**
+ * Check grade lock status. Uses both grade item lock and grade lock.
+ * Internally any date in locked field (including future ones) means locked,
+ * the date is stored for logging purposes only.
+ *
+ * @return boolean true if locked, false if not
+ */
+ function is_locked() {
+ $this->load_grade_item();
+
+ return $this->grade_item->is_locked() or !empty($this->locked);
+ }
+
+ /**
+ * Lock/unlopck this grade.
+ *
+ * @param boolean $lockstate true means lock, false unlock grade
+ * @return boolean true if sucessful, false if can not set new lock state for grade
+ */
+ function set_locked($lockedstate) {
+ $this->load_grade_item();
+
+ if ($lockedstate) {
+ if (!empty($this->locked)) {
+ return true; // already locked
+ }
+
+ if ($this->grade_item->needsupdate) {
+ //can not lock grade if final not calculated!
+ return false;
+ }
+
+ $this->locked = time();
+ $this->update();
+
+ return true;
+
+ } else {
+ if (empty($this->locked)) {
+ return true; // not locked
+ }
+
+ if ($this->grade_item->is_locked()) {
+ return false;
+ }
+
+ // remove the locked flag
+ $this->locked = 0;
+
+ $this->update();
+
+ return true;
+ }
+ }
+
/**
* Finds and returns a grade_grades object based on 1-3 field values.
* @static
* Date until which to lock this grade_item. If null, 0 or false, grade_item is not locked. Locking prevents updating.
* @var int $locked
*/
- var $locked = false;
+ var $locked;
/**
* Whether or not the module instance referred to by this grade_item has been deleted.
/**
* In addition to update() as defined in grade_object, handle the grade_outcome and grade_scale objects.
+ * Force regrading if necessary
+ *
+ * @return boolean success
*/
function update() {
- // If item is flagged as deleted, only update that flag in DB. The other changes are ignored.
- if (!empty($this->deleted) && $this->deleted) {
- return set_field('grade_items', 'deleted', 1, 'id', $this->id);
- }
if (!empty($this->outcome->id)) {
$this->outcomeid = $this->outcome->id;
$this->scale = NULL;
}
- $qualifies = $this->qualifies_for_update();
-
- $result = parent::update();
-
- if ($result && $qualifies) {
- $category = $this->get_category();
+ if ($this->qualifies_for_regrading()) {
+ return $this->force_regrading();
- if (!empty($category)) {
- $result = $result && $category->flag_for_update();
- }
+ } else {
+ return parent::update();
}
-
- return $result;
}
/**
* This assumes that this object has an id number and a matching record in DB. If not, it will return false.
* @return boolean
*/
- function qualifies_for_update() {
+ function qualifies_for_regrading() {
if (empty($this->id)) {
return false;
}
$db_item = new grade_item(array('id' => $this->id));
- $gradetypediff = $db_item->gradetype != $this->gradetype;
- $grademaxdiff = $db_item->grademax != $this->grademax;
- $grademindiff = $db_item->grademin != $this->grademin;
- $scaleiddiff = $db_item->scaleid != $this->scaleid;
- $outcomeiddiff = $db_item->outcomeid != $this->outcomeid;
- $multfactordiff = $db_item->multfactor != $this->multfactor;
- $plusfactordiff = $db_item->plusfactor != $this->plusfactor;
- $needsupdatediff = $db_item->needsupdate != $this->needsupdate;
-
- if ($gradetypediff || $grademaxdiff || $grademindiff || $scaleiddiff || $outcomeiddiff ||
- $multfactordiff || $plusfactordiff || $needsupdatediff) {
- return true;
- } else {
- return false;
- }
+ $gradetypediff = $db_item->gradetype != $this->gradetype;
+ $grademaxdiff = $db_item->grademax != $this->grademax;
+ $grademindiff = $db_item->grademin != $this->grademin;
+ $scaleiddiff = $db_item->scaleid != $this->scaleid;
+ $outcomeiddiff = $db_item->outcomeid != $this->outcomeid;
+ $multfactordiff = $db_item->multfactor != $this->multfactor;
+ $plusfactordiff = $db_item->plusfactor != $this->plusfactor;
+ $deleteddiff = $db_item->deleted != $this->deleted;
+
+ $needsupdatediff = !$db_item->needsupdate && $this->needsupdate; // force regrading only if setting the flag first time
+ $lockeddiff = !empty($db_item->locked) && empty($this->locked); // force regrading only when unlocking
+
+ return ($gradetypediff || $grademaxdiff || $grademindiff || $scaleiddiff || $outcomeiddiff ||
+ $multfactordiff || $plusfactordiff || $deleteddiff || $needsupdatediff || $lockeddiff);
}
/**
}
/**
- * If parent::delete() is successful, send flag_for_update message to parent category.
+ * If parent::delete() is successful, send force_regrading message to parent category.
* @return boolean Success or failure.
*/
function delete() {
if ($result) {
$category = $this->get_category();
if (!empty($category)) {
- return $category->flag_for_update();
+ return $category->force_regrading();
}
}
return $result;
}
/**
- * In addition to perform parent::insert(), this calls the grade_item's category's (if applicable) flag_for_update() method.
+ * In addition to perform parent::insert(), this calls the grade_item's category's (if applicable) force_regrading() method.
* @return int ID of the new grade_item record.
*/
function insert() {
global $CFG;
- // all new grade_items must be recalculated
- $this->needsupdate = true;
-
if (!isset($this->gradetype)) {
$this->gradetype = GRADE_TYPE_VALUE;
}
$result = parent::insert();
- // Notify parent category of need to update. Note that a grade_item may not have a categoryid.
if ($result) {
- $category = $this->get_category();
- if (!empty($category)) {
- if (!$category->flag_for_update()) {
- debugging("Could not notify parent category of the need to update its final grades.");
- return false;
- }
- }
+ // force regrading of items if needed
+ $this->force_regrading();
+ return true;
+
} else {
debugging("Could not insert this grade_item in the database!");
+ return false;
}
-
- return $result;
}
/**
* @return boolean Locked state
*/
function is_locked($userid=NULL) {
- // TODO: rewrite the item check
+ if (!empty($this->locked)) {
+ return true;
+ }
- if ($this->locked || empty($userid)) {
- return $this->locked; // This could be true or false (false only if no $userid given)
- } else {
- $final = $this->get_final($userid);
- return $final->locked;
+ if (!empty($userid)) {
+
+ $grade = new grade_grades(array('itemid'=>$this->id, 'userid'=>$userid));
+ $grade->grade_item =& $this; // prevent db fetching of cached grade_item
+
+ if (!empty($grade->id) and $grade->is_locked()) {
+ return true;
+ }
}
+
+ return false;
}
/**
* Locks or unlocks this grade_item and (optionally) all its associated final grades.
* @param boolean $update_final Whether to update final grades too
* @param boolean $new_state Optional new state. Will use inverse of current state otherwise.
- * @return int Number of final grades changed, or false if error occurred during update.
+ * @return boolean true if grade_item all grades updated, false if at least one update fails
*/
- function toggle_locking($update_final=false, $new_state=NULL) {
- // TODO: implement new locking
+ function set_locked($lockedstate) {
+ if ($lockedstate) {
+ /// setting lock
+ if (!empty($this->locked)) {
+ return true; // already locked
+ }
- return 0;
+ if ($this->needsupdate) {
+ return false; // can not lock grade without first calculating final grade
+ }
+
+ $this->locked = time();
+ $this->update();
+
+ // this could be improved with direct SQL update
+ $result = true;
+ $grades = $this->get_final();
+ foreach($grades as $g) {
+ $grade = new grade_grades($g, false);
+ $grade->grade_item =& $this;
+ if (!$grade->set_locked(true)) {
+ $result = false;
+ }
+ }
+
+ return $result;
+
+ } else {
+ /// removing lock
+ if (empty($this->locked)) {
+ return true; // not locked
+ }
+
+ if (!empty($this->locktime) and $this->locktime < time()) {
+ return false; // can not unlock grade item that should be already locked
+ }
+
+ $this->locked = 0;
+ $this->update();
+
+ // this could be improved with direct SQL update
+ $result = true;
+ $grades = $this->get_final();
+ foreach($grades as $g) {
+ $grade = new grade_grades($g, false);
+ $grade->grade_item =& $this;
+
+ if (!empty($grade->locktime) and $grade->locktime < time()) {
+ $result = false; // can not unlock grade that should be already locked
+ }
+
+ if (!$grade->set_locked(false)) {
+ $result = false;
+ }
+ }
+
+ return $result;
+
+ }
}
/**
function update_final_grades() {
global $CFG;
- $errors = array();
+ if ($this->is_locked()) {
+ // locked grade items already have correct final grades
+ return true;
+ }
if ($calculation = $this->get_calculation()) {
if ($calculation->compute()) {
- $this->needsupdate = false;
- $this->update();
+ $this->force_regrade();
return true;
} else {
- $errors[] = "Could not calculate grades for grade item id:".$this->id; // TODO: improve and localize
+ return array("Could not calculate grades for grade item id:".$this->id); // TODO: improve and localize
}
} else if ($this->itemtype == 'category') {
// aggregate category grade item
$category = $this->get_category();
if (!$category->generate_grades()) {
- $errors[] = "Could not calculate category grade item id:".$this->id; // TODO: improve and localize
+ return ("Could not calculate raw category grades id:".$this->id); // TODO: improve and localize
}
}
- // TODO: add locking support
+ $errors = array();
+
+ // we need it to be really fast here ==> sql only
if ($rs = get_recordset('grade_grades', 'itemid', $this->id)) {
if ($rs->RecordCount() > 0) {
while ($grade = rs_fetch_next_record($rs)) {
+ if (!empty($grade->locked)) {
+ // this grade is locked - final grade must be ok
+ continue;
+ }
+
if (!empty($errors) or is_null($grade->rawgrade)) {
// unset existing final grade when no raw present or error
if (!is_null($grade->finalgrade)) {
$errors[] = "Could not update final grade for grade item:".$this->id;
}
}
+
+ // do not use $grade->is_locked() bacause item may be still locked!
+ if (!empty($grade->locktime) and empty($grade->locked) and $grade->locktime < time()) {
+ // time to lock this grade
+ $g = new object();
+ $g->id = $grade->id;
+ $g->locked = time();
+ update_record('grade_grades', $g);
+ }
}
}
}
}
if (!empty($errors)) {
- $this->flag_for_update();
+ $this->force_regrading();
return $errors;
} else {
+ // reset the regrading flag
$this->needsupdate = false;
$this->update();
+
+ // recheck the needsupdate just to make sure ;-)
+ if (empty($this->needsupdate) and !empty($this->locktime)
+ and empty($this->locked) and $this->locktime < time()) {
+ // time to lock this grade_item
+ $this->set_locked(true);
+ }
+
return true;
}
}
/**
* Sets this grade_item's needsupdate to true. Also looks at parent category, if any, and calls
- * its flag_for_update() method.
+ * its force_regrading() method.
* This is triggered whenever any change in any raw grade may cause grade_finals
* for this grade_item to require an update. The flag needs to be propagated up all
* levels until it reaches the top category. This is then used to determine whether or not
* to regenerate the raw and final grades for each category grade_item.
* @return boolean Success or failure
*/
- function flag_for_update() {
+ function force_regrading() {
$this->needsupdate = true;
- $result = $this->update();
- $category = $this->get_category();
+ $result = parent::update();
+
+ if ($category = $this->get_category()) {
+ $category->force_regrading(); // we can ignore the result
- if (!empty($category)) {
- $result = $result && $category->flag_for_update();
+ } else {
+ $this->force_recalculation(); // recalculate all formulas - we do not know if any of them depends on this item
}
+
return $result;
}
+ /**
+ * Force recalculation of all calculated grades in course.
+ */
+ function force_recalculation($courseid=null) {
+ if (empty($courseid)) {
+ $courseid = $this->courseid;
+ }
+ $grade_item = new grade_item(array('courseid'=>$courseid), false);
+ $grade_items = $grade_item->fetch_all_using_this();
+ foreach($grade_items as $gi) {
+ if ($gi->get_calculation()) {
+ $gi->needsupdate = true;
+ $gi->update();
+ }
+ }
+ }
+
/**
* Disassociates this item from its category parent(s). The object is then updated in DB.
* @return boolean Success or Failure
}
}
$this->calculation = false; // cache no calculation present
- $this->flag_for_update();
+ $this->force_regrading();
return true;
} else { // We are updating or creating the calculation entry in the DB
if ($grade_calculation = $this->get_calculation(true)) {
$grade_calculation->calculation = $formula;
if ($grade_calculation->update()) {
- $this->flag_for_update();
+ $this->force_regrading();
return true;
} else {
$this->calculation = null; // remove cache
$this->categoryid = $parentid;
}
- /**
- * Returns the locked state/date of this grade_item. This method is also available in
- * grade_category, for cases where the object type is not known.
- * @return int 0, 1 or timestamp int(10)
- */
- function get_locked() {
- return $this->locked;
- }
-
- /**
- * Sets the grade_item's locked variable and updates the grade_item.
- * @param int $locked 0, 1 or a timestamp int(10) after which date the item will be locked.
- * @return success or failure of update() method
- */
- function set_locked($locked) {
- $this->locked = $locked;
- return $this->update();
- }
-
/**
* Returns the hidden state/date of this grade_item. This method is also available in
* grade_category, for cases where the object type is not known.
* @return int 0, 1 or timestamp int(10)
*/
- function get_hidden() {
+ function is_hidden() {
+ // to do
return $this->hidden;
}
*/
function dependson() {
+ if ($this->is_locked()) {
+ // locked items do not need to be regraded
+ return array();
+ }
+
if ($calculation = $this->get_calculation()) {
return $calculation->dependson();
* Calculated grades do not use raw grades at all, the rawgrade changes there are not logged too.
*
* @param int $userid the graded user
- * @param float $rawgrade value of raw grade
+ * @param mixed $rawgrade float value of raw grade - false means do not change
* @param string $howmodified modification source
* @param string $note optional note
- * @param string $feedback teachjers feedback
+ * @param mixed $feedback teachers feedback as string - false means do not change
* @param int $feedbackformat
- * @return boolean true if ok, false if error
+ * @return mixed grade_grades object if ok, false if error
*/
- function update_raw($userid, $rawgrade=false, $howmodified='manual', $note=NULL, $feedback=false, $feedbackformat=FORMAT_MOODLE) {
+ function update_raw_grade($userid, $rawgrade=false, $howmodified='manual', $note=NULL, $feedback=false, $feedbackformat=FORMAT_MOODLE) {
+
+ // calculated grades can not be updated
+ if ($this->get_calculation()) {
+ return false;
+ }
+
+ // do not allow grade updates when item locked - this prevents fetching of grade from db
+ if ($this->is_locked()) {
+ return false;
+ }
+
$grade = new grade_grades(array('itemid'=>$this->id, 'userid'=>$userid));
+ $grade->grade_item =& $this; // prevent db fetching of cached grade_item
+
+ if (!empty($grade->id)) {
+ if ($grade->is_locked()) {
+ // do not update locked grades at all
+ return false;
+ }
+
+ if ($grade->locktime < time()) {
+ // do not update grades that should be already locked
+ // this does not solve all problems, cron is still needed to recalculate the final grades periodically
+ return false;
+ }
+
+ }
- //TODO: add locking checks here - prevent update if item or individaul grade locked
- //TODO: if grade tree does not need to be recalculated, try to update all users grades in course and flag_for_update only if failed
+ //TODO: if grade tree does not need to be recalculated, try to update grades of all users in course and force_regrading only if failed
// fist copy current grademin/max and scale
$grade->rawgrademin = $this->grademin;
grade_history::insert_change($userid, $this->id, $grade->rawgrade, $oldgrade, $howmodified, $note);
// This grade item needs update
- $this->flag_for_update();
+ $this->force_regrading();
if ($result) {
return $grade;
// Prepare lock/unlock string
$lock_unlock = 'lock';
- if ($object->get_locked()) {
+ if ($object->is_locked()) {
$lock_unlock = 'unlock';
}
$grade_item->insert();
} else {
- if ($grade_item->locked) {
+ if ($grade_item->is_locked()) {
debugging('Grading item is locked!');
return GRADE_UPDATE_ITEM_LOCKED;
}
$userid = $grade['userid'];
}
- $rawgrade = false;
+ $rawgrade = false;
$feedback = false;
$feedbackformat = FORMAT_MOODLE;
}
// update or insert the grade
- $grade = $grade_item->update_raw($userid, $rawgrade, $source, null, $feedback, $feedbackformat);
+ $grade = $grade_item->update_raw_grade($userid, $rawgrade, $source, null, $feedback, $feedbackformat);
if (!$grade) {
$failed = true;
- debugging('Grade not updated');
continue;
}
}
if ($forceupdate) {
- $grade_item->flag_for_update();
+ $grade_item->force_regrading();
} else {
$finalitems[$gid] = $grade_item;
$finalids[] = $gid;
$grade_item->grademin = 0;
$grade_item->grademax = 100;
$grade_item->iteminfo = 'Grade item used for unit testing';
- $grade_item->locked = mktime() + 240000;
$grade_item->timecreated = mktime();
$grade_item->timemodified = mktime();
$grade_item->sortorder = 4;
$grade_item->itemtype = 'mod';
$grade_item->itemmodule = 'quiz';
$grade_item->iteminstance = 5;
+ $grade_item->itemnumber = 0;
$grade_item->gradetype = GRADE_TYPE_VALUE;
$grade_item->grademin = 10;
$grade_item->grademax = 120;
+ $grade_item->locked = time();
$grade_item->iteminfo = 'Orphan Grade item used for unit testing';
$grade_item->timecreated = mktime();
$grade_item->timemodified = mktime();
$grade->finalgrade = 60;
$grade->timecreated = mktime();
$grade->timemodified = mktime();
- $grade->locked = true;
if ($grade->id = insert_record('grade_grades', $grade)) {
$this->grade_grades[3] = $grade;
$grade->finalgrade = 70;
$grade->timecreated = mktime();
$grade->timemodified = mktime();
- $grade->locked = true;
if ($grade->id = insert_record('grade_grades', $grade)) {
$this->grade_grades[4] = $grade;
$grade->finalgrade = 100;
$grade->timecreated = mktime();
$grade->timemodified = mktime();
- $grade->locked = false;
if ($grade->id = insert_record('grade_grades', $grade)) {
$this->grade_grades[5] = $grade;
* @package moodlecore
*/
-global $CFG;if (!defined('MOODLE_INTERNAL')) {
+if (!defined('MOODLE_INTERNAL')) {
die('Direct access to this script is forbidden.'); /// It must be included from a Moodle page
}
}
+ function test_grade_grades_set_locked() {
+ $grade_item = new grade_item($this->grade_items[0]);
+ $grade = new grade_grades($grade_item->get_final(1));
+ $this->assertTrue(method_exists($grade, 'set_locked'));
+
+ $this->assertTrue(empty($grade_item->locked));
+ $this->assertTrue(empty($grade->locked));
+
+ $this->assertTrue($grade->set_locked(true));
+ $this->assertFalse(empty($grade->locked));
+ $this->assertTrue($grade->set_locked(false));
+ $this->assertTrue(empty($grade->locked));
+
+ $this->assertTrue($grade_item->set_locked(true));
+ $grade = new grade_grades($grade_item->get_final(1));
+
+ $this->assertFalse(empty($grade->locked));
+ $this->assertFalse($grade->set_locked(false));
+
+ $this->assertTrue($grade_item->set_locked(false));
+ $grade = new grade_grades($grade_item->get_final(1));
+
+ $this->assertTrue($grade->set_locked(false));
+ }
+
+ function test_grade_grades_is_locked() {
+ $grade = new grade_grades($this->grade_grades[0]);
+ $this->assertTrue(method_exists($grade, 'is_locked'));
+
+ $this->assertFalse($grade->is_locked());
+ $grade->locked = time();
+ $this->assertTrue($grade->is_locked());
+ }
+
+
}
?>
\r
class grade_item_test extends grade_test {\r
\r
- function test_grade_item_construct() { \r
+ function test_grade_item_construct() {\r
$params = new stdClass();\r
\r
$params->courseid = $this->courseid;\r
function test_grade_item_insert() {\r
$grade_item = new grade_item();\r
$this->assertTrue(method_exists($grade_item, 'insert'));\r
- \r
+\r
$grade_item->courseid = $this->courseid;\r
$grade_item->categoryid = $this->grade_categories[1]->id;\r
$grade_item->itemname = 'unittestgradeitem4';\r
$grade_item->iteminfo = 'Grade item used for unit testing';\r
\r
// Check the grade_category's needsupdate variable first\r
- $category = $grade_item->get_category(); \r
+ $category = $grade_item->get_category();\r
$category->load_grade_item();\r
$category->grade_item->needsupdate = false;\r
$this->assertNotNull($category->grade_item);\r
}\r
\r
function test_grade_item_update_when_flagged_as_deleted() {\r
- \r
+\r
}\r
\r
function test_grade_item_update_guess_outcomeid() {\r
function test_grade_item_delete() {\r
$grade_item = new grade_item($this->grade_items[0]);\r
$this->assertTrue(method_exists($grade_item, 'delete'));\r
- \r
+\r
// Check the grade_category's needsupdate variable first\r
- $category = $grade_item->get_category(); \r
+ $category = $grade_item->get_category();\r
$category->load_grade_item();\r
$this->assertNotNull($category->grade_item);\r
$category->grade_item->needsupdate = false;\r
- \r
+\r
$this->assertTrue($grade_item->delete());\r
\r
// Now check the needsupdate variable, it should have been set to true\r
function test_grade_item_update() {\r
$grade_item = new grade_item($this->grade_items[0]);\r
$this->assertTrue(method_exists($grade_item, 'update'));\r
- \r
+\r
$grade_item->iteminfo = 'Updated info for this unittest grade_item';\r
\r
// Check the grade_category's needsupdate variable first\r
- $category= $grade_item->get_category(); \r
+ $category= $grade_item->get_category();\r
$category->load_grade_item();\r
$this->assertNotNull($category->grade_item);\r
$category->grade_item->needsupdate = false;\r
- \r
+\r
$this->assertTrue($grade_item->update());\r
\r
// Now check the needsupdate variable, it should NOT have been set to true, because insufficient changes to justify update.\r
$this->assertFalse($category->grade_item->needsupdate);\r
- \r
- $grade_item->grademin = 14; \r
- $this->assertTrue($grade_item->qualifies_for_update());\r
+\r
+ $grade_item->grademin = 14;\r
+ $this->assertTrue($grade_item->qualifies_for_regrading());\r
$this->assertTrue($grade_item->update(true));\r
- \r
+\r
// Now check the needsupdate variable, it should have been set to true\r
$category->grade_item->update_from_db();\r
$this->assertTrue($category->grade_item->needsupdate);\r
$category->load_parent_category();\r
$category->parent_category->load_grade_item();\r
$this->assertTrue($category->parent_category->grade_item->needsupdate);\r
- \r
+\r
$iteminfo = get_field('grade_items', 'iteminfo', 'id', $this->grade_items[0]->id);\r
$this->assertEqual($grade_item->iteminfo, $iteminfo);\r
}\r
\r
- function test_grade_item_qualifies_for_update() {\r
+ function test_grade_item_qualifies_for_regrading() {\r
$grade_item = new grade_item($this->grade_items[0]);\r
- $this->assertTrue(method_exists($grade_item, 'qualifies_for_update'));\r
- \r
+ $this->assertTrue(method_exists($grade_item, 'qualifies_for_regrading'));\r
+\r
+ $this->assertFalse($grade_item->qualifies_for_regrading());\r
+\r
$grade_item->iteminfo = 'Updated info for this unittest grade_item';\r
- \r
- $this->assertFalse($grade_item->qualifies_for_update());\r
+\r
+ $this->assertFalse($grade_item->qualifies_for_regrading());\r
\r
$grade_item->grademin = 14;\r
\r
- $this->assertTrue($grade_item->qualifies_for_update()); \r
+ $this->assertTrue($grade_item->qualifies_for_regrading());\r
}\r
\r
function test_grade_item_fetch() {\r
- $grade_item = new grade_item(); \r
+ $grade_item = new grade_item();\r
$this->assertTrue(method_exists($grade_item, 'fetch'));\r
\r
$grade_item = grade_item::fetch('id', $this->grade_items[0]->id);\r
$this->assertEqual($this->grade_items[0]->id, $grade_item->id);\r
- $this->assertEqual($this->grade_items[0]->iteminfo, $grade_item->iteminfo); \r
- \r
+ $this->assertEqual($this->grade_items[0]->iteminfo, $grade_item->iteminfo);\r
+\r
$grade_item = grade_item::fetch('itemtype', $this->grade_items[1]->itemtype, 'itemmodule', $this->grade_items[1]->itemmodule);\r
$this->assertEqual($this->grade_items[1]->id, $grade_item->id);\r
- $this->assertEqual($this->grade_items[1]->iteminfo, $grade_item->iteminfo); \r
+ $this->assertEqual($this->grade_items[1]->iteminfo, $grade_item->iteminfo);\r
}\r
\r
function test_grade_item_fetch_all_using_this() {\r
$grade_item = new grade_item();\r
$grade_item->itemtype = 'mod';\r
$this->assertTrue(method_exists($grade_item, 'fetch_all_using_this'));\r
- \r
+\r
$grade_items = $grade_item->fetch_all_using_this();\r
$this->assertEqual(5, count($grade_items));\r
$first_grade_item = reset($grade_items);\r
$this->assertEqual($this->grade_items[0]->id, $first_grade_item->id);\r
}\r
\r
- \r
+\r
/**\r
* Retrieve all final scores for a given grade_item.\r
*/\r
function test_grade_item_get_all_finals() {\r
$grade_item = new grade_item($this->grade_items[0]);\r
$this->assertTrue(method_exists($grade_item, 'get_final'));\r
- \r
+\r
$final_grades = $grade_item->get_final();\r
- $this->assertEqual(3, count($final_grades)); \r
+ $this->assertEqual(3, count($final_grades));\r
}\r
\r
- \r
+\r
/**\r
* Retrieve all final scores for a specific userid.\r
*/\r
/* $grade_item = new grade_item($this->grade_items[1]);\r
$this->assertTrue(method_exists($grade_item, 'set_calculation'));\r
$this->assertTrue(method_exists($grade_item, 'get_calculation'));\r
- \r
+\r
$calculation = '=SUM([unittestgradeitem1], [unittestgradeitem3])';\r
$grade_item->set_calculation($calculation);\r
$new_calculation = $grade_item->get_calculation();\r
function test_grade_item_get_category() {\r
$grade_item = new grade_item($this->grade_items[0]);\r
$this->assertTrue(method_exists($grade_item, 'get_category'));\r
- \r
+\r
$category = $grade_item->get_category();\r
$this->assertEqual($this->grade_categories[1]->fullname, $category->fullname);\r
}\r
function test_grade_item_update_final_grades() {\r
$grade_item = new grade_item($this->grade_items[0]);\r
$this->assertTrue(method_exists($grade_item, 'update_final_grades'));\r
- $this->assertEqual(true, $grade_item->update_final_grades()); \r
+ $this->assertEqual(true, $grade_item->update_final_grades());\r
}\r
- \r
+\r
/**\r
* Test the adjust_grade method\r
*/\r
$grade_raw->rawgrade = 40;\r
$grade_raw->grademax = 100;\r
$grade_raw->grademin = 0;\r
- \r
+\r
$grade_item->multfactor = 1;\r
$grade_item->plusfactor = 0;\r
$grade_item->grademax = 50;\r
$grade_item->grademin = 0;\r
- \r
+\r
$original_grade_raw = clone($grade_raw);\r
$original_grade_item = clone($grade_item);\r
\r
- $this->assertEqual(20, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax)); \r
- \r
+ $this->assertEqual(20, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax));\r
+\r
// Try a larger maximum grade\r
$grade_item->grademax = 150;\r
$grade_item->grademin = 0;\r
- $this->assertEqual(60, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax)); \r
+ $this->assertEqual(60, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax));\r
\r
// Try larger minimum grade\r
$grade_item->grademin = 50;\r
\r
- $this->assertEqual(90, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax)); \r
+ $this->assertEqual(90, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax));\r
\r
// Rescaling from a small scale (0-50) to a larger scale (0-100)\r
$grade_raw->grademax = 50;\r
$grade_item->grademax = 100;\r
$grade_item->grademin = 0;\r
\r
- $this->assertEqual(80, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax)); \r
+ $this->assertEqual(80, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax));\r
\r
// Rescaling from a small scale (0-50) to a larger scale with offset (40-100)\r
$grade_item->grademax = 100;\r
$grade_item->grademin = 40;\r
\r
- $this->assertEqual(88, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax)); \r
+ $this->assertEqual(88, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax));\r
\r
// Try multfactor and plusfactor\r
$grade_raw = clone($original_grade_raw);\r
$grade_item->multfactor = 1.23;\r
$grade_item->plusfactor = 3;\r
\r
- $this->assertEqual(27.6, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax)); \r
+ $this->assertEqual(27.6, $grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax));\r
\r
// Try multfactor below 0 and a negative plusfactor\r
$grade_raw = clone($original_grade_raw);\r
$grade_item->multfactor = 0.23;\r
$grade_item->plusfactor = -3;\r
\r
- $this->assertEqual(round(1.6), round($grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax))); \r
+ $this->assertEqual(round(1.6), round($grade_item->adjust_grade($grade_raw->rawgrade, $grade_raw->grademin, $grade_raw->grademax)));\r
}\r
\r
- function test_grade_item_adjust_scale_grade() {\r
-/* // Load grade item and its scale\r
- $grade_item = new grade_item(array('scaleid' => $this->scale[1]->id), false);\r
- $grade_item->gradetype = GRADE_TYPE_SCALE;\r
- $grade_item->insert();\r
- $grade_item->load_scale();\r
- $this->assertEqual('Very Good', $grade_item->scale->scale_items[1]);\r
-\r
- // Load raw grade and its scale\r
- $grade_raw = new grade_grades(array('scaleid' => $this->scale[0]->id), false);\r
- $grade_raw->rawgrade = 4;\r
- $grade_raw->itemid = $grade_item->id;\r
- $grade_raw->userid = 1;\r
- $grade_raw->insert();\r
- $grade_raw->load_scale();\r
- $this->assertEqual('Fairly neutral', $grade_raw->scale->scale_items[2]);\r
-\r
- // Test grade_item::adjust_scale\r
- $this->assertEqual(3, $grade_item->adjust_grade($grade_raw));\r
- $grade_raw->rawgrade = 6;\r
- $this->assertEqual(4, $grade_item->adjust_grade($grade_raw));\r
-*/ }\r
+ function test_grade_item_set_locked() {\r
+ $grade_item = new grade_item($this->grade_items[0]);\r
+ $this->assertTrue(method_exists($grade_item, 'set_locked'));\r
\r
- function test_grade_item_toggle_locking() {\r
-/* $grade_item = new grade_item($this->grade_items[0]);\r
- $this->assertTrue(method_exists($grade_item, 'toggle_locking'));\r
+ $grade = new grade_grades($grade_item->get_final(1));\r
+ $this->assertTrue(empty($grade_item->locked));\r
+ $this->assertTrue(empty($grade->locked));\r
\r
- $this->assertFalse($grade_item->locked);\r
- $this->assertEqual(0, $grade_item->toggle_locking());\r
- $this->assertTrue($grade_item->locked);\r
- $grade_item->load_final();\r
- $this->assertFalse($grade_item->grade_grades[1]->locked);\r
- \r
- $grade_item->locked = false;\r
- $this->assertEqual(3, $grade_item->toggle_locking(true));\r
- $this->assertTrue($grade_item->locked);\r
- $this->assertTrue($grade_item->grade_grades[1]->locked);\r
- $this->assertTrue($grade_item->grade_grades[2]->locked);\r
- $this->assertTrue($grade_item->grade_grades[3]->locked);\r
+ $this->assertTrue($grade_item->set_locked(true));\r
+ $grade = new grade_grades($grade_item->get_final(1));\r
+\r
+ $this->assertFalse(empty($grade_item->locked));\r
+ $this->assertFalse(empty($grade->locked)); // individual grades should be locked too\r
+\r
+ $this->assertTrue($grade_item->set_locked(false));\r
+ $grade = new grade_grades($grade_item->get_final(1));\r
+\r
+ $this->assertTrue(empty($grade_item->locked));\r
+ $this->assertTrue(empty($grade->locked)); // individual grades should be unlocked too\r
}\r
\r
+ function test_grade_item_is_locked() {\r
+ $grade_item = new grade_item($this->grade_items[0]);\r
+ $this->assertTrue(method_exists($grade_item, 'is_locked'));\r
+\r
+ $this->assertFalse($grade_item->is_locked());\r
+ $this->assertFalse($grade_item->is_locked(1));\r
+ $this->assertTrue($grade_item->set_locked(true));\r
+ $this->assertTrue($grade_item->is_locked());\r
+ $this->assertTrue($grade_item->is_locked(1));\r
+ }\r
+\r
+ function test_grade_item_dependson() {\r
+ $grade_item = new grade_item($this->grade_items[0]);\r
+ //TODO\r
+ }\r
+\r
+\r
+/*\r
function test_grade_item_toggle_hiding() {\r
$grade_item = new grade_item($this->grade_items[0]);\r
$this->assertTrue(method_exists($grade_item, 'toggle_hiding'));\r
$this->assertTrue($grade_item->hidden);\r
$grade_item->load_final();\r
$this->assertFalse($grade_item->grade_grades[1]->hidden);\r
- \r
+\r
$grade_item->hidden = false;\r
$this->assertEqual(3, $grade_item->toggle_hiding(true));\r
$this->assertTrue($grade_item->hidden);\r
$this->assertTrue($grade_item->grade_grades[1]->hidden);\r
$this->assertTrue($grade_item->grade_grades[2]->hidden);\r
$this->assertTrue($grade_item->grade_grades[3]->hidden);\r
-*/ } \r
+ }\r
+*/\r
\r
function test_float_keys() {\r
}\r
-} \r
+}\r
?>\r
}
function test_grade_tree_display_grades() {
- $tree = new grade_tree($this->courseid);
+/* $tree = new grade_tree($this->courseid);
$tree->build_tree_filled();
$result_html = $tree->display_grades();
$expected_html = '<table style="text-align: center" border="1"><tr><th colspan="3">unittestcategory1</th><td class="topfiller"> </td><td colspan="2" class="topfiller"> </td></tr><tr><td colspan="2">unittestcategory2</td><td colspan="1">unittestcategory3</td><td class="subfiller"> </td><td colspan="2">level1category</td></tr><tr><td>unittestgradeitem1</td><td>unittestgradeitem2</td><td>unittestgradeitem3</td><td>unittestorphangradeitem1</td><td>singleparentitem1</td><td>singleparentitem2</td></tr></table>';
$this->assertEqual($expected_html, $result_html);
+*/
}
function test_grade_tree_get_tree() {
if (get_class($this) == 'gradelib_test') {
$grade_item = $this->grade_items[0];
$this->assertFalse(grade_is_locked($grade_item->courseid, $grade_item->itemtype, $grade_item->itemmodule, $grade_item->iteminstance, $grade_item->itemnumber));
- $grade_item = $this->grade_items[1];
+ $grade_item = $this->grade_items[6];
$this->assertTrue(grade_is_locked($grade_item->courseid, $grade_item->itemtype, $grade_item->itemmodule, $grade_item->iteminstance, $grade_item->itemnumber));
}
}