version 1.17, 2003/02/25 22:50:11
|
version 1.62, 2005/02/28 19:08:11
|
Line 36
|
Line 36
|
# The C source of the Code may not be distributed by the Licensee |
# The C source of the Code may not be distributed by the Licensee |
# to any other parties under any circumstances. |
# to any other parties under any circumstances. |
# |
# |
# 05/29/00,05/30,10/11,10/20 Gerd Kortemeyer |
|
# 5/4 Gerd Kortemeyer |
|
|
|
package Apache::lontexconvert; |
package Apache::lontexconvert; |
|
|
use strict; |
use strict; |
use tth; |
use tth(); |
use vars qw($errorstring); |
use vars qw($errorstring); |
use Apache::lonmsg; |
use Apache(); |
use Apache::lonxml; |
use Apache::lonmsg(); |
use Apache::lonmenu; |
use Apache::lonxml(); |
|
use Apache::lonmenu(); |
|
use Apache::lonlocal; |
|
|
# ====================================================================== Header |
# ====================================================================== Header |
|
|
|
sub init_tth { |
|
my $options=$ENV{'course.'.$ENV{'request.course.id'}.'.tthoptions'}; |
|
if ($ENV{'browser.mathml'}) { |
|
&tth::ttminit(); |
|
if ($ENV{'browser.unicode'}) { |
|
&tth::ttmoptions('-L -u1 '.$options); |
|
} else { |
|
&tth::ttmoptions('-L -u0 '.$options); |
|
} |
|
} else { |
|
&tth::tthinit(); |
|
if ($ENV{'browser.unicode'}) { |
|
&tth::tthoptions('-L -u1 '.$options); |
|
} else { |
|
&tth::tthoptions('-L -u0 '.$options); |
|
} |
|
} |
|
} |
|
|
sub header { |
sub header { |
$errorstring=''; |
$errorstring=''; |
my $time=time; |
my $time=time; |
if ($ENV{'browser.mathml'}) { |
&init_tth(); |
&tth::ttminit(); |
return &Apache::lonxml::xmlbegin(). |
if ($ENV{'browser.unicode'}) { |
"\n<head>\n". |
&tth::ttmoptions('-L -u1'); |
&Apache::lonxml::fontsettings(). |
} else { |
&Apache::lonmenu::registerurl(undef,'tex'). |
&tth::ttmoptions('-L -u0'); |
"\n</head>\n"; |
} |
|
} else { |
|
&tth::tthinit(); |
|
if ($ENV{'browser.unicode'}) { |
|
&tth::tthoptions('-L -u1'); |
|
} else { |
|
&tth::tthoptions('-L -u0'); |
|
} |
|
} |
|
return &Apache::lonxml::xmlbegin(). |
|
&Apache::lonxml::fontsettings(). |
|
"\n<head>\n". |
|
&Apache::lonmenu::registerurl(undef,'tex'). |
|
"\n</head>\n"; |
|
} |
} |
|
|
# ================================================================== Conversion |
# ================================================================== Conversion |
|
|
sub converted { |
$Apache::lontexconvert::messedup=0; |
my $texstring=shift; |
|
my $xmlstring='[UNDISPLAYABLE]'; |
# we need this routine because &converted can get called from inside |
eval(<<'ENDCONV'); |
# of the safespace (through &xmlparse('<m>stuff</m>') which doesn't |
{ |
# allow the opcode for alarm, so we need to compile this before we get |
local $SIG{SEGV}=sub { die; }; |
# into the safe space since opcode checks only occur at compile time |
|
sub convert_real { |
|
my ($texstring)=@_; |
|
my ($xmlstring,$errorstring); |
|
local $SIG{SEGV}=sub { $Apache::lontexconvert::messedup=1; die; }; |
|
local $SIG{ALRM}=sub { |
|
&Apache::lonnet::logthis("ALRM"); |
|
$xmlstring='['.&mt("TeX unconverted due to errors").']'; |
|
$Apache::lontexconvert::messedup=1; |
|
die &mt("TeX unconverted due to errors"); |
|
}; |
|
alarm($Apache::lonnet::perlvar{'lonScriptTimeout'}); |
if ($ENV{'browser.mathml'}) { |
if ($ENV{'browser.mathml'}) { |
$xmlstring=&tth::ttm($$texstring); |
$xmlstring=&tth::ttm($$texstring); |
$xmlstring=~s/\<math\>/\<math xmlns=\"\&mathns\;\"\>/g; |
$xmlstring=~s/\<math\>/\<math xmlns=\"\&mathns\;\"\>/g; |
$xmlstring=~s/\<br\>/\<br\/\>/g; |
$xmlstring=~s/\<br\>/\<br\/\>/g; |
$xmlstring=~s/\<p\>/\<p\>\<\/p\>/g; |
$xmlstring=~s/\<p\>/\<p\>\<\/p\>/g; |
$errorstring.=&tth::ttmerror(); |
$errorstring.=&tth::ttmerror(); |
} else { |
} else { |
$xmlstring=&tth::tth($$texstring); |
$xmlstring=&tth::tth($$texstring); |
$errorstring.=&tth::ttherror(); |
$errorstring.=&tth::ttherror(); |
|
$xmlstring=~s-</font(\s*)>-</font>-g; |
} |
} |
} |
$xmlstring=~s/^\s*\<br clear\=\"all\"/\<br/s; |
|
$xmlstring=~s/^\s*//; |
|
$xmlstring=~s/\s*$//; |
|
alarm(0); |
|
return ($xmlstring,$errorstring); |
|
} |
|
|
|
sub tth_converted { |
|
my $texstring=shift; |
|
my $xmlstring='['.&mt('UNDISPLAYABLE').']'; |
|
if ($Apache::lontexconvert::messedup) { |
|
return '['.&mt('TeX unconverted due to previous errors').']'; |
|
} |
|
$$texstring ='\\documentstyle{article}'.$$texstring; |
|
|
|
eval(<<'ENDCONV'); |
|
($xmlstring,$errorstring)=&convert_real($texstring) |
ENDCONV |
ENDCONV |
return $xmlstring; |
if ($@) { |
|
$errorstring.=&mt("Evaluation Error: ").$@; |
|
$Apache::lontexconvert::messedup=1; |
|
} |
|
if ($Apache::lontexconvert::messedup || &tth::tthmessedup() || |
|
$errorstring) { |
|
&Apache::lonnet::logthis("Trying to kill myself"); |
|
$Apache::lontexconvert::messedup=1; |
|
my $request=Apache->request(); |
|
$request->child_terminate(); |
|
} |
|
return $xmlstring; |
|
} |
|
|
|
sub clean_out_math_mode { |
|
my ($texstring)=@_; |
|
$$texstring=~s/(?!\\)\$//g; |
|
$$texstring=~s/\\[\)\(\]\[]//g; |
|
$$texstring=~s/\\ensuremath//g; |
|
return ''; |
|
} |
|
|
|
sub displaystyle { |
|
my ($texstring)=@_; |
|
#has a $$ or \[ or \displaystyle in it, guessinng it's display mode |
|
if ($$texstring=~/[^\\]\$\$/ || |
|
$$texstring=~/\\\[/ || |
|
$$texstring=~/\\displaystyle/) { return 1; } |
|
return 0; |
|
} |
|
|
|
sub jsMath_converted { |
|
my $texstring=shift; |
|
my $tag='span'; |
|
if (&displaystyle($texstring)) { $tag='div'; } |
|
&clean_out_math_mode($texstring); |
|
return '<'.$tag.' class="math">'.$$texstring.'</'.$tag.'>'; |
|
} |
|
|
|
sub mimetex_converted { |
|
my $texstring=shift; |
|
my $displaystyle=&displaystyle($texstring); |
|
|
|
&clean_out_math_mode($texstring); |
|
|
|
if ($displaystyle) { |
|
$$texstring='\\displaystyle \\Large '.$$texstring; |
|
} |
|
my $result='<img src="/cgi-bin/mimetex.cgi?'.&Apache::lonnet::escape($$texstring).'" />'; |
|
if ($displaystyle) { |
|
$result='<center>'.$result.'</center>'; |
|
} |
|
return $result; |
|
} |
|
|
|
sub converted { |
|
if ($ENV{'environment.texengine'} eq 'tth') { |
|
return &tth_converted; |
|
} elsif ($ENV{'environment.texengine'} eq 'jsMath') { |
|
return &jsMath_converted; |
|
} elsif ($ENV{'environment.texengine'} eq 'mimetex') { |
|
return &mimetex_converted; |
|
} |
|
return &tth_converted; |
} |
} |
|
|
# ====================================================================== Footer |
# ====================================================================== Footer |
Line 116 sub footer {
|
Line 212 sub footer {
|
|
|
sub to_convert { |
sub to_convert { |
my ($string) = @_; |
my ($string) = @_; |
$string=~s/\<br\s*\/?\>/ /g; |
$string=~s/\<br\s*\/?\>/ /gs; |
|
# $string=~s/\s/ /gs; |
|
$string=&HTML::Entities::decode($string); |
return &converted(\$string); |
return &converted(\$string); |
} |
} |
|
|
sub msgtexconverted { |
sub smiley { |
my $message=shift; |
my $expression=shift; |
|
if ($ENV{'browser.imagesuppress'} eq 'on') { return $expression; } |
|
my %smileys=('\:\-\)' => 'smiley', |
|
'8\-\)' => 'coolsmile', |
|
'8\-(I|\|)' => 'coolindiff', |
|
':\-(I|\|)' => 'neutral', |
|
'\:\-(o|O|\(\))' => 'shocked', |
|
':\-\(' => 'frowny', |
|
'\;\-\)' => 'wink', |
|
'\:\-P' => 'baeh', |
|
'\:\-(\\\|\\/)' => 'hrrm', |
|
'\:\-D' => 'bigsmile', |
|
'\:\-C' => 'angry', |
|
'\:(\'|\`)\-\(' => 'cry', |
|
'\:\-(X|\#)' => 'lipsrsealed', |
|
'\:\-S' => 'huh'); |
|
my $iconpath=$Apache::lonnet::perlvar{'lonIconsURL'}; |
|
foreach (keys %smileys) { |
|
$expression=~s/$_/\<img src="$iconpath\/$smileys{$_}.gif" \/\>/gs; |
|
} |
|
return $expression; |
|
} |
|
|
|
sub msgtexconverted { |
|
my ($message,$email) = @_; |
$errorstring=''; |
$errorstring=''; |
if ($ENV{'browser.mathml'}) { |
&init_tth(); |
&tth::ttminit(); |
my $outmessage=''; |
if ($ENV{'browser.unicode'}) { |
my $tex=0; |
&tth::ttmoptions('-L -u1'); |
foreach (split(/(?:\<\;|\<)\/*m\s*(?:\>\;|\>)/i,$message)) { |
|
if ($tex) { |
|
if ($email) { |
|
$outmessage.='</pre><tt>'.&to_convert($_).'</tt><pre>'; $tex=0; |
|
} else { |
|
$outmessage.=&to_convert($_); $tex=0; |
|
} |
} else { |
} else { |
&tth::ttmoptions('-L -u0'); |
$outmessage.=&smiley($_); $tex=1; |
} |
} |
|
} |
|
if (wantarray) { |
|
return ($outmessage,$errorstring); |
} else { |
} else { |
&tth::tthinit(); |
return $outmessage.$errorstring; |
if ($ENV{'browser.unicode'}) { |
} |
&tth::tthoptions('-L -u1'); |
} |
|
|
|
sub algebra { |
|
use AlgParser; |
|
|
|
my ($string,$target,$style)=@_; |
|
my $parser = new AlgParserWithImplicitExpand; |
|
$string=&prepare_algebra($string); |
|
my $ret = $parser->parse($string); |
|
my $result='['.&mt('Algebra unconverted due to previous errors').']'; |
|
if ( ref($ret) ) { |
|
#$parser->tostring(); |
|
$parser->normalize(); |
|
my $latex=$parser->tolatex(); |
|
$latex=&postprocess_algebra($latex); |
|
if ($style eq 'display') { |
|
$latex='$$'.$latex.'$$x'; |
} else { |
} else { |
&tth::tthoptions('-L -u0'); |
# style is 'inline' |
|
$latex='\\ensuremath{'.$latex.'}'; |
} |
} |
|
if ($target eq 'web' || $target eq 'analyze') { |
|
$result = &converted(\$latex); |
|
} else { |
|
$result = $latex; |
|
} |
|
} else { |
|
&Apache::lonxml::error($parser->{'htmlerror'}); |
} |
} |
$message=~s/(\$\$.+?\$\$)/&to_convert($1)/ge; |
|
$message=~s/(\$.+?\$)/&to_convert($1)/ge; |
|
$message=~s/(\\\(.+?\\\))/&to_convert($1)/ge; |
|
$message=~s/(\\\[.+?\\\])/&to_convert($1)/ge; |
|
return $message.$errorstring; |
|
} |
} |
|
|
|
sub prepare_algebra { |
|
my ($string)=@_; |
|
|
|
# makes the decision about what is a minus sign easier supposedly |
|
$string =~ s/(\<\>|\<\=|\>\=[\=\>\<] *)-/$1 zeroplace -/g; |
|
|
|
return $string; |
|
} |
|
|
|
sub postprocess_algebra { |
|
my ($string)=@_; |
|
|
|
# moodle had these and I don't know why, ignoring them for now |
|
# $string =~s/\\fun/ /g; |
|
|
|
# sqrt(3,4) -> \sqrt\let{3,4\right}, which is annoying |
|
$string =~s/\\left\{/\{/g; |
|
$string =~s/\\right\}/\}/g; |
|
|
|
# remove the extra () in the denominator of a \frac |
|
$string =~s/\\frac{(.+?)}{\\left\((.+?)\\right\)}/\\frac{$1}{$2}/gs; |
|
|
|
# sqrt(3,4) means the 4 root of 3 |
|
$string =~s/\\sqrt{([^,]+),([^\}]+)}/\\sqrt[$2]{$1}/gs; |
|
|
|
# log(3,4) means the log base 4 of 3 |
|
$string =~s/\\log\\left\((.+?),(.+?)\\right\)/\\log_{$2}\\left($1\\right)/gs; |
|
|
|
# log(3,4) means the log base 4 of 3 |
|
$string =~s/\\((?:sin|cos|tan|sec|csc|cot)(?:h)?)\\left\((.+?),(.+?)\\right\)/\\$1^{$3}\\left($2\\right)/gs; |
|
|
|
# int(3,a,b) integral from a to b of 3 |
|
$string =~s/\\int\\left\((.+?),(.+?),(.+?)\\right\)/\\int_{$2}^{$3}\\left($1\\right)/gs; |
|
|
|
# int( ... dx) -> ... |
|
$string =~s/\\int\\left\((.+?)d[a-z]\\right\)/$1/gs; |
|
|
|
# |
|
$string =~s/\\lim\\left\((.+?),(.+?),(.+?)\\right\)/\\lim_{$2\\to $3}$1/gs; |
|
return $string; |
|
} |
1; |
1; |
__END__ |
__END__ |
|
|