mirror of
https://github.com/moodle/moodle.git
synced 2025-08-06 01:16:44 +02:00

There are two well-known cases where the algebra filter messes up input that is obviously not meant for the algebra filter: 1. Copy and paste of unified diffs. 2. @@PLUGINFILE@@ tokens in the HTML that are due to be replaced by the files API. This fix detects these two cases, and just stops the algebra filter from replacing them.
250 lines
13 KiB
PHP
250 lines
13 KiB
PHP
<?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/>.
|
|
|
|
/**
|
|
* Moodle - Filter for converting simple calculator-type algebraic
|
|
* expressions to cached gif images
|
|
*
|
|
* @package filter
|
|
* @subpackage algebra
|
|
* @copyright 2004 Zbigniew Fiedorowicz fiedorow@math.ohio-state.edu
|
|
* Originally based on code provided by Bruno Vernier bruno@vsbeducation.ca
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
*/
|
|
|
|
defined('MOODLE_INTERNAL') || die();
|
|
|
|
//-------------------------------------------------------------------------
|
|
// NOTE: This Moodle text filter converts algebraic expressions delimited
|
|
// by either @@...@@ or by <algebra...>...</algebra> tags
|
|
// first converts it to TeX using WeBWorK algebra parser Perl library
|
|
// AlgParser.pm, part of the WeBWorK distribution obtained from
|
|
// http://webhost.math.rochester.edu/downloadwebwork/
|
|
// then converts the TeX to gif images using
|
|
// mimetex.cgi obtained from http://www.forkosh.com/mimetex.html authored by
|
|
// John Forkosh john@forkosh.com. The mimetex.cgi ELF binary compiled for Linux i386
|
|
// as well as AlgParser.pm are included with this distribution.
|
|
// Note that there may be patent restrictions on the production of gif images
|
|
// in Canada and some parts of Western Europe and Japan until July 2004.
|
|
//-------------------------------------------------------------------------
|
|
// You will then need to edit your moodle/config.php to invoke mathml_filter.php
|
|
//-------------------------------------------------------------------------
|
|
|
|
function filter_algebra_image($imagefile, $tex= "", $height="", $width="", $align="middle") {
|
|
// Given the path to a picture file in a course, or a URL,
|
|
// this function includes the picture in the page.
|
|
global $CFG, $OUTPUT;
|
|
|
|
$output = "";
|
|
$style = 'style="border:0px; vertical-align:'.$align.';';
|
|
$title = '';
|
|
if ($tex) {
|
|
$tex = html_entity_decode($tex, ENT_QUOTES, 'UTF-8');
|
|
$title = 'title="'.s($tex).'"';
|
|
}
|
|
if ($height) {
|
|
$style .= " height:{$height}px;";
|
|
}
|
|
if ($width) {
|
|
$style .= " width:{$width}px;";
|
|
}
|
|
$style .= '"';
|
|
$anchorcontents = '';
|
|
if ($imagefile) {
|
|
$anchorcontents .= "<img $title alt=\"".s($tex)."\" src=\"";
|
|
if ($CFG->slasharguments) { // Use this method if possible for better caching
|
|
$anchorcontents .= "$CFG->wwwroot/filter/algebra/pix.php/$imagefile";
|
|
} else {
|
|
$anchorcontents .= "$CFG->wwwroot/filter/algebra/pix.php?file=$imagefile";
|
|
}
|
|
$anchorcontents .= "\" $style />";
|
|
|
|
if (!file_exists("$CFG->dataroot/filter/algebra/$imagefile") && has_capability('moodle/site:config', context_system::instance())) {
|
|
$link = '/filter/algebra/algebradebug.php';
|
|
$action = null;
|
|
} else {
|
|
$link = new moodle_url('/filter/tex/displaytex.php', array('texexp'=>$tex));
|
|
$action = new popup_action('click', $link, 'popup', array('width'=>320,'height'=>240)); //TODO: the popups do not work when text caching is enabled!!
|
|
}
|
|
$output .= $OUTPUT->action_link($link, $anchorcontents, $action, array('title'=>'TeX'));
|
|
|
|
} else {
|
|
$output .= "Error: must pass URL or course";
|
|
}
|
|
return $output;
|
|
}
|
|
|
|
class filter_algebra extends moodle_text_filter {
|
|
public function filter($text, array $options = array()){
|
|
global $CFG, $DB;
|
|
|
|
/// Do a quick check using stripos to avoid unnecessary wor
|
|
if (!preg_match('/<algebra/i',$text) && !strstr($text,'@@')) {
|
|
return $text;
|
|
}
|
|
|
|
//restrict filtering to forum 130 (Maths Tools on moodle.org)
|
|
# $scriptname = $_SERVER['SCRIPT_NAME'];
|
|
# if (!strstr($scriptname,'/forum/')) {
|
|
# return $text;
|
|
# }
|
|
# if (strstr($scriptname,'post.php')) {
|
|
# $parent = forum_get_post_full($_GET['reply']);
|
|
# $discussion = $DB->get_record("forum_discussions",array("id"=>$parent->discussion));
|
|
# } else if (strstr($scriptname,'discuss.php')) {
|
|
# $discussion = $DB->get_record("forum_discussions",array("id"=>$_GET['d']));
|
|
# } else {
|
|
# return $text;
|
|
# }
|
|
# if ($discussion->forum != 130) {
|
|
# return $text;
|
|
# }
|
|
|
|
preg_match_all('/@(@@+)([^@])/',$text,$matches);
|
|
for ($i=0;$i<count($matches[0]);$i++) {
|
|
$replacement = str_replace('@','@',$matches[1][$i]).$matches[2][$i];
|
|
$text = str_replace($matches[0][$i],$replacement,$text);
|
|
}
|
|
|
|
// <algebra> some algebraic input expression </algebra>
|
|
// or @@ some algebraic input expression @@
|
|
|
|
preg_match_all('/<algebra>(.+?)<\/algebra>|@@(.+?)@@/is', $text, $matches);
|
|
for ($i=0; $i<count($matches[0]); $i++) {
|
|
$algebra = $matches[1][$i] . $matches[2][$i];
|
|
|
|
// Look for some common false positives, and skip processing them.
|
|
if ($algebra == 'PLUGINFILE' || $algebra == 'DRAFTFILE') {
|
|
// Raw pluginfile URL.
|
|
continue;
|
|
}
|
|
if (preg_match('/^ -\d+(,\d+)? \+\d+(,\d+)? $/', $algebra)) {
|
|
// Part of a unified diff.
|
|
continue;
|
|
}
|
|
|
|
$algebra = str_replace('<nolink>','',$algebra);
|
|
$algebra = str_replace('</nolink>','',$algebra);
|
|
$algebra = str_replace('<span class="nolink">','',$algebra);
|
|
$algebra = str_replace('</span>','',$algebra);
|
|
$align = "middle";
|
|
if (preg_match('/^align=bottom /',$algebra)) {
|
|
$align = "text-bottom";
|
|
$algebra = preg_replace('/^align=bottom /','',$algebra);
|
|
} else if (preg_match('/^align=top /',$algebra)) {
|
|
$align = "text-top";
|
|
$algebra = preg_replace('/^align=top /','',$algebra);
|
|
}
|
|
$md5 = md5($algebra);
|
|
$filename = $md5 . ".gif";
|
|
if (! $texcache = $DB->get_record("cache_filters",array("filter"=>"algebra", "md5key"=>$md5))) {
|
|
$algebra = str_replace('<','<',$algebra);
|
|
$algebra = str_replace('>','>',$algebra);
|
|
$algebra = str_replace('<>','#',$algebra);
|
|
$algebra = str_replace('<=','%',$algebra);
|
|
$algebra = str_replace('>=','!',$algebra);
|
|
$algebra = preg_replace('/([=><%!#] *)-/',"\$1 zeroplace -",$algebra);
|
|
$algebra = str_replace('delta','zdelta',$algebra);
|
|
$algebra = str_replace('beta','bita',$algebra);
|
|
$algebra = str_replace('theta','thita',$algebra);
|
|
$algebra = str_replace('zeta','zita',$algebra);
|
|
$algebra = str_replace('eta','xeta',$algebra);
|
|
$algebra = str_replace('epsilon','zepslon',$algebra);
|
|
$algebra = str_replace('upsilon','zupslon',$algebra);
|
|
$algebra = preg_replace('!\r\n?!',' ',$algebra);
|
|
$algebra = escapeshellarg($algebra);
|
|
if ( (PHP_OS == "WINNT") || (PHP_OS == "WIN32") || (PHP_OS == "Windows")) {
|
|
$cmd = "cd $CFG->dirroot\\filter\\algebra & algebra2tex.pl $algebra";
|
|
} else {
|
|
$cmd = "cd $CFG->dirroot/filter/algebra; ./algebra2tex.pl $algebra";
|
|
}
|
|
$texexp = `$cmd`;
|
|
if (preg_match('/parsehilight/',$texexp)) {
|
|
$text = str_replace( $matches[0][$i],"<b>Syntax error:</b> " . $texexp,$text);
|
|
} else if ($texexp) {
|
|
$texexp = str_replace('zeroplace','',$texexp);
|
|
$texexp = str_replace('#','\not= ',$texexp);
|
|
$texexp = str_replace('%','\leq ',$texexp);
|
|
$texexp = str_replace('!','\geq ',$texexp);
|
|
$texexp = str_replace('\left{','{',$texexp);
|
|
$texexp = str_replace('\right}','}',$texexp);
|
|
$texexp = str_replace('\fun',' ',$texexp);
|
|
$texexp = str_replace('infty','\infty',$texexp);
|
|
$texexp = str_replace('alpha','\alpha',$texexp);
|
|
$texexp = str_replace('gamma','\gamma',$texexp);
|
|
$texexp = str_replace('iota','\iota',$texexp);
|
|
$texexp = str_replace('kappa','\kappa',$texexp);
|
|
$texexp = str_replace('lambda','\lambda',$texexp);
|
|
$texexp = str_replace('mu','\mu',$texexp);
|
|
$texexp = str_replace('nu','\nu',$texexp);
|
|
$texexp = str_replace('xi','\xi',$texexp);
|
|
$texexp = str_replace('rho','\rho',$texexp);
|
|
$texexp = str_replace('sigma','\sigma',$texexp);
|
|
$texexp = str_replace('tau','\tau',$texexp);
|
|
$texexp = str_replace('phi','\phi',$texexp);
|
|
$texexp = str_replace('chi','\chi',$texexp);
|
|
$texexp = str_replace('psi','\psi',$texexp);
|
|
$texexp = str_replace('omega','\omega',$texexp);
|
|
$texexp = str_replace('zdelta','\delta',$texexp);
|
|
$texexp = str_replace('bita','\beta',$texexp);
|
|
$texexp = str_replace('thita','\theta',$texexp);
|
|
$texexp = str_replace('zita','\zeta',$texexp);
|
|
$texexp = str_replace('xeta','\eta',$texexp);
|
|
$texexp = str_replace('zepslon','\epsilon',$texexp);
|
|
$texexp = str_replace('zupslon','\upsilon',$texexp);
|
|
$texexp = str_replace('\mbox{logten}','\mbox{log}_{10}',$texexp);
|
|
$texexp = str_replace('\mbox{acos}','\mbox{cos}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{asin}','\mbox{sin}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{atan}','\mbox{tan}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{asec}','\mbox{sec}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{acsc}','\mbox{csc}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{acot}','\mbox{cot}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{acosh}','\mbox{cosh}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{asinh}','\mbox{sinh}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{atanh}','\mbox{tanh}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{asech}','\mbox{sech}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{acsch}','\mbox{csch}^{-1}',$texexp);
|
|
$texexp = str_replace('\mbox{acoth}','\mbox{coth}^{-1}',$texexp);
|
|
//$texexp = preg_replace('/\\\frac{(.+?)}{\\\left\((.+?)\\\right\)}/s','\frac{'."\$1}{\$2}",$texexp);
|
|
$texexp = preg_replace('/\\\sqrt{(.+?),(.+?)}/s','\sqrt['. "\$2]{\$1}",$texexp);
|
|
$texexp = preg_replace('/\\\mbox{abs}\\\left\((.+?)\\\right\)/s',"|\$1|",$texexp);
|
|
$texexp = preg_replace('/\\\log\\\left\((.+?),(.+?)\\\right\)/s','\log_{'. "\$2}\\left(\$1\\right)",$texexp);
|
|
$texexp = preg_replace('/(\\\cos|\\\sin|\\\tan|\\\sec|\\\csc|\\\cot)([h]*)\\\left\((.+?),(.+?)\\\right\)/s',"\$1\$2^{". "\$4}\\left(\$3\\right)",$texexp);
|
|
$texexp = preg_replace('/\\\int\\\left\((.+?),(.+?),(.+?)\\\right\)/s','\int_'. "{\$2}^{\$3}\$1 ",$texexp);
|
|
$texexp = preg_replace('/\\\int\\\left\((.+?d[a-z])\\\right\)/s','\int '. "\$1 ",$texexp);
|
|
$texexp = preg_replace('/\\\lim\\\left\((.+?),(.+?),(.+?)\\\right\)/s','\lim_'. "{\$2\\to \$3}\$1 ",$texexp);
|
|
$texexp = str_replace('\mbox', '', $texexp); // now blacklisted in tex, sorry
|
|
$texcache = new stdClass();
|
|
$texcache->filter = 'algebra';
|
|
$texcache->version = 1;
|
|
$texcache->md5key = $md5;
|
|
$texcache->rawtext = $texexp;
|
|
$texcache->timemodified = time();
|
|
$DB->insert_record("cache_filters", $texcache, false);
|
|
$text = str_replace( $matches[0][$i], filter_algebra_image($filename, $texexp, '', '', $align), $text);
|
|
} else {
|
|
$text = str_replace( $matches[0][$i],"<b>Undetermined error:</b> ",$text);
|
|
}
|
|
} else {
|
|
$text = str_replace( $matches[0][$i], filter_algebra_image($filename, $texcache->rawtext), $text);
|
|
}
|
|
}
|
|
return $text;
|
|
}
|
|
}
|
|
|
|
|