This repository has been archived on 2025-01-01. You can view files and clone it, but cannot push or open issues or pull requests.
moodle-local_treestudyplan/classes/local/aggregators/tristate_aggregator.php

164 lines
6.4 KiB
PHP
Raw Normal View History

<?php
2023-08-24 23:02:41 +02:00
// This file is part of the Studyplan plugin for Moodle
//
// 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 <https://www.gnu.org/licenses/>.
/**
*
* @package local_treestudyplan
* @copyright 2023 P.M. Kuipers
* @license https://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
namespace local_treestudyplan\local\aggregators;
use \local_treestudyplan\courseinfo;
use \local_treestudyplan\gradeinfo;
use \local_treestudyplan\studyitem;
use \local_treestudyplan\completion;
class tristate_aggregator extends \local_treestudyplan\aggregator {
public const DEPRECATED = true;
private const DEFAULT_CONDITION = "50";
2023-08-24 23:02:41 +02:00
public function needSelectGradables() { return True;}
public function isDeprecated() { return self::DEPRECATED;}
public function useRequiredGrades() { return False;}
public function useItemConditions() { return True;}
protected function aggregate_completion(array $a, $condition = "50") {
2023-08-24 23:02:41 +02:00
if (in_array($condition, ['ALL', '67', '50', 'ANY'])) {
// condition is one of the valid conditions.
$c_completed = 0;
$c_excellent = 0;
$c_progress = 0;
$c_pending = 0;
$count = sizeof($a);
2023-08-24 23:02:41 +02:00
if ($count > 0) {
2023-08-24 23:02:41 +02:00
foreach ($a as $c) {
$c_progress += ($c>=completion::PROGRESS)?1:0;
$c_completed += ($c>=completion::COMPLETED)?1:0;
$c_excellent += ($c>=completion::EXCELLENT)?1:0;
$c_pending += ($c>=completion::PENDING)?1:0;
}
$required = [
'ALL' => 1.00 * $count,
'67' => 0.67 * $count,
'50' => 0.50 * $count,
'ANY' => 1,
][$condition];
2023-08-24 23:02:41 +02:00
if ($c_excellent >= $required) {
return completion::EXCELLENT;
} else if ($c_completed >= $required) {
return completion::COMPLETED;
} else {
2023-08-24 23:02:41 +02:00
// Return PROGRESS if one or more completions are COMPLETED or EXCELLENT, but the aggregation margin is not met.
// state PROGRESS will not carry on if aggregations are chained.
if ($c_progress > 0) {
return completion::PROGRESS;
2023-08-24 23:09:20 +02:00
} else if ($c_pending > 0) {
return completion::PENDING;
2023-08-24 23:09:20 +02:00
} else {
return completion::INCOMPLETE;
}
}
2023-08-24 23:09:20 +02:00
} else {
return completion::INCOMPLETE;
}
2023-08-24 23:09:20 +02:00
} else
{
2023-08-24 23:02:41 +02:00
// indeterminable, return null.
return null;
}
}
2023-08-24 23:02:41 +02:00
public function aggregate_course(courseinfo $courseinfo, studyitem $studyitem, $userid) {
$condition = $studyitem->conditions();
2023-08-24 23:02:41 +02:00
if (empty($condition)) {
$condition = self::DEFAULT_CONDITION;
}
$list = [];
2023-08-24 23:02:41 +02:00
foreach (gradeinfo::list_studyitem_gradables($studyitem) as $gi) {
$list[] = $this->grade_completion($gi, $userid);
}
2023-08-24 23:02:41 +02:00
$completion = self::aggregate_completion($list, $condition);
return $completion;
}
2023-08-24 23:02:41 +02:00
public function aggregate_junction(array $completion, studyitem $studyitem, $userid) {
$completed = self::aggregate_completion($completion, $studyitem->conditions());
// if null result (conditions are unknown/null) - default to ALL.
return isset($completed)?$completed:(self::aggregate_completion($completion, 'ALL'));
}
public function grade_completion(gradeinfo $gradeinfo, $userid) {
global $DB;
$gradeitem = $gradeinfo->getGradeitem();
$grade = $gradeitem->get_final($userid);
2023-08-24 23:02:41 +02:00
if (empty($grade)) {
return completion::INCOMPLETE;
2023-08-24 23:09:20 +02:00
} else if ($grade->finalgrade === NULL) {
2023-08-24 23:02:41 +02:00
// on assignments, grade NULL means a submission has not yet been graded,.
// but on quizes this can also mean a quiz might have been started.
// Therefor, we treat a NULL result as a reason to check the relevant gradingscanner for presence of pending items.
// Since we want old results to be visible until a pending item was graded, we only use this state here.
2023-08-24 23:02:41 +02:00
// Pending items are otherwise expressly indicated by the "pendingsubmission" field in the user model.
if ($gradeinfo->getGradingscanner()->pending($userid)) {
return completion::PENDING;
} else {
return completion::INCOMPLETE;
}
2023-08-24 23:02:41 +02:00
2023-08-24 23:09:20 +02:00
} else {
$finalgrade = $grade->finalgrade;
$scale = $gradeinfo->getScale();
2023-08-24 23:02:41 +02:00
if ($gradeitem->gradepass > 0) {
// Base completion off of gradepass (if set).
if ($gradeitem->grademax > $gradeitem->gradepass && $finalgrade >= $gradeitem->grademax) {
// If gradepass is configured .
return completion::EXCELLENT;
2023-08-24 23:09:20 +02:00
} else if ($finalgrade >= $gradeitem->gradepass) {
return completion::COMPLETED;
2023-08-24 23:09:20 +02:00
} else {
return completion::PROGRESS;
}
2023-08-24 23:09:20 +02:00
} else {
2023-08-24 23:02:41 +02:00
// Blind assumptions:.
// over 55% of range is completed.
// over 85% of range is excellent.
$g = floatval($finalgrade - $gradeitem->grademin);
$range = floatval($gradeitem->grademax - $gradeitem->grademin);
$score = $g / $range;
2023-08-24 23:02:41 +02:00
if ($score > 0.85) {
return completion::EXCELLENT;
2023-08-24 23:09:20 +02:00
} else if ($score > 0.55) {
return completion::COMPLETED;
2023-08-24 23:09:20 +02:00
} else {
return completion::PROGRESS;
}
}
}
}
}