You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
391 lines
12 KiB
391 lines
12 KiB
2 years ago
|
<?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/>.
|
||
|
|
||
|
/**
|
||
|
* The user screen.
|
||
|
*
|
||
|
* @package gradereport_singleview
|
||
|
* @copyright 2014 Moodle Pty Ltd (http://moodle.com)
|
||
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
||
|
*/
|
||
|
|
||
|
namespace gradereport_singleview\local\screen;
|
||
|
|
||
|
use grade_seq;
|
||
|
use gradereport_singleview;
|
||
|
use moodle_url;
|
||
|
use pix_icon;
|
||
|
use html_writer;
|
||
|
use gradereport_singleview\local\ui\range;
|
||
|
use gradereport_singleview\local\ui\bulk_insert;
|
||
|
use grade_item;
|
||
|
use grade_grade;
|
||
|
use stdClass;
|
||
|
|
||
|
defined('MOODLE_INTERNAL') || die;
|
||
|
|
||
|
/**
|
||
|
* The user screen.
|
||
|
*
|
||
|
* @package gradereport_singleview
|
||
|
* @copyright 2014 Moodle Pty Ltd (http://moodle.com)
|
||
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
||
|
*/
|
||
|
class user extends tablelike implements selectable_items {
|
||
|
|
||
|
/** @var array $categories A cache for grade_item categories */
|
||
|
private $categories = array();
|
||
|
|
||
|
/** @var int $requirespaging Do we have more items than the paging limit? */
|
||
|
private $requirespaging = true;
|
||
|
|
||
|
/**
|
||
|
* Get the label for the select box that chooses items for this page.
|
||
|
* @return string
|
||
|
*/
|
||
|
public function select_label() {
|
||
|
return get_string('selectgrade', 'gradereport_singleview');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the description for the screen.
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function description() {
|
||
|
return get_string('gradeitems', 'grades');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Convert the list of items to a list of options.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
public function options() {
|
||
|
$result = array();
|
||
|
foreach ($this->items as $itemid => $item) {
|
||
|
$result[$itemid] = $item->get_name();
|
||
|
}
|
||
|
return $result;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the type of items on this screen.
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function item_type() {
|
||
|
return 'grade';
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Init the screen
|
||
|
*
|
||
|
* @param bool $selfitemisempty Have we selected an item yet?
|
||
|
*/
|
||
|
public function init($selfitemisempty = false) {
|
||
|
global $DB;
|
||
|
|
||
|
if (!$selfitemisempty) {
|
||
|
$validusers = $this->load_users();
|
||
|
if (!isset($validusers[$this->itemid])) {
|
||
|
// If the passed user id is not valid, show the first user from the list instead.
|
||
|
$this->item = reset($validusers);
|
||
|
$this->itemid = $this->item->id;
|
||
|
} else {
|
||
|
$this->item = $validusers[$this->itemid];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$params = array('courseid' => $this->courseid);
|
||
|
|
||
|
$seq = new grade_seq($this->courseid, true);
|
||
|
|
||
|
$this->items = array();
|
||
|
foreach ($seq->items as $itemid => $item) {
|
||
|
if (grade::filter($item)) {
|
||
|
$this->items[$itemid] = $item;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$this->requirespaging = count($this->items) > $this->perpage;
|
||
|
|
||
|
$this->setup_structure();
|
||
|
|
||
|
$this->definition = array(
|
||
|
'finalgrade', 'feedback', 'override', 'exclude'
|
||
|
);
|
||
|
$this->set_headers($this->original_headers());
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the list of headers for the table.
|
||
|
*
|
||
|
* @return array List of headers
|
||
|
*/
|
||
|
public function original_headers() {
|
||
|
return array(
|
||
|
'', // For filter icon.
|
||
|
get_string('assessmentname', 'gradereport_singleview'),
|
||
|
get_string('gradecategory', 'grades'),
|
||
|
get_string('range', 'grades'),
|
||
|
get_string('grade', 'grades'),
|
||
|
get_string('feedback', 'grades'),
|
||
|
$this->make_toggle_links('override'),
|
||
|
$this->make_toggle_links('exclude')
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Format each row of the table.
|
||
|
*
|
||
|
* @param grade_item $item
|
||
|
* @return string
|
||
|
*/
|
||
|
public function format_line($item) {
|
||
|
global $OUTPUT;
|
||
|
|
||
|
$grade = $this->fetch_grade_or_default($item, $this->item->id);
|
||
|
$lockicon = '';
|
||
|
|
||
|
$lockeditem = $lockeditemgrade = 0;
|
||
|
if (!empty($grade->locked)) {
|
||
|
$lockeditem = 1;
|
||
|
}
|
||
|
if (!empty($grade->grade_item->locked)) {
|
||
|
$lockeditemgrade = 1;
|
||
|
}
|
||
|
// Check both grade and grade item.
|
||
|
if ($lockeditem || $lockeditemgrade) {
|
||
|
$lockicon = $OUTPUT->pix_icon('t/locked', 'grade is locked');
|
||
|
}
|
||
|
|
||
|
$iconstring = get_string('filtergrades', 'gradereport_singleview', $item->get_name());
|
||
|
|
||
|
// Create a fake gradetreeitem so we can call get_element_header().
|
||
|
// The type logic below is from grade_category->_get_children_recursion().
|
||
|
$gradetreeitem = array();
|
||
|
if (in_array($item->itemtype, array('course', 'category'))) {
|
||
|
$gradetreeitem['type'] = $item->itemtype.'item';
|
||
|
} else {
|
||
|
$gradetreeitem['type'] = 'item';
|
||
|
}
|
||
|
$gradetreeitem['object'] = $item;
|
||
|
$gradetreeitem['userid'] = $this->item->id;
|
||
|
|
||
|
$itemlabel = $this->structure->get_element_header($gradetreeitem, true, false, false, false, true);
|
||
|
$grade->label = $item->get_name();
|
||
|
|
||
|
$line = array(
|
||
|
$OUTPUT->action_icon($this->format_link('grade', $item->id), new pix_icon('t/editstring', $iconstring)),
|
||
|
$this->format_icon($item) . $lockicon . $itemlabel,
|
||
|
$this->category($item),
|
||
|
new range($item)
|
||
|
);
|
||
|
$lineclasses = array(
|
||
|
"action",
|
||
|
"gradeitem",
|
||
|
"category",
|
||
|
"range"
|
||
|
);
|
||
|
|
||
|
$outputline = array();
|
||
|
$i = 0;
|
||
|
foreach ($line as $key => $value) {
|
||
|
$cell = new \html_table_cell($value);
|
||
|
if ($isheader = $i == 1) {
|
||
|
$cell->header = $isheader;
|
||
|
$cell->scope = "row";
|
||
|
}
|
||
|
if (array_key_exists($key, $lineclasses)) {
|
||
|
$cell->attributes['class'] = $lineclasses[$key];
|
||
|
}
|
||
|
$outputline[] = $cell;
|
||
|
$i++;
|
||
|
}
|
||
|
|
||
|
return $this->format_definition($outputline, $grade);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Helper to get the icon for an item.
|
||
|
*
|
||
|
* @param grade_item $item
|
||
|
* @return string
|
||
|
*/
|
||
|
private function format_icon($item) {
|
||
|
$element = array('type' => 'item', 'object' => $item);
|
||
|
return $this->structure->get_element_icon($element);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Helper to get the category for an item.
|
||
|
*
|
||
|
* @param grade_item $item
|
||
|
* @return grade_category
|
||
|
*/
|
||
|
private function category($item) {
|
||
|
global $DB;
|
||
|
|
||
|
if (empty($item->categoryid)) {
|
||
|
|
||
|
if ($item->itemtype == 'course') {
|
||
|
return $this->course->fullname;
|
||
|
}
|
||
|
|
||
|
$params = array('id' => $item->iteminstance);
|
||
|
$elem = $DB->get_record('grade_categories', $params);
|
||
|
|
||
|
return $elem->fullname;
|
||
|
}
|
||
|
|
||
|
if (!isset($this->categories[$item->categoryid])) {
|
||
|
$category = $item->get_parent_category();
|
||
|
|
||
|
$this->categories[$category->id] = $category;
|
||
|
}
|
||
|
|
||
|
return $this->categories[$item->categoryid]->get_name();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the heading for the page.
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function heading() {
|
||
|
return get_string('gradeuser', 'gradereport_singleview', fullname($this->item));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the summary for this table.
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function summary() {
|
||
|
return get_string('summaryuser', 'gradereport_singleview');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Default pager
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function pager() {
|
||
|
global $OUTPUT;
|
||
|
|
||
|
if (!$this->supports_paging()) {
|
||
|
return '';
|
||
|
}
|
||
|
|
||
|
return $OUTPUT->paging_bar(
|
||
|
count($this->items), $this->page, $this->perpage,
|
||
|
new moodle_url('/grade/report/singleview/index.php', array(
|
||
|
'perpage' => $this->perpage,
|
||
|
'id' => $this->courseid,
|
||
|
'group' => $this->groupid,
|
||
|
'itemid' => $this->itemid,
|
||
|
'item' => 'user'
|
||
|
))
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Does this page require paging?
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function supports_paging() {
|
||
|
return $this->requirespaging;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Process the data from the form.
|
||
|
*
|
||
|
* @param array $data
|
||
|
* @return array of warnings
|
||
|
*/
|
||
|
public function process($data) {
|
||
|
$bulk = new bulk_insert($this->item);
|
||
|
// Bulk insert messages the data to be passed in
|
||
|
// ie: for all grades of empty grades apply the specified value.
|
||
|
if ($bulk->is_applied($data)) {
|
||
|
$filter = $bulk->get_type($data);
|
||
|
$insertvalue = $bulk->get_insert_value($data);
|
||
|
|
||
|
$userid = $this->item->id;
|
||
|
foreach ($this->items as $gradeitemid => $gradeitem) {
|
||
|
$null = $gradeitem->gradetype == GRADE_TYPE_SCALE ? -1 : '';
|
||
|
$field = "finalgrade_{$gradeitem->id}_{$this->itemid}";
|
||
|
if (isset($data->$field)) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$oldfinalgradefield = "oldfinalgrade_{$gradeitem->id}_{$this->itemid}";
|
||
|
// Bulk grade changes for all grades need to be processed and shouldn't be skipped if they had a previous grade.
|
||
|
if ($gradeitem->is_course_item() || ($filter != 'all' && !empty($data->$oldfinalgradefield))) {
|
||
|
if ($gradeitem->is_course_item()) {
|
||
|
// The course total should not be overridden.
|
||
|
unset($data->$field);
|
||
|
unset($data->oldfinalgradefield);
|
||
|
$oldoverride = "oldoverride_{$gradeitem->id}_{$this->itemid}";
|
||
|
unset($data->$oldoverride);
|
||
|
$oldfeedback = "oldfeedback_{$gradeitem->id}_{$this->itemid}";
|
||
|
unset($data->$oldfeedback);
|
||
|
}
|
||
|
continue;
|
||
|
}
|
||
|
$grade = grade_grade::fetch(array(
|
||
|
'itemid' => $gradeitemid,
|
||
|
'userid' => $userid
|
||
|
));
|
||
|
|
||
|
$data->$field = empty($grade) ? $null : $grade->finalgrade;
|
||
|
$data->{"old$field"} = $data->$field;
|
||
|
}
|
||
|
|
||
|
foreach ($data as $varname => $value) {
|
||
|
if (preg_match('/^oldoverride_(\d+)_(\d+)/', $varname, $matches)) {
|
||
|
// If we've selected overriding all grades.
|
||
|
if ($filter == 'all') {
|
||
|
$override = "override_{$matches[1]}_{$matches[2]}";
|
||
|
$data->$override = '1';
|
||
|
}
|
||
|
}
|
||
|
if (!preg_match('/^finalgrade_(\d+)_(\d+)/', $varname, $matches)) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$gradeitem = grade_item::fetch(array(
|
||
|
'courseid' => $this->courseid,
|
||
|
'id' => $matches[1]
|
||
|
));
|
||
|
|
||
|
$isscale = ($gradeitem->gradetype == GRADE_TYPE_SCALE);
|
||
|
|
||
|
$empties = (trim($value) === '' or ($isscale and $value == -1));
|
||
|
|
||
|
if ($filter == 'all' or $empties) {
|
||
|
$data->$varname = ($isscale and empty($insertvalue)) ?
|
||
|
-1 : $insertvalue;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return parent::process($data);
|
||
|
}
|
||
|
}
|