Merge branch 'MDL-60174-master' of https://github.com/sammarshallou/moodle
[moodle.git] / user / renderer.php
blobca0fe04d4075f7d8b059def0eab569e045c96861
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.
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/>.
17 /**
18 * Provides user rendering functionality such as printing private files tree and displaying a search utility
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
25 defined('MOODLE_INTERNAL') || die();
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
32 class core_user_renderer extends plugin_renderer_base {
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
45 public function user_search($url, $firstinitial, $lastinitial, $usercount, $totalcount, $heading = null) {
46 global $OUTPUT;
48 if ($firstinitial !== 'all') {
49 set_user_preference('ifirst', $firstinitial);
51 if ($lastinitial !== 'all') {
52 set_user_preference('ilast', $lastinitial);
55 if (!isset($heading)) {
56 $heading = get_string('allparticipants');
59 $content = html_writer::start_tag('form', array('action' => new moodle_url($url)));
60 $content .= html_writer::start_tag('div');
62 // Search utility heading.
63 $content .= $OUTPUT->heading($heading.get_string('labelsep', 'langconfig').$usercount.'/'.$totalcount, 3);
65 // Initials bar.
66 $prefixfirst = 'sifirst';
67 $prefixlast = 'silast';
68 $content .= $OUTPUT->initials_bar($firstinitial, 'firstinitial', get_string('firstname'), $prefixfirst, $url);
69 $content .= $OUTPUT->initials_bar($lastinitial, 'lastinitial', get_string('lastname'), $prefixlast, $url);
71 $content .= html_writer::end_tag('div');
72 $content .= html_writer::tag('div', '&nbsp;');
73 $content .= html_writer::end_tag('form');
75 return $content;
78 /**
79 * Displays the list of tagged users
81 * @param array $userlist
82 * @param bool $exclusivemode if set to true it means that no other entities tagged with this tag
83 * are displayed on the page and the per-page limit may be bigger
84 * @return string
86 public function user_list($userlist, $exclusivemode) {
87 $tagfeed = new core_tag\output\tagfeed();
88 foreach ($userlist as $user) {
89 $userpicture = $this->output->user_picture($user, array('size' => $exclusivemode ? 100 : 35));
90 $fullname = fullname($user);
91 if (user_can_view_profile($user)) {
92 $profilelink = new moodle_url('/user/view.php', array('id' => $user->id));
93 $fullname = html_writer::link($profilelink, $fullname);
95 $tagfeed->add($userpicture, $fullname);
98 $items = $tagfeed->export_for_template($this->output);
100 if ($exclusivemode) {
101 $output = '<div><ul class="inline-list">';
102 foreach ($items['items'] as $item) {
103 $output .= '<li><div class="user-box">'. $item['img'] . $item['heading'] ."</div></li>\n";
105 $output .= "</ul></div>\n";
106 return $output;
109 return $this->output->render_from_template('core_tag/tagfeed', $items);
113 * Renders the unified filter element for the course participants page.
115 * @param stdClass $course The course object.
116 * @param context $context The context object.
117 * @param array $filtersapplied Array of currently applied filters.
118 * @return bool|string
120 public function unified_filter($course, $context, $filtersapplied) {
121 global $CFG, $DB, $USER;
123 require_once($CFG->dirroot . '/enrol/locallib.php');
124 require_once($CFG->dirroot . '/lib/grouplib.php');
125 $manager = new course_enrolment_manager($this->page, $course);
127 $filteroptions = [];
129 // Filter options for role.
130 $roles = role_fix_names(get_profile_roles($context), $context, ROLENAME_ALIAS, true);
131 $criteria = get_string('role');
132 $roleoptions = [];
133 foreach ($roles as $id => $role) {
134 $roleoptions += $this->format_filter_option(USER_FILTER_ROLE, $criteria, $id, $role);
136 $filteroptions += $roleoptions;
138 // Filter options for groups, if available.
139 if (has_capability('moodle/site:accessallgroups', $context) || $course->groupmode != SEPARATEGROUPS) {
140 // List all groups if the user can access all groups, or we are in visible group mode or no groups mode.
141 $groups = $manager->get_all_groups();
142 } else {
143 // Otherwise, just list the groups the user belongs to.
144 $groups = groups_get_all_groups($course->id, $USER->id);
146 $criteria = get_string('group');
147 $groupoptions = [];
148 foreach ($groups as $id => $group) {
149 $groupoptions += $this->format_filter_option(USER_FILTER_GROUP, $criteria, $id, $group->name);
151 $filteroptions += $groupoptions;
153 $canreviewenrol = has_capability('moodle/course:enrolreview', $context);
155 // Filter options for status.
156 if ($canreviewenrol) {
157 $criteria = get_string('status');
158 // Add statuses.
159 $filteroptions += $this->format_filter_option(USER_FILTER_STATUS, $criteria, ENROL_USER_ACTIVE, get_string('active'));
160 $filteroptions += $this->format_filter_option(USER_FILTER_STATUS, $criteria, ENROL_USER_SUSPENDED,
161 get_string('inactive'));
164 // Filter options for enrolment methods.
165 if ($canreviewenrol && $enrolmentmethods = $manager->get_enrolment_instance_names(true)) {
166 $criteria = get_string('enrolmentinstances', 'enrol');
167 $enroloptions = [];
168 foreach ($enrolmentmethods as $id => $enrolname) {
169 $enroloptions += $this->format_filter_option(USER_FILTER_ENROLMENT, $criteria, $id, $enrolname);
171 $filteroptions += $enroloptions;
174 $isfrontpage = ($course->id == SITEID);
176 // Get the list of fields we have to hide.
177 $hiddenfields = array();
178 if (!has_capability('moodle/course:viewhiddenuserfields', $context)) {
179 $hiddenfields = array_flip(explode(',', $CFG->hiddenuserfields));
181 $haslastaccess = !isset($hiddenfields['lastaccess']);
182 // Filter options for last access.
183 if ($haslastaccess) {
184 // Get minimum lastaccess for this course and display a dropbox to filter by lastaccess going back this far.
185 // We need to make it diferently for normal courses and site course.
186 if (!$isfrontpage) {
187 $params = ['courseid' => $course->id, 'timeaccess' => 0];
188 $select = 'courseid = :courseid AND timeaccess != :timeaccess';
189 $minlastaccess = $DB->get_field_select('user_lastaccess', 'MIN(timeaccess)', $select, $params);
190 $lastaccess0exists = $DB->record_exists('user_lastaccess', $params);
191 } else {
192 $params = ['lastaccess' => 0];
193 $select = 'lastaccess != :lastaccess';
194 $minlastaccess = $DB->get_field_select('user', 'MIN(lastaccess)', $select, $params);
195 $lastaccess0exists = $DB->record_exists('user', $params);
197 $now = usergetmidnight(time());
198 $timeoptions = [];
199 $criteria = get_string('usersnoaccesssince');
201 // Days.
202 for ($i = 1; $i < 7; $i++) {
203 $timestamp = strtotime('-' . $i . ' days', $now);
204 if ($timestamp < $minlastaccess) {
205 break;
207 $value = get_string('numdays', 'moodle', $i);
208 $timeoptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $value);
210 // Weeks.
211 for ($i = 1; $i < 10; $i++) {
212 $timestamp = strtotime('-'.$i.' weeks', $now);
213 if ($timestamp < $minlastaccess) {
214 break;
216 $value = get_string('numweeks', 'moodle', $i);
217 $timeoptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $value);
219 // Months.
220 for ($i = 2; $i < 12; $i++) {
221 $timestamp = strtotime('-'.$i.' months', $now);
222 if ($timestamp < $minlastaccess) {
223 break;
225 $value = get_string('nummonths', 'moodle', $i);
226 $timeoptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $value);
228 // Try a year.
229 $timestamp = strtotime('-1 year', $now);
230 if ($timestamp >= $minlastaccess) {
231 $value = get_string('numyear', 'moodle', 1);
232 $timeoptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $value);
234 if (!empty($lastaccess0exists)) {
235 $value = get_string('never', 'moodle');
236 $timeoptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $value);
238 if (count($timeoptions) > 1) {
239 $filteroptions += $timeoptions;
243 // Add missing applied filters to the filter options.
244 $filteroptions = $this->handle_missing_applied_filters($filtersapplied, $filteroptions);
246 $indexpage = new \core_user\output\unified_filter($filteroptions, $filtersapplied);
247 $context = $indexpage->export_for_template($this->output);
249 return $this->output->render_from_template('core_user/unified_filter', $context);
253 * Returns a formatted filter option.
255 * @param int $filtertype The filter type (e.g. status, role, group, enrolment, last access).
256 * @param string $criteria The string label of the filter type.
257 * @param int $value The value for the filter option.
258 * @param string $label The string representation of the filter option's value.
259 * @return array The formatted option with the ['filtertype:value' => 'criteria: label'] format.
261 protected function format_filter_option($filtertype, $criteria, $value, $label) {
262 $optionlabel = get_string('filteroption', 'moodle', (object)['criteria' => $criteria, 'value' => $label]);
263 $optionvalue = "$filtertype:$value";
264 return [$optionvalue => $optionlabel];
268 * Handles cases when after reloading the applied filters are missing in the filter options.
270 * @param array $filtersapplied The applied filters.
271 * @param array $filteroptions The filter options.
272 * @return array The formatted options with the ['filtertype:value' => 'criteria: label'] format.
274 private function handle_missing_applied_filters($filtersapplied, $filteroptions) {
275 global $DB;
277 foreach ($filtersapplied as $filter) {
278 if (!array_key_exists($filter, $filteroptions)) {
279 $filtervalue = explode(':', $filter);
280 if (count($filtervalue) !== 2) {
281 continue;
283 $key = $filtervalue[0];
284 $value = $filtervalue[1];
286 switch($key) {
287 case USER_FILTER_LAST_ACCESS:
288 $now = usergetmidnight(time());
289 $criteria = get_string('usersnoaccesssince');
290 // Days.
291 for ($i = 1; $i < 7; $i++) {
292 $timestamp = strtotime('-' . $i . ' days', $now);
293 if ($timestamp < $value) {
294 break;
296 $val = get_string('numdays', 'moodle', $i);
297 $filteroptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $val);
299 // Weeks.
300 for ($i = 1; $i < 10; $i++) {
301 $timestamp = strtotime('-'.$i.' weeks', $now);
302 if ($timestamp < $value) {
303 break;
305 $val = get_string('numweeks', 'moodle', $i);
306 $filteroptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $val);
308 // Months.
309 for ($i = 2; $i < 12; $i++) {
310 $timestamp = strtotime('-'.$i.' months', $now);
311 if ($timestamp < $value) {
312 break;
314 $val = get_string('nummonths', 'moodle', $i);
315 $filteroptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $val);
317 // Try a year.
318 $timestamp = strtotime('-1 year', $now);
319 if ($timestamp >= $value) {
320 $val = get_string('numyear', 'moodle', 1);
321 $filteroptions += $this->format_filter_option(USER_FILTER_LAST_ACCESS, $criteria, $timestamp, $val);
323 break;
324 case USER_FILTER_ROLE:
325 $criteria = get_string('role');
326 if ($role = $DB->get_record('role', array('id' => $value))) {
327 $role = role_get_name($role);
328 $filteroptions += $this->format_filter_option(USER_FILTER_ROLE, $criteria, $value, $role);
330 break;
334 return $filteroptions;