--- loncom/xml/lontexconvert.pm 2011/05/27 18:34:07 1.107 +++ loncom/xml/lontexconvert.pm 2019/02/15 17:52:54 1.112.2.8 @@ -1,7 +1,7 @@ # The LearningOnline Network with CAPA # TeX Conversion Module # -# $Id: lontexconvert.pm,v 1.107 2011/05/27 18:34:07 raeburn Exp $ +# $Id: lontexconvert.pm,v 1.112.2.8 2019/02/15 17:52:54 raeburn Exp $ # # Copyright Michigan State University Board of Trustees # @@ -50,6 +50,78 @@ use LONCAPA; use URI::Escape; use IO::Socket::INET; + +# +# Table of substitutions to unicode characters. +# + +my %unicode_harpoons = ( + '\rightleftharpoons' => 0x21cc, + ); + +my %unicode_translations = ( + +# Brackets - unicode for browsers/OS which support it. + + '' => 0x23a1, + '' => 0x23a2, + '' => 0x23a3, + '' => 0x23a4, + '' => 0x23a5, + '' => 0x23a6, + +# Parens - unicode for browsers/OS which support it + + '' => 0x239b, + '' => 0x239c, + '' => 0x239d, + '' => 0x239e, + '' => 0x239f, + '' => 0x23a0, + +); + +my %ascii_8bit_translations = ( + +# Brackets - pure 8-bit ascii ugliness for browsers/OS which can't handle unicode + + '' => 0x5b, + '' => 0x5b, # '[' + '' => 0x5b, + '' => 0x5d, # ']' + '' => 0x5d, + '' => 0x5d, + +# Parens - pure 8-bit ascii ugliness for browsers/OS which can't handle unicode + + '' => 0x28, + '' => 0x28, # '(' + '' => 0x28, + '' => 0x29, + '' => 0x29, # '(' + '' => 0x29, + +); + +## +# Utility to convert elements of a string to unicode: +# +# @param input - Input string +# @param pattern - Pattern to convert +# @param unicode - Unicode to substitute for pattern. +# +# @return string - resulting string. +# +sub unicode_subst { + my ($input, $pattern, $unicode) = @_; + + my $char = pack('U', $unicode); + + $input =~ s/$pattern/$char/g; + + return $input; +} + # ====================================================================== Header sub init_tth { @@ -106,15 +178,37 @@ sub convert_real { $xmlstring=~s/^\s*\<br clear\=\"all\"/\<br/s; $xmlstring=~s/^\s*//; $xmlstring=~s/\s*$//; + &Apache::lonxml::end_alarm(); + # - # \rightleftharpoons is not converted by tth but maps - # reasonably well to ⇔. If we get many more of these, - # we're going to need to have a translation sub. - # - my $lrharpoon = pack("U", 0x21cc); - $xmlstring=~s/\\rightleftharpoons/$lrharpoon/g; + # Several strings produced by tth require + # transliteration -> unicode equivalents to render reliably + # in browsers. %unicode_translations and %unicode_harpoons are tables of + # string->substitution which we now apply. (%ascii_8bit_translations used + # instead for Windows XP and mobile devices. + + my $use_ascii; + if ($env{'browser.os'} eq 'win') { + if (($env{'browser.osversion'}) && ($env{'browser.osversion'} < 6.0)) { + $use_ascii = 1; + } + } + if ($env{'browser.mobile'}) { + $use_ascii = 1; + } + + foreach my $pattern (keys(%unicode_translations)) { + my $unicode = $unicode_translations{$pattern}; + if ($use_ascii) { + $unicode = $ascii_8bit_translations{$pattern}; + } + $xmlstring = &unicode_subst($xmlstring, $pattern, $unicode); + } + + foreach my $pattern (keys(%unicode_harpoons)) { + $xmlstring = &unicode_subst($xmlstring, $pattern, $unicode_harpoons{$pattern}); + } - &Apache::lonxml::end_alarm(); return ($xmlstring,$errorstring); } @@ -174,72 +268,79 @@ sub displaystyle { return 0; } -sub jsMath_converted { +sub MathJax_converted { my $texstring=shift; - my $tag='span'; - if (&displaystyle($texstring)) { $tag='div'; } + my ($tag,$startspan,$endspan); + $tag='math/tex;'; + if (&displaystyle($texstring)) { + $tag='math/tex; mode=display'; + $startspan=''; + $endspan=''; + } else { + $startspan='<span style="display:inline-block;">'; + $endspan='</span>'; + } &clean_out_math_mode($texstring); - return &jsMath_header(). - '<'.$tag.' class="math">'.$$texstring.'</'.$tag.'>'; + return &MathJax_header().$startspan. + '<script type="'.$tag.'">'.$$texstring.'</script>'.$endspan; } { - my @jsMath_sent_header; - sub jsMath_reset { - undef(@jsMath_sent_header); - } - sub jsMath_push { - push(@jsMath_sent_header,0); - } - sub jsMath_header { - if (!@jsMath_sent_header) { - &Apache::lonnet::logthis("mismatched calls of jsMath_header and jsMath_process"); - return ''; - } - return '' if $jsMath_sent_header[-1]; - $jsMath_sent_header[-1]=1; - return - '<script type="text/javascript"> - function NoFontMessage () {} - jsMath = {Parser: {prototype: {environments: {\'eqnarray*\' :[\'Array\',null,null,\'rcl\',[5/18,5/18],3,\'D\']}}}}; - </script>'."\n". - '<script type="text/javascript" src="/adm/jsMath/jsMath.js"></script>'."\n"; - } - sub jsMath_process { - my $state = pop(@jsMath_sent_header); - return '' if !$state; - return "\n". - '<script type="text/javascript">jsMath.Process()</script>'."\n"; - } - sub jsMath_state { - my ($level) = @_; - return $jsMath_sent_header[$level]; + #Relies heavily on the previous jsMath installation + my @MathJax_sent_header; + sub MathJax_reset { + undef(@MathJax_sent_header); + } + sub MathJax_push { + push(@MathJax_sent_header,0); + } + sub MathJax_header { + if (!@MathJax_sent_header) { + &Apache::lonnet::logthis("mismatched calls of MathJax_header and MathJax_process"); + return ''; + } + return '' if $MathJax_sent_header[-1]; + $MathJax_sent_header[-1]=1; + return + '<script type="text/javascript" src="/adm/MathJax/MathJax.js?config=TeX-AMS-MML_HTMLorMML"></script>'."\n"; } } sub tex_engine { if (exists($env{'form.texengine'})) { if ($env{'form.texengine'} ne '') { + if (lc($env{'form.texengine'}) eq 'jsmath') { + return 'MathJax'; + } return $env{'form.texengine'}; } } if ($env{'request.course.id'} && exists($env{'course.'.$env{'request.course.id'}.'.texengine'})) { + if (lc($env{'course.'.$env{'request.course.id'}.'.texengine'}) eq 'jsmath') { + return 'MathJax'; + } return $env{'course.'.$env{'request.course.id'}.'.texengine'}; } if (exists($env{'environment.texengine'})) { + if (lc($env{'environment.texengine'}) eq 'jsmath') { + return 'MathJax'; + } return $env{'environment.texengine'}; } + my $dom = $env{'request.role.domain'} || $env{'user.domain'}; + my %domdefaults = &Apache::lonnet::get_domain_defaults($dom); + if ($domdefaults{'texengine'} ne '') { + return $domdefaults{'texengine'}; + } return 'tth'; } sub init_math_support { - my ($inherit_jsmath) = @_; &init_tth(); - &Apache::lontexconvert::jsMath_push(); - if (lc(&tex_engine()) eq 'jsmath' || - ($inherit_jsmath && &jsMath_state(-2))) { - return &Apache::lontexconvert::jsMath_header(); + &Apache::lontexconvert::MathJax_push(); + if (lc(&tex_engine()) eq 'mathjax') { + return &Apache::lontexconvert::MathJax_header(); } return; } @@ -247,12 +348,10 @@ sub init_math_support { sub mimetex_valign { my ($esc_texstring)=@_; my $valign = 0; - my $lonhost = $Apache::lonnet::perlvar{'lonHostID'}; - my $hostname = &Apache::lonnet::hostname($lonhost); my $path = '/cgi-bin/mimetex.cgi?'.$esc_texstring; my $socket; &Apache::lonxml::start_alarm(); - $socket = IO::Socket::INET->new(PeerAddr => $hostname, + $socket = IO::Socket::INET->new(PeerAddr => 'localhost', PeerPort => 'http(80)', Proto => 'tcp'); if ($socket) { @@ -302,7 +401,9 @@ sub converted { if ($mode =~ /tth/i) { return &tth_converted($string); } elsif ($mode =~ /jsmath/i) { - return &jsMath_converted($string); + return &MathJax_converted($string); + } elsif ($mode =~ /mathjax/i) { + return &MathJax_converted($string); } elsif ($mode =~ /mimetex/i) { return &mimetex_converted($string); } elsif ($mode =~ /raw/i) { @@ -382,7 +483,7 @@ sub msgtexconverted { foreach my $fragment (split(/(?:\<\;|\<)\/*algebra\s*(?:\>\;|\>)/i, $message)) { if ($tex) { - my $algebra = &algebra($fragment, 'web', undef, undef, undef, undef, 'tth'); + my $algebra = &algebra($fragment, 'web', undef, undef, undef, 'tth'); if ($email) { $outmessage.='</pre><tt>'.$algebra.'</tt><pre>'; $tex=0; @@ -449,7 +550,7 @@ sub postprocess_algebra { # $string =~s/\\fun/ /g; # sqrt(3,4) means the 4 root of 3 - $string =~s/\\sqrt{([^,]+),([^\}]+)}/\\sqrt[$2]{$1}/gs; + $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; @@ -467,6 +568,8 @@ sub postprocess_algebra { $string =~s/\\lim\\left\((.+?),(.+?),(.+?)\\right\)/\\lim_{$2\\to $3}$1/gs; return $string; } + + 1; __END__ @@ -509,12 +612,10 @@ Header =item displaystyle() -=item jsMath_converted() - +=item MathJax_converted() =item tex_engine() - =item init_math_support() =item mimetex_valign()