See Release Notes
Long Term Support Release
<?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/>. /** * Tour helper. * * @package tool_usertours * @copyright 2016 Andrew Nicols <andrew@nicols.co.uk> * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ namespace tool_usertours;> use core\output\inplace_editable;use tool_usertours\local\clientside_filter\clientside_filter; defined('MOODLE_INTERNAL') || die(); /** * Tour helper. * * @copyright 2016 Andrew Nicols <andrew@nicols.co.uk> * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class helper { /** * @var MOVE_UP */ const MOVE_UP = -1; /** * @var MOVE_DOWN */ const MOVE_DOWN = 1; /** * @var boolean Has it been bootstrapped? */ private static $bootstrapped = false; /**> * @var string Regex to check any matching lang string. * Get the link to edit the step. > */ * > protected const LANG_STRING_REGEX = '|^([a-zA-Z][a-zA-Z0-9\.:/_-]*),([a-zA-Z][a-zA-Z0-9\.:/_-]*)$|'; * If no stepid is specified, then a link to create a new step is provided. The $targettype must be specified in this case. > * > /*** @param int $tourid The tour that the step belongs to. * @param int $stepid The step ID. * @param int $targettype The type of step. * * @return moodle_url */ public static function get_edit_step_link($tourid, $stepid = null, $targettype = null) { $link = new \moodle_url('/admin/tool/usertours/configure.php'); if ($stepid) { $link->param('action', manager::ACTION_EDITSTEP); $link->param('id', $stepid); } else { $link->param('action', manager::ACTION_NEWSTEP); $link->param('tourid', $tourid); } return $link; } /** * Get the link to move the tour. * * @param int $tourid The tour ID. * @param int $direction The direction to move in * * @return moodle_url */ public static function get_move_tour_link($tourid, $direction = self::MOVE_DOWN) { $link = new \moodle_url('/admin/tool/usertours/configure.php'); $link->param('action', manager::ACTION_MOVETOUR); $link->param('id', $tourid); $link->param('direction', $direction); $link->param('sesskey', sesskey()); return $link; } /** * Get the link to move the step. * * @param int $stepid The step ID. * @param int $direction The direction to move in * * @return moodle_url */ public static function get_move_step_link($stepid, $direction = self::MOVE_DOWN) { $link = new \moodle_url('/admin/tool/usertours/configure.php'); $link->param('action', manager::ACTION_MOVESTEP); $link->param('id', $stepid); $link->param('direction', $direction); $link->param('sesskey', sesskey()); return $link; } /** * Get the link ot create a new step. * * @param int $tourid The ID of the tour to attach this step to. * @param int $targettype The type of target. * * @return moodle_url The required URL. */ public static function get_new_step_link($tourid, $targettype = null) { $link = new \moodle_url('/admin/tool/usertours/configure.php'); $link->param('action', manager::ACTION_NEWSTEP); $link->param('tourid', $tourid); $link->param('targettype', $targettype); return $link; } /** * Get the link used to view the tour. * * @param int $tourid The ID of the tour to display. * @return moodle_url The URL. */ public static function get_view_tour_link($tourid) { return new \moodle_url('/admin/tool/usertours/configure.php', [ 'id' => $tourid, 'action' => manager::ACTION_VIEWTOUR, ]); } /** * Get the link used to reset the tour state for all users. * * @param int $tourid The ID of the tour to display. * @return moodle_url The URL. */ public static function get_reset_tour_for_all_link($tourid) { return new \moodle_url('/admin/tool/usertours/configure.php', [ 'id' => $tourid, 'action' => manager::ACTION_RESETFORALL, 'sesskey' => sesskey(), ]); } /** * Get the link used to edit the tour. * * @param int $tourid The ID of the tour to edit. * @return moodle_url The URL. */ public static function get_edit_tour_link($tourid = null) { $link = new \moodle_url('/admin/tool/usertours/configure.php'); if ($tourid) { $link->param('action', manager::ACTION_EDITTOUR); $link->param('id', $tourid); } else { $link->param('action', manager::ACTION_NEWTOUR); } return $link; } /** * Get the link used to import the tour. * * @return moodle_url The URL. */ public static function get_import_tour_link() { $link = new \moodle_url('/admin/tool/usertours/configure.php', [ 'action' => manager::ACTION_IMPORTTOUR, ]); return $link; } /** * Get the link used to export the tour. * * @param int $tourid The ID of the tour to export. * @return moodle_url The URL. */ public static function get_export_tour_link($tourid) { $link = new \moodle_url('/admin/tool/usertours/configure.php', [ 'action' => manager::ACTION_EXPORTTOUR, 'id' => $tourid, ]); return $link; } /** * Get the link used to duplicate the tour. * * @param int $tourid The ID of the tour to duplicate. * @return moodle_url The URL. */ public static function get_duplicate_tour_link($tourid) { $link = new \moodle_url('/admin/tool/usertours/configure.php', [ 'action' => manager::ACTION_DUPLICATETOUR, 'id' => $tourid, ]); return $link; } /** * Get the link used to delete the tour. * * @param int $tourid The ID of the tour to delete. * @return moodle_url The URL. */ public static function get_delete_tour_link($tourid) { return new \moodle_url('/admin/tool/usertours/configure.php', [ 'id' => $tourid, 'action' => manager::ACTION_DELETETOUR, 'sesskey' => sesskey(), ]); } /** * Get the link for listing tours. * * @return moodle_url The URL. */ public static function get_list_tour_link() { $link = new \moodle_url('/admin/tool/usertours/configure.php'); $link->param('action', manager::ACTION_LISTTOURS); return $link; } /** * Get a filler icon for display in the actions column of a table. * * @param string $url The URL for the icon. * @param string $icon The icon identifier. * @param string $alt The alt text for the icon. * @param string $iconcomponent The icon component. * @param array $options Display options. * @return string */ public static function format_icon_link($url, $icon, $alt, $iconcomponent = 'moodle', $options = array()) { global $OUTPUT; return $OUTPUT->action_icon( $url, new \pix_icon($icon, $alt, $iconcomponent, [ 'title' => $alt, ]), null, $options ); } /** * Get a filler icon for display in the actions column of a table. * * @param array $options Display options. * @return string */ public static function get_filler_icon($options = array()) { global $OUTPUT; return \html_writer::span( $OUTPUT->pix_icon('t/filler', '', 'tool_usertours', $options), 'action-icon' ); } /** * Get the link for deleting steps. * * @param int $stepid The ID of the step to display. * @return moodle_url The URL. */ public static function get_delete_step_link($stepid) { return new \moodle_url('/admin/tool/usertours/configure.php', [ 'action' => manager::ACTION_DELETESTEP, 'id' => $stepid, 'sesskey' => sesskey(), ]); } /** * Render the inplace editable used to edit the tour name. * * @param tour $tour The tour to edit.< * @return string> * @return inplace_editable*/< public static function render_tourname_inplace_editable(tour $tour) { < return new \core\output\inplace_editable(> public static function render_tourname_inplace_editable(tour $tour): inplace_editable { > $name = format_text(static::get_string_from_input($tour->get_name()), FORMAT_HTML); > return new inplace_editable('tool_usertours', 'tourname', $tour->get_id(), true, \html_writer::link( $tour->get_view_link(),< $tour->get_name()> $name), $tour->get_name() ); } /** * Render the inplace editable used to edit the tour description. * * @param tour $tour The tour to edit.< * @return string> * @return inplace_editable*/< public static function render_tourdescription_inplace_editable(tour $tour) { < return new \core\output\inplace_editable(> public static function render_tourdescription_inplace_editable(tour $tour): inplace_editable { > $description = format_text(static::get_string_from_input($tour->get_description()), FORMAT_HTML); > return new inplace_editable('tool_usertours', 'tourdescription', $tour->get_id(), true,< $tour->get_description(),> $description,$tour->get_description() ); } /** * Render the inplace editable used to edit the tour enable state. * * @param tour $tour The tour to edit.< * @return string> * @return inplace_editable*/< public static function render_tourenabled_inplace_editable(tour $tour) {> public static function render_tourenabled_inplace_editable(tour $tour): inplace_editable {global $OUTPUT; if ($tour->is_enabled()) { $icon = 't/hide'; $alt = get_string('disable'); $value = 1; } else { $icon = 't/show'; $alt = get_string('enable'); $value = 0; }< $editable = new \core\output\inplace_editable(> $editable = new inplace_editable('tool_usertours', 'tourenabled', $tour->get_id(), true, $OUTPUT->pix_icon($icon, $alt, 'moodle', [ 'title' => $alt, ]), $value ); $editable->set_type_toggle(); return $editable; } /** * Render the inplace editable used to edit the step name. * * @param step $step The step to edit.< * @return string> * @return inplace_editable*/< public static function render_stepname_inplace_editable(step $step) { < $title = format_text(step::get_string_from_input($step->get_title()), FORMAT_HTML);> public static function render_stepname_inplace_editable(step $step): inplace_editable { > $title = format_text(static::get_string_from_input($step->get_title()), FORMAT_HTML);< return new \core\output\inplace_editable(> return new inplace_editable('tool_usertours', 'stepname', $step->get_id(), true, \html_writer::link( $step->get_edit_link(), $title ), $step->get_title() ); } /** * Get all of the tours. * * @return stdClass[] */ public static function get_tours() { global $DB; $tours = $DB->get_records('tool_usertours_tours', array(), 'sortorder ASC'); $return = []; foreach ($tours as $tour) { $return[$tour->id] = tour::load_from_record($tour); } return $return; } /** * Get the specified tour. * * @param int $tourid The tour that the step belongs to.< * @return stdClass> * @return tour*/ public static function get_tour($tourid) { return tour::instance($tourid); } /** * Fetch the tour with the specified sortorder. * * @param int $sortorder The sortorder of the tour. * @return tour */ public static function get_tour_from_sortorder($sortorder) { global $DB; $tour = $DB->get_record('tool_usertours_tours', array('sortorder' => $sortorder)); return tour::load_from_record($tour); } /** * Return the count of all tours. * * @return int */ public static function count_tours() { global $DB; return $DB->count_records('tool_usertours_tours'); } /** * Reset the sortorder for all tours. */ public static function reset_tour_sortorder() { global $DB; $tours = $DB->get_records('tool_usertours_tours', null, 'sortorder ASC, pathmatch DESC', 'id, sortorder'); $index = 0; foreach ($tours as $tour) { if ($tour->sortorder != $index) { $DB->set_field('tool_usertours_tours', 'sortorder', $index, array('id' => $tour->id)); } $index++; } // Notify the cache that a tour has changed. // Tours are only stored in the cache if there are steps. // If there step count has changed for some reason, this will change the potential cache results. cache::notify_tour_change(); } /** * Get all of the steps in the tour. * * @param int $tourid The tour that the step belongs to.< * @return stdClass[]> * @return step[]*/ public static function get_steps($tourid) { $steps = cache::get_stepdata($tourid); $return = []; foreach ($steps as $step) { $return[$step->id] = step::load_from_record($step); } return $return; } /** * Fetch the specified step. * * @param int $stepid The id of the step to fetch. * @return step */ public static function get_step($stepid) { return step::instance($stepid); } /** * Fetch the step with the specified sortorder. * * @param int $tourid The tour that the step belongs to. * @param int $sortorder The sortorder of the step. * @return step */ public static function get_step_from_sortorder($tourid, $sortorder) { global $DB; $step = $DB->get_record('tool_usertours_steps', array('tourid' => $tourid, 'sortorder' => $sortorder)); return step::load_from_record($step); } /** * Handle addition of the tour into the current page. */ public static function bootstrap() { global $PAGE; if (!isloggedin() || isguestuser()) { return; } if (in_array($PAGE->pagelayout, ['maintenance', 'print', 'redirect'])) { // Do not try to show user tours inside iframe, in maintenance mode, // when printing, or during redirects. return; } if (self::$bootstrapped) { return; } self::$bootstrapped = true; $tours = manager::get_current_tours(); if ($tours) { $filters = static::get_all_clientside_filters(); $tourdetails = array_map(function($tour) use ($filters) { return [ 'tourId' => $tour->get_id(), 'startTour' => $tour->should_show_for_user(), 'filtervalues' => $tour->get_client_filter_values($filters), ]; }, $tours); $filternames = []; foreach ($filters as $filter) { $filternames[] = $filter::get_filter_name(); } $PAGE->requires->js_call_amd('tool_usertours/usertours', 'init', [ $tourdetails, $filternames, ]); } } /** * Get a list of all possible filters. * * @return array */ public static function get_all_filters() { $filters = \core_component::get_component_classes_in_namespace('tool_usertours', 'local\filter'); $filters = array_keys($filters); $filters = array_filter($filters, function($filterclass) { $rc = new \ReflectionClass($filterclass); return $rc->isInstantiable(); }); $filters = array_merge($filters, static::get_all_clientside_filters()); return $filters; } /** * Get a list of all clientside filters. * * @return array */ public static function get_all_clientside_filters() { $filters = \core_component::get_component_classes_in_namespace('tool_usertours', 'local\clientside_filter'); $filters = array_keys($filters); $filters = array_filter($filters, function($filterclass) { $rc = new \ReflectionClass($filterclass); return $rc->isInstantiable(); }); return $filters;> } } > } > /** > * Attempt to fetch any matching langstring if the content is in the > * format identifier,component. > * > * @param string $content Step's content or Tour's name or Tour's description > * @return string Processed content, any langstring will be converted to translated text > */ > public static function get_string_from_input(string $content): string { > $content = trim($content); > > if (preg_match(static::LANG_STRING_REGEX, $content, $matches)) { > if ($matches[2] === 'moodle') { > $matches[2] = 'core'; > } > > if (get_string_manager()->string_exists($matches[1], $matches[2])) { > $content = get_string($matches[1], $matches[2]); > } > } > > return $content; > } > > /** > * Check if the given string contains any matching langstring. > * > * @param string $string > * @return bool > */ > public static function is_language_string_from_input(string $string): bool { > return preg_match(static::LANG_STRING_REGEX, $string) == true;