File manager - Edit - /home/missmand/public_html/learning/old/main/inc/lib/xht.lib.php
Back
<?php /* <!-- xht.lib.php --> <!-- XML HTML Templates, 2006/12/14 --> <!-- Copyright (C) 2006 rene.haentjens@UGent.be - see note at end of text --> <!-- Released under the GNU GPL V2, see http://www.gnu.org/licenses/gpl.html --> */ /** * This is an XML HTML template library. * Include/require it in your code to use its functionality. * * This library defines function xht_htmlwchars & class xhtdoc with methods: * - xht_fill_template($template_name) * - xht_substitute($subtext) * * Check htt_error after defining a new xhtdoc(htt_file_contents). * * Assign xht_xmldoc (, xht_get_lang, xht_resource, xht_dbgn) * before calling the class methods. * * @package chamilo.library */ function xht_htmlwchars($s) // use only where ISO-8859-1 is not required! { //return ereg_replace('\[((/?(b|big|i|small|sub|sup|u))|br/)\]', '<\\1>', // str_replace('@�@', '&', // htmlspecialchars(ereg_replace('&#([0-9]+);', '@�@#\\1;', $s)))); global $charset; return api_ereg_replace('\[((/?(b|big|i|small|sub|sup|u))|br/)\]', '<\\1>', str_replace('@�@', '&', htmlspecialchars(api_ereg_replace('&#([0-9]+);', '@�@#\\1;', $s), ENT_QUOTES, $charset))); // replaces htmlspecialchars for double-escaped xml chars like '&#nnn;' // and when html tags <...> are represented as [...] } function xht_is_assoclist($s) // ":key1:value1,, key2:value2,, ..." { return is_string($s) && strpos($s, ',,'); } function xht_explode_assoclist($s) { $result = array(); if (!xht_is_assoclist($s)) return $result; foreach (explode(',,', api_substr($s, 1)) as $keyplusvalue) if ($cp = api_strpos($keyplusvalue, api_substr($s, 0, 1))) $result[trim(api_substr($keyplusvalue, 0, $cp))] = api_substr($keyplusvalue, $cp+1); return $result; } class xhtdoc { var $htt_array; // array with HTML templates var $htt_error; // error while parsing htt_file_contents to templates var $xht_xmldoc; // XML Mini-DOM document for which templates are processed var $xht_param; // parameter array for template processing var $xht_get_lang; // user-supplied function for {-L xx-}, e.g. Dokeos get_lang var $xht_resource; // user-supplied function for '=/' in path var $xht_dbgn; // set to a positive value for debugging output var $xht_dbgo; // debugging output accumulates here var $_prev_param; // old parameter values function xht_fill_template($template_name, $cur_elem = 0) { $template_name = trim($template_name); if (!$template_name || (api_strpos($template_name, ' ') !== FALSE)) return ''; if (!is_string($httext = $this->htt_array[$template_name])) return ''; if ($this->xht_dbgn) $this->xht_dbgo .= '<!-- ' . XHT_LP . $template_name . ' ' . XHT_RP . $this->_show_param() . " -->\n"; $prev_lpp = 0; $prev_sub = ''; $scanpos = 0; while (($rpp = api_strpos($httext, XHT_RP, $scanpos)) !== FALSE) // first -} { if (($lpp = api_strpos($httext, XHT_LP)) === FALSE) break; // no {- for -} if ($lpp > $rpp) break; // no {- preceding -} while (($next_lpp = api_strpos($httext, XHT_LP, $lpp + XHT_PL)) !== FALSE) if ($next_lpp > $rpp) break; else $lpp = $next_lpp; // find {- closest to -} $subtext = api_substr($httext, $lpp + XHT_PL, $rpp - $lpp - XHT_PL); $httext = api_substr($httext, 0, $lpp) . $this->xht_substitute($subtext, $cur_elem, XHT_LP, XHT_RP) . api_substr($httext, $rpp + XHT_PL); // substitute or leave intact if ($lpp == $prev_lpp && $subtext == $prev_sub) // prevent looping { $scanpos = $rpp + 1; $prev_lpp = 0; $prev_sub = ''; } else { $prev_lpp = $lpp; $prev_sub = $subtext; } } return $httext; } function xht_substitute($subtext, $cur_elem = 0, $pre = '', $post = '') { global $charset; $regs = array(); // for use with ereg() if (!api_ereg(XHT_SUBS2, $subtext, $regs) && !api_ereg(XHT_SUBS1, $subtext, $regs)) return $pre . $subtext . $post; $type = $regs[1]; $text = $regs[2]; $result = ''; $subnum = FALSE; $subtext = isset($regs[3]) ? $regs[3] : ''; if ($this->xht_dbgn) // generate debugging output, with new number $subnum { $subnum = ++ $this->xht_dbgn; $this->xht_dbgo .= '<!-- ' . XHT_LP . $type . $subnum . '+ ' . htmlspecialchars($text, ENT_QUOTES, $charset) . ' ' . XHT_RP . $this->_show_param() . " -->\n"; } if ($type == 'D') // Define, e.g. {-D key value-} { // Assign the value to parameter [key] $this->xht_param[$text] = $subtext; // used to be: = $this->xht_substitute($subtext, $cur_elem); } elseif ($type == 'E') // Escape, e.g. {-E userFunction subtext-} { $result = call_user_func($text, FALSE); // get cached result, if any if ($result === FALSE) // no cached result available { $result = $this->xht_substitute($subtext, $cur_elem); $result = call_user_func($text, $result); // allow to cache } } elseif ($type == 'R') // Repeat, e.g. {-R general/title/string subtext-} { $rdepthn = 'rdepth' . (++ $this->xht_param['rdepth']); $n = 0; $this->xht_param['number'] = '0'; if (is_array($a = $this->_lang($text, $cur_elem))) // repeat on array { foreach ($a as $key => $value) { $this->xht_param['number'] = (string) (++ $n); $this->xht_param[$rdepthn] = (string) ($n); $this->xht_param['key'] = $key; $this->xht_param['value'] = $value; $result .= $this->xht_substitute($subtext, $cur_elem); } } elseif (xht_is_assoclist($a)) // repeat on associative list { foreach (xht_explode_assoclist($a) as $key => $value) { $this->xht_param['number'] = (string) (++ $n); $this->xht_param[$rdepthn] = (string) ($n); $this->xht_param['key'] = $key; $this->xht_param['value'] = $value; $result .= $this->xht_substitute($subtext, $cur_elem); } } elseif (!is_object($this->xht_xmldoc)) { $result = '? R Error: no XML doc has been assigned to xht_xmldoc'; } else // repeat on XML elements { $sub_elems = $this->xht_xmldoc->xmd_select_elements($text, $cur_elem); foreach ($sub_elems as $subElem) { $this->xht_param['number'] = (string) (++ $n); $this->xht_param[$rdepthn] = (string) ($n); $result .= $this->xht_substitute($subtext, $subElem); } } // removed 2004/10/05: template_array (security) // added 2005/03/08: associative list (lang arrays deprecated) $this->xht_param['rdepth'] --; // As there is only one ['number'] or one set ['key'] + ['value'], // using them in nested repeats may not have the desired result. } elseif ($type == 'T') // Test, e.g. {-T key1 == key2 text-} { if (api_ereg('^(=|==|<=|<|!=|<>|>|>=) +([^ ]+) +(.*)$', $subtext, $regs)) { // Comparand= parameter value if set, else languagevar value $cmp1 = isset($this->xht_param[$text]) ? $this->xht_param[$text] : $this->_lang($text, $cur_elem); $cmp3 = isset($this->xht_param[$cmp3 = $regs[2]]) ? $this->xht_param[$cmp3] : $this->_lang($cmp3, $cur_elem); $cmp = strcmp($cmp1, $cmp3); $op = ' ' . $regs[1] . ' '; if ($subnum) $this->xht_dbgo .= '<!-- ' . XHT_LP . $type . $subnum . ' ' . htmlspecialchars($cmp1.$op.$cmp3.' = '.$cmp, ENT_QUOTES, $charset) . ' ' . XHT_RP . " -->\n"; // debugging output if ( ($cmp < 0 && api_strpos(' <= < != <> ', $op)) || ($cmp == 0 && api_strpos(' = == <= >= ', $op)) || ($cmp > 0 && api_strpos(' != <> > >= ', $op)) ) $result = $this->xht_substitute($regs[3], $cur_elem); // else $result is empty } else { $result = $pre . $subtext . $post; } } else { if (api_strpos('CLPVX', $type) !== FALSE) // used to be always $text = $this->xht_substitute($text, $cur_elem); // nested escape if ($type == 'C') // Call, e.g. {-C SUBTEMPLATE-} $result = $this->xht_fill_template($text, $cur_elem); elseif ($type == 'H') $result = htmlspecialchars($text, ENT_QUOTES, $charset); elseif ($type == 'L') $result = $this->_lang($text, $cur_elem); elseif ($type == 'P') $result = $this->xht_param[$text]; elseif ($type == 'U') $result = urlencode($text); elseif ($type == 'W') $result = xht_htmlwchars($text); elseif (!is_object($this->xht_xmldoc)) { $result = '? V/X Error: no XML doc has been assigned to xht_xmldoc'; } else // $type == 'V' or 'X' { if (api_ereg('^(.*)=/(.+)$', $text, $regs)) // special resource-marker { $path = $regs[1]; $text = $regs[2]; if (api_substr($path, -1) == '/') $path = api_substr($path, 0, -1); if ($path) $cur_elem = $this->xht_xmldoc-> xmd_select_single_element($path, $cur_elem); $cur_elem = call_user_func($this->xht_resource, $cur_elem); } $result = ($type == 'V') ? $this->xht_xmldoc->xmd_value($text, $cur_elem) : $this->xht_xmldoc-> xmd_html_value($text, $cur_elem, 'xht_htmlwchars'); } } if ($subnum) $this->xht_dbgo .= '<!-- ' . XHT_LP . $type . $subnum . '- ' . htmlspecialchars((api_strlen($result) <= 60) ? $result . ' ': api_substr($result, 0, 57).'...', ENT_QUOTES, $charset) . XHT_RP . " -->\n"; return $result; } function xht_add_template($template_name, $httext) { $this->htt_array[$template_name] = $httext; // removed 2004/10/05: (substr($httext, 0, 6) == 'array(') ? // removed 2004/10/05: @eval('return ' . $httext . ';') : $httext; if (!$this->htt_array[$template_name]) { $this->htt_error = 'template ' . $template_name . ' is empty or invalid'; return; } } function xhtdoc($htt_file_contents) { $htt_file_contents = // normalize \r (Mac) and \r\n (Windows) to \n str_replace("\r", "\n", str_replace("\r\n", "\n", $htt_file_contents)); while (api_substr($htt_file_contents, -1) == "\n") $htt_file_contents = api_substr($htt_file_contents, 0, -1); $last_line = api_strrchr($htt_file_contents, "\n"); $this->htt_error = ''; if (api_strlen($last_line) < 12 || api_substr($last_line, 0, 6) != "\n<!-- " || api_strlen($last_line) % 2 != 0 || api_substr($last_line, -4) != " -->") { $this->htt_error = 'last line must be of the form <!-- {--} -->'; return; } define('XHT_PL', (int) (api_strlen($last_line) - 10) / 2); // Parentheses Lth define('XHT_LP', api_substr($last_line, 6, XHT_PL)); // Left Par define('XHT_RP', api_substr($last_line, 6 + XHT_PL, XHT_PL)); // Right Par if (XHT_LP == XHT_RP) { $this->htt_error = 'parentheses in last line <!-- {--} --> must not be identical'; return; } $this->htt_array = array(); // named elements are arrays and strings foreach (explode("\n<!-- " . XHT_LP, "\n" . $htt_file_contents) as $name_and_text) { if (($name_length = api_strpos($name_and_text, XHT_RP . " -->\n"))) { $template_name = trim(api_substr($name_and_text, 0, $name_length)); if (api_strpos($template_name, ' ') !== FALSE) give_up('Template ' . $template_name . ' has a space in its name'); $httext = api_substr($name_and_text, $name_length + XHT_PL + 5); while (api_substr($httext, 0, 1) == "\n") $httext = api_substr($httext, 1); while (api_substr($httext, -1) == "\n") $httext = api_substr($httext,0,-1); $this->xht_add_template($template_name, $httext); } } define('XHT_SUBS1', '^(C|H|L|P|U|V|W|X) +(.*)$'); // substitution types 1: // Call, Htmlchars, Lang, Param, Urlencode, Value, Wchars, Xvalue define('XHT_SUBS2', '^(D|E|R|T) +([^ ]+) +(.*)$'); // substitution types 2: // Define, Escape, Repeat, Test $this->xht_dbgo = ''; $this->xht_param = array(0 => '0', 1 => '1', '' => '', 'empty' => '', 'rdepth' => 0); $this->_prev_param = $this->xht_param; // empty: {-R * P empty-} puts the number of subelements in {-P number-} // rdepth: current number of nested R's // rdepth1, rdepth2, ...: key or number, see R } function _show_param() // for debugging info { global $charset; $result = ''; foreach ($this->xht_param as $k => $v) if ($v !== $this->_prev_param[$k]) { $this->_prev_param[$k] = $v; $result .= ', ' . $k . ': ' . ((api_strlen($v) <= 20) ? $v : api_substr($v, 0, 17).'...'); } return $result ? htmlspecialchars(api_substr($result, 1), ENT_QUOTES, $charset) : ''; } function _lang($var, $cur_elem = 0) { $result = @call_user_func($this->xht_get_lang, $var, $cur_elem); // This is a hack for proper working of the language selectors // in the forms that are generated through templates. if ($var == 'Langs') { global $langLangs; if (isset($langLangs)) { $result = $langLangs; } } return isset($result) ? $result : $var; } } /* A word of explanation... The last line of a template file (example below) defines the special markers that are used around template names such as INPUT and OPTION and around HTML escapes such as 'P value' and 'L Store', { and } in the example. You can also use markers of more than one character as long as both have the same length, e.g. <% and %>. The markers must not be equal. In templates with JavaScript or <style>, the use of { and } might be confusing; however, it does work. A template starts with a special comment line giving the name of the template and ends where the next template starts. Templates contain escapes of the form {one-letter the-rest}, e.g. {L IdentifierTip}, where the-rest can contain a nested-escape as in e.g. {R metadata/lom/general/keyword C KEYWORDTEMPLATE}. Multiple spaces count as one in many places, but not in all. Best is to use correct spacing. In particular, a T escape such as the example below requires a space at the end of the first line: {T x == y text } Names (of templates, parameters, langvars and user-functions) are case sensitive and cannot contain spaces. Templates are called with a certain currency in the XML doc, usually the root element; for an xml-path repeat, the found element is the current one. One single array with named elements and string values contains the template parameters; they are shared for all template processing. Initially, params contains 0 => '0', 1 => '1', '' => '', 'empty' => '', 'rdepth' => 0. rdepth keeps track of the current number of nested repeats. In a repeat, params number, key, value and rdepthN (N is 1, 2, ...) get assigned values. Types of escapes: C template-name/nested-escape Call a sub-template D parameter-name text Define a parameter value E user-function nested-escape Escape to user-function or do nested-escape H text HtmlSpecialChars (e.g. transform < to <) L languagevar/nested-escape Language variable value (see also below) P parameter-name/nested-escape Parameter value R repeat-part nested-escape Repeat nested-escape for each ... (see below) T key1 operator key2 nested-esc Test, do nested-esc if test succeeds (id.) U text UrlEncode (e.g. transform < to %XX) V xml-path/nested-escape Value from XML document element or attribute W text xht_htmlwchars (see below) X extended-xml-path/nested-esc eXtended Value from XML (see below) nested-escape= without the special markers. Nesting with special markers is always possible and in that case the inner nesting is evaluated first. Note that as from the 2005/03/15 version, some implicit nestings are no longer possible, e.g. {H {P key}} and {D label {L Keyword}} now require the inner markers. Escape allows the caller to cache template output: user-function is called with parameter (FALSE); if it returns a string, that string is the result; if it returns FALSE, nested-escape is executed to produce a result and user-function is called again with the result, allowing it to cache the result somewhere; user-function then returns the real result. repeat-part can be: the name of a languagevar which contains an associative list such as: ":key1:value1,, key2:value2,, key3:value3" (note double ,,): nested-escape is repeated for each list element, params 'key' and 'value' refer to the current element; first character defines key-value-separator (here a colon); key cannot contain key-value-separators and is trimmed; value can contain anything except ,, and it is not trimmed; put ',,' at the end of a 0- or 1-element-list. the name of a languagevar which has an array value: nested-escape is repeated for each array element, params 'key' and 'value' refer to the current element. an xml-path to 0, 1 or more elements in the XML document: nested-escape is repeated for each element, param 'number' = 1, 2, ... test operators compare strings: = == <= < != <> > >= key1 and key2 can be parameter-name or languagevar-name. xml-path: see XML Mini-DOM; examples: organizations/@default, body/p[1] (1=first), keyword/*, node/*[-3] / * /... starts from XML document root, regardless of context other extensions: .. - + @* @. (parent, prev, next, number, name) .. stops at the root if too many -name, +name and @*name means only elements of that name xht_htmlwchars is like HtmlSpecialChars, but moreover translates: [b] to <b>, likewise for big, i, small, sub, sup, u, br, and closing tags; &#nnn; to &#nnn; (double-escape for non-UTF8-channels) eXtended Value: see method xmd_html_value of the XML Mini-DOM; the values (but not the pre-, in- and suffixes) are always W-processed; extended-xml-path examples: 'keyword/string ,' generates e.g. 'kwd1,kwd2,kwd3' 'keyword/string , ' generates e.g. 'kwd1, kwd2, kwd3' '( -% keyword/string , %- )' generates e.g. '(kwd1,kwd2,kwd3)' but will generate nothing if no keywords are found in the XML doc; note that the special markers ' -% ' and ' %- ' must have the spaces. V and X escapes can have an xml-path containing '=/': this calls a user-supplied function for finding an associated element. It can e.g. be used when reading a SCORM manifest, for finding the <resource> of an <item>. L also calls a user-supplied function. Its main target is language-dependent texts, e.g. Dokeos 'get_lang'. But the current element is passed as second argument, allowing other functionality in the callback. Array templates functionality has been removed on 2004/10/05, because of a security issue when users can submit templates. -------------------------------------------------------------------------------- Example template file: <!-- {HEAD} --> <style type="text/css"> body {font-family: Arial} </style> <!-- {METADATA} --> <h3>{H {L Tool}: {P entry}}</h3> <table> <tr> <td>{D label {L Language}}{D tip {L LanguageTip}}{C LABEL}</td> <td><select>{D thislang {V general/language}}{R Langs C OPTION}</select></td> <td><input type="text" disabled value="urn:ugent-be:minerva."/></td> </tr> {R general/keyword C KEYWORD} </table> <!---------------------- E-n-d Of script Output ----------------------> <!-- {LABEL} --> <span title="{H {P tip}}">{H {P label}} :</span> <!-- {OPTION} --> <option value="{H {P key}}" {T key == thislang selected}>{H {P value}}</option> <!-- {INPUT} --> <input type="text" title="{P title}" class="wide" value="{H {P value}}"/> <!-- {KEYWORD} --> <tr> <td>{D label {L Keyword}}{D tip {L KeywordTip}}{C LABEL}</td> <td nowrap><select>{D thislang {V string/@language}}{R Langs C OPTION}</select></td> <td>{D value {X string}}{D title general/keyword[{P number}]/string}{C INPUT}</td> </tr> <!-- {} --> -------------------------------------------------------------------------------- <!-- This program 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 2 of the License, or (at your option) any later version. This program 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. --> */ ?>
| ver. 1.4 |
Github
|
.
| PHP 7.4.33 | Generation time: 0.01 |
proxy
|
phpinfo
|
Settings