See Release Notes
Long Term Support Release
Differences Between: [Versions 39 and 401] [Versions 39 and 402] [Versions 39 and 403]
1 <?php 2 // This file is part of Moodle - http://moodle.org/ 3 // 4 // Moodle is free software: you can redistribute it and/or modify 5 // it under the terms of the GNU General Public License as published by 6 // the Free Software Foundation, either version 3 of the License, or 7 // (at your option) any later version. 8 // 9 // Moodle is distributed in the hope that it will be useful, 10 // but WITHOUT ANY WARRANTY; without even the implied warranty of 11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 // GNU General Public License for more details. 13 // 14 // You should have received a copy of the GNU General Public License 15 // along with Moodle. If not, see <http://www.gnu.org/licenses/>. 16 17 /** 18 * Provides user rendering functionality such as printing private files tree and displaying a search utility 19 * 20 * @package core_user 21 * @copyright 2010 Dongsheng Cai <dongsheng@moodle.com> 22 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later 23 */ 24 25 defined('MOODLE_INTERNAL') || die(); 26 27 /** 28 * Provides user rendering functionality such as printing private files tree and displaying a search utility 29 * @copyright 2010 Dongsheng Cai <dongsheng@moodle.com> 30 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later 31 */ 32 class core_user_renderer extends plugin_renderer_base { 33 34 /** 35 * Prints user search utility that can search user by first initial of firstname and/or first initial of lastname 36 * Prints a header with a title and the number of users found within that subset 37 * @param string $url the url to return to, complete with any parameters needed for the return 38 * @param string $firstinitial the first initial of the firstname 39 * @param string $lastinitial the first initial of the lastname 40 * @param int $usercount the amount of users meeting the search criteria 41 * @param int $totalcount the amount of users of the set/subset being searched 42 * @param string $heading heading of the subset being searched, default is All Participants 43 * @return string html output 44 */ 45 public function user_search($url, $firstinitial, $lastinitial, $usercount, $totalcount, $heading = null) { 46 47 if ($firstinitial !== 'all') { 48 set_user_preference('ifirst', $firstinitial); 49 } 50 if ($lastinitial !== 'all') { 51 set_user_preference('ilast', $lastinitial); 52 } 53 54 if (!isset($heading)) { 55 $heading = get_string('allparticipants'); 56 } 57 58 $content = html_writer::start_tag('form', array('action' => new moodle_url($url))); 59 $content .= html_writer::start_tag('div'); 60 61 // Search utility heading. 62 $content .= $this->output->heading($heading.get_string('labelsep', 'langconfig').$usercount.'/'.$totalcount, 3); 63 64 // Initials bar. 65 $prefixfirst = 'sifirst'; 66 $prefixlast = 'silast'; 67 $content .= $this->output->initials_bar($firstinitial, 'firstinitial', get_string('firstname'), $prefixfirst, $url); 68 $content .= $this->output->initials_bar($lastinitial, 'lastinitial', get_string('lastname'), $prefixlast, $url); 69 70 $content .= html_writer::end_tag('div'); 71 $content .= html_writer::tag('div', ' '); 72 $content .= html_writer::end_tag('form'); 73 74 return $content; 75 } 76 77 /** 78 * Displays the list of tagged users 79 * 80 * @param array $userlist 81 * @param bool $exclusivemode if set to true it means that no other entities tagged with this tag 82 * are displayed on the page and the per-page limit may be bigger 83 * @return string 84 */ 85 public function user_list($userlist, $exclusivemode) { 86 $tagfeed = new core_tag\output\tagfeed(); 87 foreach ($userlist as $user) { 88 $userpicture = $this->output->user_picture($user, array('size' => $exclusivemode ? 100 : 35)); 89 $fullname = fullname($user); 90 if (user_can_view_profile($user)) { 91 $profilelink = new moodle_url('/user/view.php', array('id' => $user->id)); 92 $fullname = html_writer::link($profilelink, $fullname); 93 } 94 $tagfeed->add($userpicture, $fullname); 95 } 96 97 $items = $tagfeed->export_for_template($this->output); 98 99 if ($exclusivemode) { 100 $output = '<div><ul class="inline-list">'; 101 foreach ($items['items'] as $item) { 102 $output .= '<li><div class="user-box">'. $item['img'] . $item['heading'] ."</div></li>\n"; 103 } 104 $output .= "</ul></div>\n"; 105 return $output; 106 } 107 108 return $this->output->render_from_template('core_tag/tagfeed', $items); 109 } 110 111 /** 112 * Renders the unified filter element for the course participants page. 113 * @deprecated since Moodle 3.9 MDL-68612 - Please use participants_filter() instead. 114 * 115 * @param stdClass $course The course object. 116 * @param context $context The context object. 117 * @param array $filtersapplied Array of currently applied filters. 118 * @param string|moodle_url $baseurl The url with params needed to call up this page. 119 * @return bool|string 120 */ 121 public function unified_filter($course, $context, $filtersapplied, $baseurl = null) { 122 global $CFG, $DB, $USER; 123 124 debugging('core_user_renderer->unified_filter() is deprecated. Please use participants_filter() instead.', DEBUG_DEVELOPER); 125 126 require_once($CFG->dirroot . '/enrol/locallib.php'); 127 require_once($CFG->dirroot . '/lib/grouplib.php'); 128 $manager = new course_enrolment_manager($this->page, $course); 129 130 $filteroptions = []; 131 132 // Filter options for role. 133 $roleseditable = has_capability('moodle/role:assign', $context); 134 $roles = get_viewable_roles($context); 135 if ($roleseditable) { 136 $roles += get_assignable_roles($context, ROLENAME_ALIAS); 137 } 138 139 $criteria = get_string('role'); 140 $roleoptions = $this->format_filter_option(USER_FILTER_ROLE, $criteria, -1, get_string('noroles', 'role')); 141 foreach ($roles as $id => $role) { 142 $roleoptions += $this->format_filter_option(USER_FILTER_ROLE, $criteria, $id, $role); 143 } 144 $filteroptions += $roleoptions; 145 146 // Filter options for groups, if available. 147 if (has_capability('moodle/site:accessallgroups', $context) || $course->groupmode != SEPARATEGROUPS) { 148 // List all groups if the user can access all groups, or we are in visible group mode or no groups mode. 149 $groups = $manager->get_all_groups(); 150 if (!empty($groups)) { 151 // Add 'No group' option, to enable filtering users without any group. 152 $nogroup[USERSWITHOUTGROUP] = (object)['name' => get_string('nogroup', 'group')]; 153 $groups = $nogroup + $groups; 154 } 155 } else { 156 // Otherwise, just list the groups the user belongs to. 157 $groups = groups_get_all_groups($course->id, $USER->id); 158 } 159 $criteria = get_string('group'); 160 $groupoptions = []; 161 foreach ($groups as $id => $group) { 162 $groupoptions += $this->format_filter_option(USER_FILTER_GROUP, $criteria, $id, $group->name); 163 } 164 $filteroptions += $groupoptions; 165 166 $canreviewenrol = has_capability('moodle/course:enrolreview', $context); 167 168 // Filter options for status. 169 if ($canreviewenrol) { 170 $criteria = get_string('status'); 171 // Add statuses. 172 $filteroptions += $this->format_filter_option(USER_FILTER_STATUS, $criteria, ENROL_USER_ACTIVE, get_string('active')); 173 $filteroptions += $this->format_filter_option(USER_FILTER_STATUS, $criteria, ENROL_USER_SUSPENDED, 174 get_string('inactive')); 175 } 176 177 // Filter options for enrolment methods. 178 if ($canreviewenrol && $enrolmentmethods = $manager->get_enrolment_instance_names(true)) { 179 $criteria = get_string('enrolmentinstances', 'enrol'); 180 $enroloptions = []; 181 foreach ($enrolmentmethods as $id => $enrolname) { 182 $enroloptions += $this->format_filter_option(USER_FILTER_ENROLMENT, $criteria, $id, $enrolname); 183 } 184 $filteroptions += $enroloptions; 185 } 186 187 $isfrontpage = ($course->id == SITEID); 188 189 // Get the list of fields we have to hide. 190 $hiddenfields = array(); 191 if (!has_capability('moodle/course:viewhiddenuserfields', $context)) { 192 $hiddenfields = array_flip(explode(',', $CFG->hiddenuserfields)); 193 } 194 $haslastaccess = !isset($hiddenfields['lastaccess']); 195 // Filter options for last access. 196 if ($haslastaccess) { 197 // Get minimum lastaccess for this course and display a dropbox to filter by lastaccess going back this far. 198 // We need to make it diferently for normal courses and site course. 199 if (!$isfrontpage) { 200 $params = ['courseid' => $course->id, 'timeaccess' => 0]; 201 $select = 'courseid = :courseid AND timeaccess != :timeaccess'; 202 $minlastaccess = $DB->get_field_select('user_lastaccess', 'MIN(timeaccess)', $select, $params); 203 $lastaccess0exists = $DB->record_exists('user_lastaccess', $params); 204 } else { 205 $params = ['lastaccess' => 0]; 206 $select = 'lastaccess != :lastaccess'; 207 $minlastaccess = $DB->get_field_select('user', 'MIN(lastaccess)', $select, $params); 208 $lastaccess0exists = $DB->record_exists('user', $params); 209 } 210 $now = usergetmidnight(time()); 211 $timeoptions = []; 212 $criteria = get_string('usersnoaccesssince'); 213 214 // Days. 215 for ($i = 1; $i < 7; $i++) { 216 $timestamp = strtotime('-' . $i . ' days', $now); 217 if ($timestamp < $minlastaccess) { 218 break; 219 } 220 $value = get_string('numdays', 'moodle', $i); 221 $timeoptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $value); 222 } 223 // Weeks. 224 for ($i = 1; $i < 10; $i++) { 225 $timestamp = strtotime('-'.$i.' weeks', $now); 226 if ($timestamp < $minlastaccess) { 227 break; 228 } 229 $value = get_string('numweeks', 'moodle', $i); 230 $timeoptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $value); 231 } 232 // Months. 233 for ($i = 2; $i < 12; $i++) { 234 $timestamp = strtotime('-'.$i.' months', $now); 235 if ($timestamp < $minlastaccess) { 236 break; 237 } 238 $value = get_string('nummonths', 'moodle', $i); 239 $timeoptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $value); 240 } 241 // Try a year. 242 $timestamp = strtotime('-1 year', $now); 243 if ($timestamp >= $minlastaccess) { 244 $value = get_string('numyear', 'moodle', 1); 245 $timeoptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $value); 246 } 247 if (!empty($lastaccess0exists)) { 248 $value = get_string('never', 'moodle'); 249 $timeoptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $value); 250 } 251 if (count($timeoptions) > 1) { 252 $filteroptions += $timeoptions; 253 } 254 } 255 256 // Add missing applied filters to the filter options. 257 $filteroptions = $this->handle_missing_applied_filters($filtersapplied, $filteroptions); 258 259 $indexpage = new \core_user\output\unified_filter($filteroptions, $filtersapplied, $baseurl); 260 $context = $indexpage->export_for_template($this->output); 261 262 return $this->output->render_from_template('core_user/unified_filter', $context); 263 } 264 265 /** 266 * Render the data required for the participants filter on the course participants page. 267 * 268 * @param context $context The context of the course being displayed 269 * @param string $tableregionid The table to be updated by this filter 270 * @return string 271 */ 272 public function participants_filter(context $context, string $tableregionid): string { 273 $renderable = new \core_user\output\participants_filter($context, $tableregionid); 274 $templatecontext = $renderable->export_for_template($this->output); 275 276 return $this->output->render_from_template('core_user/participantsfilter', $templatecontext); 277 } 278 279 /** 280 * Returns a formatted filter option. 281 * 282 * @param int $filtertype The filter type (e.g. status, role, group, enrolment, last access). 283 * @param string $criteria The string label of the filter type. 284 * @param int $value The value for the filter option. 285 * @param string $label The string representation of the filter option's value. 286 * @return array The formatted option with the ['filtertype:value' => 'criteria: label'] format. 287 */ 288 protected function format_filter_option($filtertype, $criteria, $value, $label) { 289 $optionlabel = get_string('filteroption', 'moodle', (object)['criteria' => $criteria, 'value' => $label]); 290 $optionvalue = "$filtertype:$value"; 291 return [$optionvalue => $optionlabel]; 292 } 293 294 /** 295 * Handles cases when after reloading the applied filters are missing in the filter options. 296 * 297 * @param array $filtersapplied The applied filters. 298 * @param array $filteroptions The filter options. 299 * @return array The formatted options with the ['filtertype:value' => 'criteria: label'] format. 300 */ 301 private function handle_missing_applied_filters($filtersapplied, $filteroptions) { 302 global $DB; 303 304 foreach ($filtersapplied as $filter) { 305 if (!array_key_exists($filter, $filteroptions)) { 306 $filtervalue = explode(':', $filter); 307 if (count($filtervalue) !== 2) { 308 continue; 309 } 310 $key = $filtervalue[0]; 311 $value = $filtervalue[1]; 312 313 switch($key) { 314 case USER_FILTER_LAST_ACCESS: 315 $now = usergetmidnight(time()); 316 $criteria = get_string('usersnoaccesssince'); 317 // Days. 318 for ($i = 1; $i < 7; $i++) { 319 $timestamp = strtotime('-' . $i . ' days', $now); 320 if ($timestamp < $value) { 321 break; 322 } 323 $val = get_string('numdays', 'moodle', $i); 324 $filteroptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $val); 325 } 326 // Weeks. 327 for ($i = 1; $i < 10; $i++) { 328 $timestamp = strtotime('-'.$i.' weeks', $now); 329 if ($timestamp < $value) { 330 break; 331 } 332 $val = get_string('numweeks', 'moodle', $i); 333 $filteroptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $val); 334 } 335 // Months. 336 for ($i = 2; $i < 12; $i++) { 337 $timestamp = strtotime('-'.$i.' months', $now); 338 if ($timestamp < $value) { 339 break; 340 } 341 $val = get_string('nummonths', 'moodle', $i); 342 $filteroptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $val); 343 } 344 // Try a year. 345 $timestamp = strtotime('-1 year', $now); 346 if ($timestamp >= $value) { 347 $val = get_string('numyear', 'moodle', 1); 348 $filteroptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $val); 349 } 350 break; 351 case USER_FILTER_ROLE: 352 $criteria = get_string('role'); 353 if ($role = $DB->get_record('role', array('id' => $value))) { 354 $role = role_get_name($role); 355 $filteroptions += $this->format_filter_option(USER_FILTER_ROLE, $criteria, $value, $role); 356 } 357 break; 358 } 359 } 360 } 361 return $filteroptions; 362 } 363 }
title
Description
Body
title
Description
Body
title
Description
Body
title
Body