. /** * Library of functions for web output * * Library of all general-purpose Moodle PHP functions and constants * that produce HTML output * * Other main libraries: * - datalib.php - functions that access the database. * - moodlelib.php - general-purpose Moodle functions. * * @package core * @subpackage lib * @copyright 1999 onwards Martin Dougiamas {@link http://moodle.com} * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ use Psr\Http\Message\UriInterface; defined('MOODLE_INTERNAL') || die(); // Constants. // Define text formatting types ... eventually we can add Wiki, BBcode etc. /** * Does all sorts of transformations and filtering. */ define('FORMAT_MOODLE', '0'); /** * Plain HTML (with some tags stripped). */ define('FORMAT_HTML', '1'); /** * Plain text (even tags are printed in full). */ define('FORMAT_PLAIN', '2'); /** * Wiki-formatted text. * Deprecated: left here just to note that '3' is not used (at the moment) * and to catch any latent wiki-like text (which generates an error) * @deprecated since 2005! */ define('FORMAT_WIKI', '3'); /** * Markdown-formatted text http://daringfireball.net/projects/markdown/ */ define('FORMAT_MARKDOWN', '4'); /** * A moodle_url comparison using this flag will return true if the base URLs match, params are ignored. */ define('URL_MATCH_BASE', 0); /** * A moodle_url comparison using this flag will return true if the base URLs match and the params of url1 are part of url2. */ define('URL_MATCH_PARAMS', 1); /** * A moodle_url comparison using this flag will return true if the two URLs are identical, except for the order of the params. */ define('URL_MATCH_EXACT', 2); // Functions. /** * Add quotes to HTML characters. * * Returns $var with HTML characters (like "<", ">", etc.) properly quoted. * Related function {@link p()} simply prints the output of this function. * * @param string $var the string potentially containing HTML characters * @return string */ function s($var) { if ($var === false) { return '0'; } if ($var === null || $var === '') { return ''; } return preg_replace( '/&#(\d+|x[0-9a-f]+);/i', '$1;', htmlspecialchars($var, ENT_QUOTES | ENT_HTML401 | ENT_SUBSTITUTE) ); } /** * Add quotes to HTML characters. * * Prints $var with HTML characters (like "<", ">", etc.) properly quoted. * This function simply calls & displays {@link s()}. * @see s() * * @param string $var the string potentially containing HTML characters */ function p($var) { echo s($var); } /** * Does proper javascript quoting. * * Do not use addslashes anymore, because it does not work when magic_quotes_sybase is enabled. * * @param mixed $var String, Array, or Object to add slashes to * @return mixed quoted result */ function addslashes_js($var) { if (is_string($var)) { $var = str_replace('\\', '\\\\', $var); $var = str_replace(array('\'', '"', "\n", "\r", "\0"), array('\\\'', '\\"', '\\n', '\\r', '\\0'), $var); $var = str_replace('', '<\/', $var); // XHTML compliance. } else if (is_array($var)) { $var = array_map('addslashes_js', $var); } else if (is_object($var)) { $a = get_object_vars($var); foreach ($a as $key => $value) { $a[$key] = addslashes_js($value); } $var = (object)$a; } return $var; } /** * Remove query string from url. * * Takes in a URL and returns it without the querystring portion. * * @param string $url the url which may have a query string attached. * @return string The remaining URL. */ function strip_querystring($url) { if ($url === null || $url === '') { return ''; } if ($commapos = strpos($url, '?')) { return substr($url, 0, $commapos); } else { return $url; } } /** * Returns the name of the current script, WITH the querystring portion. * * This function is necessary because PHP_SELF and REQUEST_URI and SCRIPT_NAME * return different things depending on a lot of things like your OS, Web * server, and the way PHP is compiled (ie. as a CGI, module, ISAPI, etc.) * NOTE: This function returns false if the global variables needed are not set. * * @return mixed String or false if the global variables needed are not set. */ function me() { global $ME; return $ME; } /** * Guesses the full URL of the current script. * * This function is using $PAGE->url, but may fall back to $FULLME which * is constructed from PHP_SELF and REQUEST_URI or SCRIPT_NAME * * @return mixed full page URL string or false if unknown */ function qualified_me() { global $FULLME, $PAGE, $CFG; if (isset($PAGE) and $PAGE->has_set_url()) { // This is the only recommended way to find out current page. return $PAGE->url->out(false); } else { if ($FULLME === null) { // CLI script most probably. return false; } if (!empty($CFG->sslproxy)) { // Return only https links when using SSL proxy. return preg_replace('/^http:/', 'https:', $FULLME, 1); } else { return $FULLME; } } } /** * Determines whether or not the Moodle site is being served over HTTPS. * * This is done simply by checking the value of $CFG->wwwroot, which seems * to be the only reliable method. * * @return boolean True if site is served over HTTPS, false otherwise. */ function is_https() { global $CFG; return (strpos($CFG->wwwroot, 'https://') === 0); } /** * Returns the cleaned local URL of the HTTP_REFERER less the URL query string parameters if required. * * @param bool $stripquery if true, also removes the query part of the url. * @return string The resulting referer or empty string. */ function get_local_referer($stripquery = true) { if (isset($_SERVER['HTTP_REFERER'])) { $referer = clean_param($_SERVER['HTTP_REFERER'], PARAM_LOCALURL); if ($stripquery) { return strip_querystring($referer); } else { return $referer; } } else { return ''; } } /** * Class for creating and manipulating urls. * * It can be used in moodle pages where config.php has been included without any further includes. * * It is useful for manipulating urls with long lists of params. * One situation where it will be useful is a page which links to itself to perform various actions * and / or to process form data. A moodle_url object : * can be created for a page to refer to itself with all the proper get params being passed from page call to * page call and methods can be used to output a url including all the params, optionally adding and overriding * params and can also be used to * - output the url without any get params * - and output the params as hidden fields to be output within a form * * @copyright 2007 jamiesensei * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later * @package core */ class moodle_url { /** * Scheme, ex.: http, https * @var string */ protected $scheme = ''; /** * Hostname. * @var string */ protected $host = ''; /** * Port number, empty means default 80 or 443 in case of http. * @var int */ protected $port = ''; /** * Username for http auth. * @var string */ protected $user = ''; /** * Password for http auth. * @var string */ protected $pass = ''; /** * Script path. * @var string */ protected $path = ''; /** * Optional slash argument value. * @var string */ protected $slashargument = ''; /** * Anchor, may be also empty, null means none. * @var string */ protected $anchor = null; /** * Url parameters as associative array. * @var array */ protected $params = array(); /** * Create new instance of moodle_url. * * @param moodle_url|string $url - moodle_url means make a copy of another * moodle_url and change parameters, string means full url or shortened * form (ex.: '/course/view.php'). It is strongly encouraged to not include * query string because it may result in double encoded values. Use the * $params instead. For admin URLs, just use /admin/script.php, this * class takes care of the $CFG->admin issue. * @param array $params these params override current params or add new * @param string $anchor The anchor to use as part of the URL if there is one. * @throws moodle_exception */ public function __construct($url, array $params = null, $anchor = null) { global $CFG; if ($url instanceof moodle_url) { $this->scheme = $url->scheme; $this->host = $url->host; $this->port = $url->port; $this->user = $url->user; $this->pass = $url->pass; $this->path = $url->path; $this->slashargument = $url->slashargument; $this->params = $url->params; $this->anchor = $url->anchor; } else { $url = $url ?? ''; // Detect if anchor used. $apos = strpos($url, '#'); if ($apos !== false) { $anchor = substr($url, $apos); $anchor = ltrim($anchor, '#'); $this->set_anchor($anchor); $url = substr($url, 0, $apos); } // Normalise shortened form of our url ex.: '/course/view.php'. if (strpos($url, '/') === 0) { $url = $CFG->wwwroot.$url; } if ($CFG->admin !== 'admin') { if (strpos($url, "$CFG->wwwroot/admin/") === 0) { $url = str_replace("$CFG->wwwroot/admin/", "$CFG->wwwroot/$CFG->admin/", $url); } } // Parse the $url. $parts = parse_url($url); if ($parts === false) { throw new moodle_exception('invalidurl'); } if (isset($parts['query'])) { // Note: the values may not be correctly decoded, url parameters should be always passed as array. parse_str(str_replace('&', '&', $parts['query']), $this->params); } unset($parts['query']); foreach ($parts as $key => $value) { $this->$key = $value; } // Detect slashargument value from path - we do not support directory names ending with .php. $pos = strpos($this->path, '.php/'); if ($pos !== false) { $this->slashargument = substr($this->path, $pos + 4); $this->path = substr($this->path, 0, $pos + 4); } } $this->params($params); if ($anchor !== null) { $this->anchor = (string)$anchor; } } /** * Add an array of params to the params for this url. * * The added params override existing ones if they have the same name. * * @param array $params Defaults to null. If null then returns all params. * @return array Array of Params for url. * @throws coding_exception */ public function params(array $params = null) { $params = (array)$params; foreach ($params as $key => $value) { if (is_int($key)) { throw new coding_exception('Url parameters can not have numeric keys!'); } if (!is_string($value)) { if (is_array($value)) { throw new coding_exception('Url parameters values can not be arrays!'); } if (is_object($value) and !method_exists($value, '__toString')) { throw new coding_exception('Url parameters values can not be objects, unless __toString() is defined!'); } } $this->params[$key] = (string)$value; } return $this->params; } /** * Remove all params if no arguments passed. * Remove selected params if arguments are passed. * * Can be called as either remove_params('param1', 'param2') * or remove_params(array('param1', 'param2')). * * @param string[]|string $params,... either an array of param names, or 1..n string params to remove as args. * @return array url parameters */ public function remove_params($params = null) { if (!is_array($params)) { $params = func_get_args(); } foreach ($params as $param) { unset($this->params[$param]); } return $this->params; } /** * Remove all url parameters. * * @todo remove the unused param. * @param array $params Unused param * @return void */ public function remove_all_params($params = null) { $this->params = array(); $this->slashargument = ''; } /** * Add a param to the params for this url. * * The added param overrides existing one if they have the same name. * * @param string $paramname name * @param string $newvalue Param value. If new value specified current value is overriden or parameter is added * @return mixed string parameter value, null if parameter does not exist */ public function param($paramname, $newvalue = '') { if (func_num_args() > 1) { // Set new value. $this->params(array($paramname => $newvalue)); } if (isset($this->params[$paramname])) { return $this->params[$paramname]; } else { return null; } } /** * Merges parameters and validates them * * @param array $overrideparams * @return array merged parameters * @throws coding_exception */ protected function merge_overrideparams(array $overrideparams = null) { $overrideparams = (array)$overrideparams; $params = $this->params; foreach ($overrideparams as $key => $value) { if (is_int($key)) { throw new coding_exception('Overridden parameters can not have numeric keys!'); } if (is_array($value)) { throw new coding_exception('Overridden parameters values can not be arrays!'); } if (is_object($value) and !method_exists($value, '__toString')) { throw new coding_exception('Overridden parameters values can not be objects, unless __toString() is defined!'); } $params[$key] = (string)$value; } return $params; } /** * Get the params as as a query string. * * This method should not be used outside of this method. * * @param bool $escaped Use & as params separator instead of plain & * @param array $overrideparams params to add to the output params, these * override existing ones with the same name. * @return string query string that can be added to a url. */ public function get_query_string($escaped = true, array $overrideparams = null) { $arr = array(); if ($overrideparams !== null) { $params = $this->merge_overrideparams($overrideparams); } else { $params = $this->params; } foreach ($params as $key => $val) { if (is_array($val)) { foreach ($val as $index => $value) { $arr[] = rawurlencode($key.'['.$index.']')."=".rawurlencode($value); } } else { if (isset($val) && $val !== '') { $arr[] = rawurlencode($key)."=".rawurlencode($val); } else { $arr[] = rawurlencode($key); } } } if ($escaped) { return implode('&', $arr); } else { return implode('&', $arr); } } /** * Get the url params as an array of key => value pairs. * * This helps in handling cases where url params contain arrays. * * @return array params array for templates. */ public function export_params_for_template(): array { $data = []; foreach ($this->params as $key => $val) { if (is_array($val)) { foreach ($val as $index => $value) { $data[] = ['name' => $key.'['.$index.']', 'value' => $value]; } } else { $data[] = ['name' => $key, 'value' => $val]; } } return $data; } /** * Shortcut for printing of encoded URL. * * @return string */ public function __toString() { return $this->out(true); } /** * Output url. * * If you use the returned URL in HTML code, you want the escaped ampersands. If you use * the returned URL in HTTP headers, you want $escaped=false. * * @param bool $escaped Use & as params separator instead of plain & * @param array $overrideparams params to add to the output url, these override existing ones with the same name. * @return string Resulting URL */ public function out($escaped = true, array $overrideparams = null) { global $CFG; if (!is_bool($escaped)) { debugging('Escape parameter must be of type boolean, '.gettype($escaped).' given instead.'); } $url = $this; // Allow url's to be rewritten by a plugin. if (isset($CFG->urlrewriteclass) && !isset($CFG->upgraderunning)) { $class = $CFG->urlrewriteclass; $pluginurl = $class::url_rewrite($url); if ($pluginurl instanceof moodle_url) { $url = $pluginurl; } } return $url->raw_out($escaped, $overrideparams); } /** * Output url without any rewrites * * This is identical in signature and use to out() but doesn't call the rewrite handler. * * @param bool $escaped Use & as params separator instead of plain & * @param array $overrideparams params to add to the output url, these override existing ones with the same name. * @return string Resulting URL */ public function raw_out($escaped = true, array $overrideparams = null) { if (!is_bool($escaped)) { debugging('Escape parameter must be of type boolean, '.gettype($escaped).' given instead.'); } $uri = $this->out_omit_querystring().$this->slashargument; $querystring = $this->get_query_string($escaped, $overrideparams); if ($querystring !== '') { $uri .= '?' . $querystring; } $uri .= $this->get_encoded_anchor(); return $uri; } /** * Encode the anchor according to RFC 3986. * * @return string The encoded anchor */ public function get_encoded_anchor(): string { if (is_null($this->anchor)) { return ''; } // RFC 3986 allows the following characters in a fragment without them being encoded: // pct-encoded: "%" HEXDIG HEXDIG // unreserved: ALPHA / DIGIT / "-" / "." / "_" / "~" / // sub-delims: "!" / "$" / "&" / "'" / "(" / ")" / "*" / "+" / "," / ";" / "=" / ":" / "@" // fragment: "/" / "?" // // All other characters should be encoded. // These should not be encoded in the fragment unless they were already encoded. // The following characters are allowed in the fragment without encoding. // In addition to this list is pct-encoded, but we can't easily handle this with a regular expression. $allowed = 'a-zA-Z0-9\\-._~!$&\'()*+,;=:@\/?'; $anchor = '#'; $remainder = $this->anchor; do { // Split the string on any %. $parts = explode('%', $remainder, 2); $anchorparts = array_shift($parts); // The first part can go through our preg_replace_callback to quote any relevant characters. $anchor .= preg_replace_callback( '/[^' . $allowed . ']/', fn ($matches) => rawurlencode($matches[0]), $anchorparts, ); // The second part _might_ be a valid pct-encoded character. if (count($parts) === 0) { break; } // If the second part is a valid pct-encoded character, append it to the anchor. $remainder = array_shift($parts); if (preg_match('/^[a-fA-F0-9]{2}/', $remainder, $matches)) { $anchor .= "%{$matches[0]}"; $remainder = substr($remainder, 2); } else { // This was not a valid pct-encoded character. Encode the % and continue with the next part. $anchor .= rawurlencode('%'); } } while (strlen($remainder) > 0); return $anchor; } /** * Returns url without parameters, everything before '?'. * * @param bool $includeanchor if {@link self::anchor} is defined, should it be returned? * @return string */ public function out_omit_querystring($includeanchor = false) { $uri = $this->scheme ? $this->scheme.':'.((strtolower($this->scheme) == 'mailto') ? '':'//'): ''; $uri .= $this->user ? $this->user.($this->pass? ':'.$this->pass:'').'@':''; $uri .= $this->host ? $this->host : ''; $uri .= $this->port ? ':'.$this->port : ''; $uri .= $this->path ? $this->path : ''; if ($includeanchor) { $uri .= $this->get_encoded_anchor(); } return $uri; } /** * Compares this moodle_url with another. * * See documentation of constants for an explanation of the comparison flags. * * @param moodle_url $url The moodle_url object to compare * @param int $matchtype The type of comparison (URL_MATCH_BASE, URL_MATCH_PARAMS, URL_MATCH_EXACT) * @return bool */ public function compare(moodle_url $url, $matchtype = URL_MATCH_EXACT) { $baseself = $this->out_omit_querystring(); $baseother = $url->out_omit_querystring(); // Append index.php if there is no specific file. if (substr($baseself, -1) == '/') { $baseself .= 'index.php'; } if (substr($baseother, -1) == '/') { $baseother .= 'index.php'; } // Compare the two base URLs. if ($baseself != $baseother) { return false; } if ($matchtype == URL_MATCH_BASE) { return true; } $urlparams = $url->params(); foreach ($this->params() as $param => $value) { if ($param == 'sesskey') { continue; } if (!array_key_exists($param, $urlparams) || $urlparams[$param] != $value) { return false; } } if ($matchtype == URL_MATCH_PARAMS) { return true; } foreach ($urlparams as $param => $value) { if ($param == 'sesskey') { continue; } if (!array_key_exists($param, $this->params()) || $this->param($param) != $value) { return false; } } if ($url->anchor !== $this->anchor) { return false; } return true; } /** * Sets the anchor for the URI (the bit after the hash) * * @param string $anchor null means remove previous */ public function set_anchor($anchor) { if (is_null($anchor)) { // Remove. $this->anchor = null; } else { $this->anchor = $anchor; } } /** * Sets the scheme for the URI (the bit before ://) * * @param string $scheme */ public function set_scheme($scheme) { // See http://www.ietf.org/rfc/rfc3986.txt part 3.1. if (preg_match('/^[a-zA-Z][a-zA-Z0-9+.-]*$/', $scheme)) { $this->scheme = $scheme; } else { throw new coding_exception('Bad URL scheme.'); } } /** * Sets the url slashargument value. * * @param string $path usually file path * @param string $parameter name of page parameter if slasharguments not supported * @param bool $supported usually null, then it depends on $CFG->slasharguments, use true or false for other servers * @return void */ public function set_slashargument($path, $parameter = 'file', $supported = null) { global $CFG; if (is_null($supported)) { $supported = !empty($CFG->slasharguments); } if ($supported) { $parts = explode('/', $path); $parts = array_map('rawurlencode', $parts); $path = implode('/', $parts); $this->slashargument = $path; unset($this->params[$parameter]); } else { $this->slashargument = ''; $this->params[$parameter] = $path; } } // Static factory methods. /** * Create a new moodle_url instance from a UriInterface. * * @param UriInterface $uri * @return self */ public static function from_uri(UriInterface $uri): self { $url = new self( url: $uri->getScheme() . '://' . $uri->getAuthority() . $uri->getPath(), anchor: $uri->getFragment() ?: null, ); $params = $uri->getQuery(); foreach (explode('&', $params) as $param) { $url->param(...explode('=', $param, 2)); } return $url; } /** * General moodle file url. * * @param string $urlbase the script serving the file * @param string $path * @param bool $forcedownload * @return moodle_url */ public static function make_file_url($urlbase, $path, $forcedownload = false) { $params = array(); if ($forcedownload) { $params['forcedownload'] = 1; } $url = new moodle_url($urlbase, $params); $url->set_slashargument($path); return $url; } /** * Factory method for creation of url pointing to plugin file. * * Please note this method can be used only from the plugins to * create urls of own files, it must not be used outside of plugins! * * @param int $contextid * @param string $component * @param string $area * @param ?int $itemid * @param string $pathname * @param string $filename * @param bool $forcedownload * @param mixed $includetoken Whether to use a user token when displaying this group image. * True indicates to generate a token for current user, and integer value indicates to generate a token for the * user whose id is the value indicated. * If the group picture is included in an e-mail or some other location where the audience is a specific * user who will not be logged in when viewing, then we use a token to authenticate the user. * @return moodle_url */ public static function make_pluginfile_url($contextid, $component, $area, $itemid, $pathname, $filename, $forcedownload = false, $includetoken = false) { global $CFG, $USER; $path = []; if ($includetoken) { $urlbase = "$CFG->wwwroot/tokenpluginfile.php"; $userid = $includetoken === true ? $USER->id : $includetoken; $token = get_user_key('core_files', $userid); if ($CFG->slasharguments) { $path[] = $token; } } else { $urlbase = "$CFG->wwwroot/pluginfile.php"; } $path[] = $contextid; $path[] = $component; $path[] = $area; if ($itemid !== null) { $path[] = $itemid; } $path = "/" . implode('/', $path) . "{$pathname}{$filename}"; $url = self::make_file_url($urlbase, $path, $forcedownload, $includetoken); if ($includetoken && empty($CFG->slasharguments)) { $url->param('token', $token); } return $url; } /** * Factory method for creation of url pointing to plugin file. * This method is the same that make_pluginfile_url but pointing to the webservice pluginfile.php script. * It should be used only in external functions. * * @since 2.8 * @param int $contextid * @param string $component * @param string $area * @param int $itemid * @param string $pathname * @param string $filename * @param bool $forcedownload * @return moodle_url */ public static function make_webservice_pluginfile_url($contextid, $component, $area, $itemid, $pathname, $filename, $forcedownload = false) { global $CFG; $urlbase = "$CFG->wwwroot/webservice/pluginfile.php"; if ($itemid === null) { return self::make_file_url($urlbase, "/$contextid/$component/$area".$pathname.$filename, $forcedownload); } else { return self::make_file_url($urlbase, "/$contextid/$component/$area/$itemid".$pathname.$filename, $forcedownload); } } /** * Factory method for creation of url pointing to draft file of current user. * * @param int $draftid draft item id * @param string $pathname * @param string $filename * @param bool $forcedownload * @return moodle_url */ public static function make_draftfile_url($draftid, $pathname, $filename, $forcedownload = false) { global $CFG, $USER; $urlbase = "$CFG->wwwroot/draftfile.php"; $context = context_user::instance($USER->id); return self::make_file_url($urlbase, "/$context->id/user/draft/$draftid".$pathname.$filename, $forcedownload); } /** * Factory method for creating of links to legacy course files. * * @param int $courseid * @param string $filepath * @param bool $forcedownload * @return moodle_url */ public static function make_legacyfile_url($courseid, $filepath, $forcedownload = false) { global $CFG; $urlbase = "$CFG->wwwroot/file.php"; return self::make_file_url($urlbase, '/'.$courseid.'/'.$filepath, $forcedownload); } /** * Checks if URL is relative to $CFG->wwwroot. * * @return bool True if URL is relative to $CFG->wwwroot; otherwise, false. */ public function is_local_url(): bool { global $CFG; $url = $this->out(); // Does URL start with wwwroot? Otherwise, URL isn't relative to wwwroot. return ( ($url === $CFG->wwwroot) || (strpos($url, $CFG->wwwroot.'/') === 0) ); } /** * Returns URL as relative path from $CFG->wwwroot * * Can be used for passing around urls with the wwwroot stripped * * @param boolean $escaped Use & as params separator instead of plain & * @param array $overrideparams params to add to the output url, these override existing ones with the same name. * @return string Resulting URL * @throws coding_exception if called on a non-local url */ public function out_as_local_url($escaped = true, array $overrideparams = null) { global $CFG; // URL should be relative to wwwroot. If not then throw exception. if ($this->is_local_url()) { $url = $this->out($escaped, $overrideparams); $localurl = substr($url, strlen($CFG->wwwroot)); return !empty($localurl) ? $localurl : ''; } else { throw new coding_exception('out_as_local_url called on a non-local URL'); } } /** * Returns the 'path' portion of a URL. For example, if the URL is * http://www.example.org:447/my/file/is/here.txt?really=1 then this will * return '/my/file/is/here.txt'. * * By default the path includes slash-arguments (for example, * '/myfile.php/extra/arguments') so it is what you would expect from a * URL path. If you don't want this behaviour, you can opt to exclude the * slash arguments. (Be careful: if the $CFG variable slasharguments is * disabled, these URLs will have a different format and you may need to * look at the 'file' parameter too.) * * @param bool $includeslashargument If true, includes slash arguments * @return string Path of URL */ public function get_path($includeslashargument = true) { return $this->path . ($includeslashargument ? $this->slashargument : ''); } /** * Returns a given parameter value from the URL. * * @param string $name Name of parameter * @return string Value of parameter or null if not set */ public function get_param($name) { if (array_key_exists($name, $this->params)) { return $this->params[$name]; } else { return null; } } /** * Returns the 'scheme' portion of a URL. For example, if the URL is * http://www.example.org:447/my/file/is/here.txt?really=1 then this will * return 'http' (without the colon). * * @return string Scheme of the URL. */ public function get_scheme() { return $this->scheme; } /** * Returns the 'host' portion of a URL. For example, if the URL is * http://www.example.org:447/my/file/is/here.txt?really=1 then this will * return 'www.example.org'. * * @return string Host of the URL. */ public function get_host() { return $this->host; } /** * Returns the 'port' portion of a URL. For example, if the URL is * http://www.example.org:447/my/file/is/here.txt?really=1 then this will * return '447'. * * @return string Port of the URL. */ public function get_port() { return $this->port; } } /** * Determine if there is data waiting to be processed from a form * * Used on most forms in Moodle to check for data * Returns the data as an object, if it's found. * This object can be used in foreach loops without * casting because it's cast to (array) automatically * * Checks that submitted POST data exists and returns it as object. * * @return mixed false or object */ function data_submitted() { if (empty($_POST)) { return false; } else { return (object)fix_utf8($_POST); } } /** * Given some normal text this function will break up any * long words to a given size by inserting the given character * * It's multibyte savvy and doesn't change anything inside html tags. * * @param string $string the string to be modified * @param int $maxsize maximum length of the string to be returned * @param string $cutchar the string used to represent word breaks * @return string */ function break_up_long_words($string, $maxsize=20, $cutchar=' ') { // First of all, save all the tags inside the text to skip them. $tags = array(); filter_save_tags($string, $tags); // Process the string adding the cut when necessary. $output = ''; $length = core_text::strlen($string); $wordlength = 0; for ($i=0; $i<$length; $i++) { $char = core_text::substr($string, $i, 1); if ($char == ' ' or $char == "\t" or $char == "\n" or $char == "\r" or $char == "<" or $char == ">") { $wordlength = 0; } else { $wordlength++; if ($wordlength > $maxsize) { $output .= $cutchar; $wordlength = 0; } } $output .= $char; } // Finally load the tags back again. if (!empty($tags)) { $output = str_replace(array_keys($tags), $tags, $output); } return $output; } /** * Try and close the current window using JavaScript, either immediately, or after a delay. * * Echo's out the resulting XHTML & javascript * * @param integer $delay a delay in seconds before closing the window. Default 0. * @param boolean $reloadopener if true, we will see if this window was a pop-up, and try * to reload the parent window before this one closes. */ function close_window($delay = 0, $reloadopener = false) { global $PAGE, $OUTPUT; if (!$PAGE->headerprinted) { $PAGE->set_title(get_string('closewindow')); echo $OUTPUT->header(); } else { $OUTPUT->container_end_all(false); } if ($reloadopener) { // Trigger the reload immediately, even if the reload is after a delay. $PAGE->requires->js_function_call('window.opener.location.reload', array(true)); } $OUTPUT->notification(get_string('windowclosing'), 'notifysuccess'); $PAGE->requires->js_function_call('close_window', array(new stdClass()), false, $delay); echo $OUTPUT->footer(); exit; } /** * Returns a string containing a link to the user documentation for the current page. * * Also contains an icon by default. Shown to teachers and admin only. * * @param string $text The text to be displayed for the link * @return string The link to user documentation for this current page */ function page_doc_link($text='') { global $OUTPUT, $PAGE; $path = page_get_doc_link_path($PAGE); if (!$path) { return ''; } return $OUTPUT->doc_link($path, $text); } /** * Returns the path to use when constructing a link to the docs. * * @since Moodle 2.5.1 2.6 * @param moodle_page $page * @return string */ function page_get_doc_link_path(moodle_page $page) { global $CFG; if (empty($CFG->docroot) || during_initial_install()) { return ''; } if (!has_capability('moodle/site:doclinks', $page->context)) { return ''; } $path = $page->docspath; if (!$path) { return ''; } return $path; } /** * Validates an email to make sure it makes sense. * * @param string $address The email address to validate. * @return boolean */ function validate_email($address) { global $CFG; if ($address === null || $address === false || $address === '') { return false; } require_once("{$CFG->libdir}/phpmailer/moodle_phpmailer.php"); return moodle_phpmailer::validateAddress($address ?? '') && !preg_match('/[<>]/', $address); } /** * Extracts file argument either from file parameter or PATH_INFO * * Note: $scriptname parameter is not needed anymore * * @return string file path (only safe characters) */ function get_file_argument() { global $SCRIPT; $relativepath = false; $hasforcedslashargs = false; if (isset($_SERVER['REQUEST_URI']) && !empty($_SERVER['REQUEST_URI'])) { // Checks whether $_SERVER['REQUEST_URI'] contains '/pluginfile.php/' // instead of '/pluginfile.php?', when serving a file from e.g. mod_imscp or mod_scorm. if ((strpos($_SERVER['REQUEST_URI'], '/pluginfile.php/') !== false) && isset($_SERVER['PATH_INFO']) && !empty($_SERVER['PATH_INFO'])) { // Exclude edge cases like '/pluginfile.php/?file='. $args = explode('/', ltrim($_SERVER['PATH_INFO'], '/')); $hasforcedslashargs = (count($args) > 2); // Always at least: context, component and filearea. } } if (!$hasforcedslashargs) { $relativepath = optional_param('file', false, PARAM_PATH); } if ($relativepath !== false and $relativepath !== '') { return $relativepath; } $relativepath = false; // Then try extract file from the slasharguments. if (stripos($_SERVER['SERVER_SOFTWARE'], 'iis') !== false) { // NOTE: IIS tends to convert all file paths to single byte DOS encoding, // we can not use other methods because they break unicode chars, // the only ways are to use URL rewriting // OR // to properly set the 'FastCGIUtf8ServerVariables' registry key. if (isset($_SERVER['PATH_INFO']) and $_SERVER['PATH_INFO'] !== '') { // Check that PATH_INFO works == must not contain the script name. if (strpos($_SERVER['PATH_INFO'], $SCRIPT) === false) { $relativepath = clean_param(urldecode($_SERVER['PATH_INFO']), PARAM_PATH); } } } else { // All other apache-like servers depend on PATH_INFO. if (isset($_SERVER['PATH_INFO'])) { if (isset($_SERVER['SCRIPT_NAME']) and strpos($_SERVER['PATH_INFO'], $_SERVER['SCRIPT_NAME']) === 0) { $relativepath = substr($_SERVER['PATH_INFO'], strlen($_SERVER['SCRIPT_NAME'])); } else { $relativepath = $_SERVER['PATH_INFO']; } $relativepath = clean_param($relativepath, PARAM_PATH); } } return $relativepath; } /** * Just returns an array of text formats suitable for a popup menu * * @return array */ function format_text_menu() { return array (FORMAT_MOODLE => get_string('formattext'), FORMAT_HTML => get_string('formathtml'), FORMAT_PLAIN => get_string('formatplain'), FORMAT_MARKDOWN => get_string('formatmarkdown')); } /** * Given text in a variety of format codings, this function returns the text as safe HTML. * * This function should mainly be used for long strings like posts, * answers, glossary items etc. For short strings {@link format_string()}. * *
* Options: * trusted : If true the string won't be cleaned. Default false required noclean=true. * noclean : If true the string won't be cleaned, unless $CFG->forceclean is set. Default false required trusted=true. * filter : If true the string will be run through applicable filters as well. Default true. * para : If true then the returned string will be wrapped in div tags. Default true. * newlines : If true then lines newline breaks will be converted to HTML newline breaks. Default true. * context : The context that will be used for filtering. * overflowdiv : If set to true the formatted text will be encased in a div * with the class no-overflow before being returned. Default false. * allowid : If true then id attributes will not be removed, even when * using htmlpurifier. Default false. * blanktarget : If true all tags will have target="_blank" added unless target is explicitly specified. ** * @param string $text The text to be formatted. This is raw text originally from user input. * @param int $format Identifier of the text format to be used * [FORMAT_MOODLE, FORMAT_HTML, FORMAT_PLAIN, FORMAT_MARKDOWN] * @param stdClass|array $options text formatting options * @param int $courseiddonotuse deprecated course id, use context option instead * @return string */ function format_text($text, $format = FORMAT_MOODLE, $options = null, $courseiddonotuse = null) { global $CFG; // Manually include the formatting class for now until after the release after 4.5 LTS. require_once("{$CFG->libdir}/classes/formatting.php"); if ($format === FORMAT_WIKI) { // This format was deprecated in Moodle 1.5. throw new \coding_exception( 'Wiki-like formatting is not supported.' ); } if ($options instanceof \core\context) { // A common mistake has been to call this function with a context object. // This has never been expected, or nor supported. debugging( 'The options argument should not be a context object directly. ' . ' Please pass an array with a context key instead.', DEBUG_DEVELOPER, ); $params['context'] = $options; $options = []; } if ($options) { $options = (array) $options; } if (empty($CFG->version) || $CFG->version < 2013051400 || during_initial_install()) { // Do not filter anything during installation or before upgrade completes. $params['context'] = null; } else if ($options && isset($options['context'])) { // First by explicit passed context option. if (is_numeric($options['context'])) { // A contextid was passed. $params['context'] = \core\context::instance_by_id($options['context']); } else if ($options['context'] instanceof \core\context) { $params['context'] = $options['context']; } else { debugging( 'Unknown context passed to format_text(). Content will not be filtered.', DEBUG_DEVELOPER, ); } // Unset the context from $options to prevent it overriding the configured value. unset($options['context']); } else if ($courseiddonotuse) { // Legacy courseid. $params['context'] = \core\context\course::instance($courseiddonotuse); debugging( "Passing a courseid to format_text() is deprecated, please pass a context instead.", DEBUG_DEVELOPER, ); } $params['text'] = $text; if ($options) { // The smiley option was deprecated in Moodle 2.0. if (array_key_exists('smiley', $options)) { unset($options['smiley']); debugging( 'The smiley option is deprecated and no longer used.', DEBUG_DEVELOPER, ); } // The nocache option was deprecated in Moodle 2.3 in MDL-34347. if (array_key_exists('nocache', $options)) { unset($options['nocache']); debugging( 'The nocache option is deprecated and no longer used.', DEBUG_DEVELOPER, ); } $validoptions = [ 'text', 'format', 'context', 'trusted', 'clean', 'filter', 'para', 'newlines', 'overflowdiv', 'blanktarget', 'allowid', 'noclean', ]; $invalidoptions = array_diff(array_keys($options), $validoptions); if ($invalidoptions) { debugging(sprintf( 'The following options are not valid: %s', implode(', ', $invalidoptions), ), DEBUG_DEVELOPER); foreach ($invalidoptions as $option) { unset($options[$option]); } } foreach ($options as $option => $value) { $params[$option] = $value; } // The noclean option has been renamed to clean. if (array_key_exists('noclean', $params)) { $params['clean'] = !$params['noclean']; unset($params['noclean']); } } if ($format !== null) { $params['format'] = $format; } return \core\di::get(\core\formatting::class)->format_text(...$params); } /** * Resets some data related to filters, called during upgrade or when general filter settings change. * * @param bool $phpunitreset true means called from our PHPUnit integration test reset * @return void */ function reset_text_filters_cache($phpunitreset = false) { global $CFG, $DB; if ($phpunitreset) { // HTMLPurifier does not change, DB is already reset to defaults, // nothing to do here, the dataroot was cleared too. return; } // The purge_all_caches() deals with cachedir and localcachedir purging, // the individual filter caches are invalidated as necessary elsewhere. // Update $CFG->filterall cache flag. if (empty($CFG->stringfilters)) { set_config('filterall', 0); return; } $installedfilters = core_component::get_plugin_list('filter'); $filters = explode(',', $CFG->stringfilters); foreach ($filters as $filter) { if (isset($installedfilters[$filter])) { set_config('filterall', 1); return; } } set_config('filterall', 0); } /** * Given a simple string, this function returns the string * processed by enabled string filters if $CFG->filterall is enabled * * This function should be used to print short strings (non html) that * need filter processing e.g. activity titles, post subjects, * glossary concepts. * * @staticvar bool $strcache * @param string $string The string to be filtered. Should be plain text, expect * possibly for multilang tags. * @param ?bool $striplinks To strip any link in the result text. Moodle 1.8 default changed from false to true! MDL-8713 * @param array $options options array/object or courseid * @return string */ function format_string($string, $striplinks = true, $options = null) { global $CFG; // Manually include the formatting class for now until after the release after 4.5 LTS. require_once("{$CFG->libdir}/classes/formatting.php"); $params = [ 'string' => $string, 'striplinks' => (bool) $striplinks, ]; // This method only expects either: // - an array of options; // - a stdClass of options to be cast to an array; or // - an integer courseid. if ($options instanceof \core\context) { // A common mistake has been to call this function with a context object. // This has never been expected, or nor supported. debugging( 'The options argument should not be a context object directly. ' . ' Please pass an array with a context key instead.', DEBUG_DEVELOPER, ); $params['context'] = $options; $options = []; } else if (is_numeric($options)) { // Legacy courseid usage. $params['context'] = \core\context\course::instance($options); $options = []; } else if (is_array($options) || is_a($options, \stdClass::class)) { $options = (array) $options; if (isset($options['context'])) { if (is_numeric($options['context'])) { // A contextid was passed usage. $params['context'] = \core\context::instance_by_id($options['context']); } else if ($options['context'] instanceof \core\context) { $params['context'] = $options['context']; } else { debugging( 'An invalid value for context was provided.', DEBUG_DEVELOPER, ); } } } else if ($options !== null) { // Something else was passed, so we'll just use an empty array. debugging(sprintf( 'The options argument should be an Array, or stdclass. %s passed.', gettype($options), ), DEBUG_DEVELOPER); // Attempt to cast to array since we always used to, but throw in some debugging. $options = array_filter( (array) $options, fn ($key) => !is_numeric($key), ARRAY_FILTER_USE_KEY, ); } if (isset($options['filter'])) { $params['filter'] = (bool) $options['filter']; } else { $params['filter'] = true; } if (isset($options['escape'])) { $params['escape'] = (bool) $options['escape']; } else { $params['escape'] = true; } $validoptions = [ 'string', 'striplinks', 'context', 'filter', 'escape', ]; if ($options) { $invalidoptions = array_diff(array_keys($options), $validoptions); if ($invalidoptions) { debugging(sprintf( 'The following options are not valid: %s', implode(', ', $invalidoptions), ), DEBUG_DEVELOPER); } } return \core\di::get(\core\formatting::class)->format_string( ...$params, ); } /** * Given a string, performs a negative lookahead looking for any ampersand character * that is not followed by a proper HTML entity. If any is found, it is replaced * by &. The string is then returned. * * @param string $string * @return string */ function replace_ampersands_not_followed_by_entity($string) { return preg_replace("/\&(?![a-zA-Z0-9#]{1,8};)/", "&", $string ?? ''); } /** * Given a string, replaces all .* by .* and returns the string. * * @param string $string * @return string */ function strip_links($string) { return preg_replace('/(]+?>)(.+?)(<\/a>)/is', '$2', $string); } /** * This expression turns links into something nice in a text format. (Russell Jungwirth) * * @param string $string * @return string */ function wikify_links($string) { return preg_replace('~(]*>([^<]*))~i', '$3 [ $2 ]', $string); } /** * Given text in a variety of format codings, this function returns the text as plain text suitable for plain email. * * @param string $text The text to be formatted. This is raw text originally from user input. * @param int $format Identifier of the text format to be used * [FORMAT_MOODLE, FORMAT_HTML, FORMAT_PLAIN, FORMAT_WIKI, FORMAT_MARKDOWN] * @return string */ function format_text_email($text, $format) { switch ($format) { case FORMAT_PLAIN: return $text; break; case FORMAT_WIKI: // There should not be any of these any more! $text = wikify_links($text); return core_text::entities_to_utf8(strip_tags($text), true); break; case FORMAT_HTML: return html_to_text($text); break; case FORMAT_MOODLE: case FORMAT_MARKDOWN: default: $text = wikify_links($text); return core_text::entities_to_utf8(strip_tags($text), true); break; } } /** * Formats activity intro text * * @param string $module name of module * @param object $activity instance of activity * @param int $cmid course module id * @param bool $filter filter resulting html text * @return string */ function format_module_intro($module, $activity, $cmid, $filter=true) { global $CFG; require_once("$CFG->libdir/filelib.php"); $context = context_module::instance($cmid); $options = array('noclean' => true, 'para' => false, 'filter' => $filter, 'context' => $context, 'overflowdiv' => true); $intro = file_rewrite_pluginfile_urls($activity->intro, 'pluginfile.php', $context->id, 'mod_'.$module, 'intro', null); return trim(format_text($intro, $activity->introformat, $options, null)); } /** * Removes the usage of Moodle files from a text. * * In some rare cases we need to re-use a text that already has embedded links * to some files hosted within Moodle. But the new area in which we will push * this content does not support files... therefore we need to remove those files. * * @param string $source The text * @return string The stripped text */ function strip_pluginfile_content($source) { $baseurl = '@@PLUGINFILE@@'; // Looking for something like < .* "@@pluginfile@@.*" .* > $pattern = '$<[^<>]+["\']' . $baseurl . '[^"\']*["\'][^<>]*>$'; $stripped = preg_replace($pattern, '', $source); // Use purify html to rebalence potentially mismatched tags and generally cleanup. return purify_html($stripped); } /** * Legacy function, used for cleaning of old forum and glossary text only. * * @param string $text text that may contain legacy TRUSTTEXT marker * @return string text without legacy TRUSTTEXT marker */ function trusttext_strip($text) { if (!is_string($text)) { // This avoids the potential for an endless loop below. throw new coding_exception('trusttext_strip parameter must be a string'); } while (true) { // Removing nested TRUSTTEXT. $orig = $text; $text = str_replace('#####TRUSTTEXT#####', '', $text); if (strcmp($orig, $text) === 0) { return $text; } } } /** * Must be called before editing of all texts with trust flag. Removes all XSS nasties from texts stored in database if needed. * * @param stdClass $object data object with xxx, xxxformat and xxxtrust fields * @param string $field name of text field * @param context $context active context * @return stdClass updated $object */ function trusttext_pre_edit($object, $field, $context) { $trustfield = $field.'trust'; $formatfield = $field.'format'; if ($object->$formatfield == FORMAT_MARKDOWN) { // We do not have a way to sanitise Markdown texts, // luckily editors for this format should not have XSS problems. return $object; } if (!$object->$trustfield or !trusttext_trusted($context)) { $object->$field = clean_text($object->$field, $object->$formatfield); } return $object; } /** * Is current user trusted to enter no dangerous XSS in this context? * * Please note the user must be in fact trusted everywhere on this server!! * * @param context $context * @return bool true if user trusted */ function trusttext_trusted($context) { return (trusttext_active() and has_capability('moodle/site:trustcontent', $context)); } /** * Is trusttext feature active? * * @return bool */ function trusttext_active() { global $CFG; return !empty($CFG->enabletrusttext); } /** * Cleans raw text removing nasties. * * Given raw text (eg typed in by a user) this function cleans it up and removes any nasty tags that could mess up * Moodle pages through XSS attacks. * * The result must be used as a HTML text fragment, this function can not cleanup random * parts of html tags such as url or src attributes. * * NOTE: the format parameter was deprecated because we can safely clean only HTML. * * @param string $text The text to be cleaned * @param int|string $format deprecated parameter, should always contain FORMAT_HTML or FORMAT_MOODLE * @param array $options Array of options; currently only option supported is 'allowid' (if true, * does not remove id attributes when cleaning) * @return string The cleaned up text */ function clean_text($text, $format = FORMAT_HTML, $options = array()) { $text = (string)$text; if ($format != FORMAT_HTML and $format != FORMAT_HTML) { // TODO: we need to standardise cleanup of text when loading it into editor first. // debugging('clean_text() is designed to work only with html');. } if ($format == FORMAT_PLAIN) { return $text; } if (is_purify_html_necessary($text)) { $text = purify_html($text, $options); } // Originally we tried to neutralise some script events here, it was a wrong approach because // it was trivial to work around that (for example using style based XSS exploits). // We must not give false sense of security here - all developers MUST understand how to use // rawurlencode(), htmlentities(), htmlspecialchars(), p(), s(), moodle_url, html_writer and friends!!! return $text; } /** * Is it necessary to use HTMLPurifier? * * @private * @param string $text * @return bool false means html is safe and valid, true means use HTMLPurifier */ function is_purify_html_necessary($text) { if ($text === '') { return false; } if ($text === (string)((int)$text)) { return false; } if (strpos($text, '&') !== false or preg_match('|<[^pesb/]|', $text)) { // We need to normalise entities or other tags except p, em, strong and br present. return true; } $altered = htmlspecialchars($text, ENT_NOQUOTES, 'UTF-8', true); if ($altered === $text) { // No < > or other special chars means this must be safe. return false; } // Let's try to convert back some safe html tags. $altered = preg_replace('|<p>(.*?)</p>|m', '
$1
', $altered); if ($altered === $text) { return false; } $altered = preg_replace('|<em>([^<>]+?)</em>|m', '$1', $altered); if ($altered === $text) { return false; } $altered = preg_replace('|<strong>([^<>]+?)</strong>|m', '$1', $altered); if ($altered === $text) { return false; } $altered = str_replace('<br />', '
* if (!isset($CFG->additionalhtmlhead)) {
* $CFG->additionalhtmlhead = '';
* }
* $CFG->additionalhtmlhead .= '';
* header('X-UA-Compatible: IE=8');
* echo $OUTPUT->header();
*
*
* Please note the $CFG->additionalhtmlhead alone might not work,
* you should send the IE compatibility header() too.
*
* @param string $contenttype
* @param bool $cacheable Can this page be cached on back?
* @return void, sends HTTP headers
*/
function send_headers($contenttype, $cacheable = true) {
global $CFG;
@header('Content-Type: ' . $contenttype);
@header('Content-Script-Type: text/javascript');
@header('Content-Style-Type: text/css');
if (empty($CFG->additionalhtmlhead) or stripos($CFG->additionalhtmlhead, 'X-UA-Compatible') === false) {
@header('X-UA-Compatible: IE=edge');
}
if ($cacheable) {
// Allow caching on "back" (but not on normal clicks).
@header('Cache-Control: private, pre-check=0, post-check=0, max-age=0, no-transform');
@header('Pragma: no-cache');
@header('Expires: ');
} else {
// Do everything we can to always prevent clients and proxies caching.
@header('Cache-Control: no-store, no-cache, must-revalidate');
@header('Cache-Control: post-check=0, pre-check=0, no-transform', false);
@header('Pragma: no-cache');
@header('Expires: Mon, 20 Aug 1969 09:23:00 GMT');
@header('Last-Modified: ' . gmdate('D, d M Y H:i:s') . ' GMT');
}
@header('Accept-Ranges: none');
// The Moodle app must be allowed to embed content always.
if (empty($CFG->allowframembedding) && !core_useragent::is_moodle_app()) {
@header('X-Frame-Options: sameorigin');
}
// If referrer policy is set, add a referrer header.
if (!empty($CFG->referrerpolicy) && ($CFG->referrerpolicy !== 'default')) {
@header('Referrer-Policy: ' . $CFG->referrerpolicy);
}
}
/**
* Return the right arrow with text ('next'), and optionally embedded in a link.
*
* @param string $text HTML/plain text label (set to blank only for breadcrumb separator cases).
* @param string $url An optional link to use in a surrounding HTML anchor.
* @param bool $accesshide True if text should be hidden (for screen readers only).
* @param string $addclass Additional class names for the link, or the arrow character.
* @return string HTML string.
*/
function link_arrow_right($text, $url='', $accesshide=false, $addclass='', $addparams = []) {
global $OUTPUT; // TODO: move to output renderer.
$arrowclass = 'arrow ';
if (!$url) {
$arrowclass .= $addclass;
}
$arrow = ' ';
$htmltext = '';
if ($text) {
$htmltext = ''.$text.' ';
if ($accesshide) {
$htmltext = get_accesshide($htmltext);
}
}
if ($url) {
$class = 'arrow_link';
if ($addclass) {
$class .= ' '.$addclass;
}
$linkparams = [
'class' => $class,
'href' => $url,
'title' => preg_replace('/<.*?>/', '', $text),
];
$linkparams += $addparams;
return html_writer::link($url, $htmltext . $arrow, $linkparams);
}
return $htmltext.$arrow;
}
/**
* Return the left arrow with text ('previous'), and optionally embedded in a link.
*
* @param string $text HTML/plain text label (set to blank only for breadcrumb separator cases).
* @param string $url An optional link to use in a surrounding HTML anchor.
* @param bool $accesshide True if text should be hidden (for screen readers only).
* @param string $addclass Additional class names for the link, or the arrow character.
* @return string HTML string.
*/
function link_arrow_left($text, $url='', $accesshide=false, $addclass='', $addparams = []) {
global $OUTPUT; // TODO: move to utput renderer.
$arrowclass = 'arrow ';
if (! $url) {
$arrowclass .= $addclass;
}
$arrow = ' ';
$htmltext = '';
if ($text) {
$htmltext = ' '.$text.'';
if ($accesshide) {
$htmltext = get_accesshide($htmltext);
}
}
if ($url) {
$class = 'arrow_link';
if ($addclass) {
$class .= ' '.$addclass;
}
$linkparams = [
'class' => $class,
'href' => $url,
'title' => preg_replace('/<.*?>/', '', $text),
];
$linkparams += $addparams;
return html_writer::link($url, $arrow . $htmltext, $linkparams);
}
return $arrow.$htmltext;
}
/**
* Return a HTML element with the class "accesshide", for accessibility.
*
* Please use cautiously - where possible, text should be visible!
*
* @param string $text Plain text.
* @param string $elem Lowercase element name, default "span".
* @param string $class Additional classes for the element.
* @param string $attrs Additional attributes string in the form, "name='value' name2='value2'"
* @return string HTML string.
*/
function get_accesshide($text, $elem='span', $class='', $attrs='') {
return "<$elem class=\"accesshide $class\" $attrs>$text$elem>";
}
/**
* Return the breadcrumb trail navigation separator.
*
* @return string HTML string.
*/
function get_separator() {
// Accessibility: the 'hidden' slash is preferred for screen readers.
return ' '.link_arrow_right($text='/', $url='', $accesshide=true, 'sep').' ';
}
/**
* Print (or return) a collapsible region, that has a caption that can be clicked to expand or collapse the region.
*
* If JavaScript is off, then the region will always be expanded.
*
* @param string $contents the contents of the box.
* @param string $classes class names added to the div that is output.
* @param string $id id added to the div that is output. Must not be blank.
* @param string $caption text displayed at the top. Clicking on this will cause the region to expand or contract.
* @param string $userpref the name of the user preference that stores the user's preferred default state.
* (May be blank if you do not wish the state to be persisted.
* @param boolean $default Initial collapsed state to use if the user_preference it not set.
* @param boolean $return if true, return the HTML as a string, rather than printing it.
* @return string|void If $return is false, returns nothing, otherwise returns a string of HTML.
*/
function print_collapsible_region($contents, $classes, $id, $caption, $userpref = '', $default = false, $return = false) {
$output = print_collapsible_region_start($classes, $id, $caption, $userpref, $default, true);
$output .= $contents;
$output .= print_collapsible_region_end(true);
if ($return) {
return $output;
} else {
echo $output;
}
}
/**
* Print (or return) the start of a collapsible region
*
* The collapsibleregion has a caption that can be clicked to expand or collapse the region. If JavaScript is off, then the region
* will always be expanded.
*
* @param string $classes class names added to the div that is output.
* @param string $id id added to the div that is output. Must not be blank.
* @param string $caption text displayed at the top. Clicking on this will cause the region to expand or contract.
* @param string $userpref the name of the user preference that stores the user's preferred default state.
* (May be blank if you do not wish the state to be persisted.
* @param boolean $default Initial collapsed state to use if the user_preference it not set.
* @param boolean $return if true, return the HTML as a string, rather than printing it.
* @param string $extracontent the extra content will show next to caption, eg.Help icon.
* @return string|void if $return is false, returns nothing, otherwise returns a string of HTML.
*/
function print_collapsible_region_start($classes, $id, $caption, $userpref = '', $default = false, $return = false,
$extracontent = null) {
global $PAGE;
// Work out the initial state.
if (!empty($userpref) and is_string($userpref)) {
$collapsed = get_user_preferences($userpref, $default);
} else {
$collapsed = $default;
$userpref = false;
}
if ($collapsed) {
$classes .= ' collapsed';
}
$output = '';
$output .= 'print_location_comment(__FILE__, __LINE__);
*
* @param string $file
* @param integer $line
* @param boolean $return Whether to return or print the comment
* @return string|void Void unless true given as third parameter
*/
function print_location_comment($file, $line, $return = false) {
if ($return) {
return "\n";
} else {
echo "\n";
}
}
/**
* Returns true if the user is using a right-to-left language.
*
* @return boolean true if the current language is right-to-left (Hebrew, Arabic etc)
*/
function right_to_left() {
return (get_string('thisdirection', 'langconfig') === 'rtl');
}
/**
* Returns swapped left<=> right if in RTL environment.
*
* Part of RTL Moodles support.
*
* @param string $align align to check
* @return string
*/
function fix_align_rtl($align) {
if (!right_to_left()) {
return $align;
}
if ($align == 'left') {
return 'right';
}
if ($align == 'right') {
return 'left';
}
return $align;
}
/**
* Returns true if the page is displayed in a popup window.
*
* Gets the information from the URL parameter inpopup.
*
* @todo Use a central function to create the popup calls all over Moodle and
* In the moment only works with resources and probably questions.
*
* @return boolean
*/
function is_in_popup() {
$inpopup = optional_param('inpopup', '', PARAM_BOOL);
return ($inpopup);
}
/**
* Progress trace class.
*
* Use this class from long operations where you want to output occasional information about
* what is going on, but don't know if, or in what format, the output should be.
*
* @copyright 2009 Tim Hunt
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
* @package core
*/
abstract class progress_trace {
/**
* Output an progress message in whatever format.
*
* @param string $message the message to output.
* @param integer $depth indent depth for this message.
*/
abstract public function output($message, $depth = 0);
/**
* Called when the processing is finished.
*/
public function finished() {
}
}
/**
* This subclass of progress_trace does not ouput anything.
*
* @copyright 2009 Tim Hunt
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
* @package core
*/
class null_progress_trace extends progress_trace {
/**
* Does Nothing
*
* @param string $message
* @param int $depth
* @return void Does Nothing
*/
public function output($message, $depth = 0) {
}
}
/**
* This subclass of progress_trace outputs to plain text.
*
* @copyright 2009 Tim Hunt
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
* @package core
*/
class text_progress_trace extends progress_trace {
/**
* Output the trace message.
*
* @param string $message
* @param int $depth
* @return void Output is echo'd
*/
public function output($message, $depth = 0) {
mtrace(str_repeat(' ', $depth) . $message);
}
}
/**
* This subclass of progress_trace outputs as HTML.
*
* @copyright 2009 Tim Hunt
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
* @package core
*/
class html_progress_trace extends progress_trace {
/**
* Output the trace message.
*
* @param string $message
* @param int $depth
* @return void Output is echo'd
*/
public function output($message, $depth = 0) {
echo '', str_repeat(' ', $depth), htmlspecialchars($message, ENT_COMPAT), "
\n"; flush(); } } /** * HTML List Progress Tree * * @copyright 2009 Tim Hunt * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later * @package core */ class html_list_progress_trace extends progress_trace { /** @var int */ protected $currentdepth = -1; /** * Echo out the list * * @param string $message The message to display * @param int $depth * @return void Output is echoed */ public function output($message, $depth = 0) { $samedepth = true; while ($this->currentdepth > $depth) { echo "\n\n"; $this->currentdepth -= 1; if ($this->currentdepth == $depth) { echo '