2003-08-19 10:04:35 +00:00
|
|
|
/**
|
2007-04-26 04:53:06 +00:00
|
|
|
* \file math.cpp
|
2003-08-19 10:04:35 +00:00
|
|
|
* This file is part of LyX, the document processor.
|
|
|
|
* Licence details can be found in the file COPYING.
|
|
|
|
*
|
2008-11-14 15:58:50 +00:00
|
|
|
* \author André Pönitz
|
2003-08-19 10:04:35 +00:00
|
|
|
*
|
2003-08-23 00:17:00 +00:00
|
|
|
* Full author contact details are available in file CREDITS.
|
2003-04-17 09:47:21 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
// {[(
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
2011-12-08 20:05:51 +00:00
|
|
|
#include "Preamble.h"
|
2003-04-17 09:47:21 +00:00
|
|
|
#include "tex2lyx.h"
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
|
2007-12-12 10:16:00 +00:00
|
|
|
using namespace std;
|
2006-10-21 00:16:43 +00:00
|
|
|
|
|
|
|
namespace lyx {
|
|
|
|
|
2003-04-17 09:47:21 +00:00
|
|
|
|
|
|
|
bool is_math_env(string const & name)
|
|
|
|
{
|
2005-07-14 15:19:01 +00:00
|
|
|
return known_math_environments.find(name) != known_math_environments.end();
|
2003-04-17 09:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-13 11:29:48 +00:00
|
|
|
bool is_display_math_env(string const & name)
|
|
|
|
{
|
|
|
|
CommandMap::const_iterator it = known_math_environments.find(name);
|
|
|
|
if (it != known_math_environments.end())
|
|
|
|
if (!it->second.empty())
|
|
|
|
return it->second.back() == displaymath;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-04-17 09:47:21 +00:00
|
|
|
void parse_math(Parser & p, ostream & os, unsigned flags, const mode_type mode)
|
|
|
|
{
|
|
|
|
while (p.good()) {
|
2003-04-23 15:14:43 +00:00
|
|
|
Token const & t = p.get_token();
|
2003-04-17 09:47:21 +00:00
|
|
|
|
|
|
|
#ifdef FILEDEBUG
|
|
|
|
cerr << "t: " << t << " flags: " << flags << "\n";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (flags & FLAG_ITEM) {
|
|
|
|
if (t.cat() == catSpace)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
flags &= ~FLAG_ITEM;
|
|
|
|
if (t.cat() == catBegin) {
|
|
|
|
// skip the brace and collect everything to the next matching
|
|
|
|
// closing brace
|
|
|
|
flags |= FLAG_BRACE_LAST;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle only this single token, leave the loop if done
|
|
|
|
flags |= FLAG_LEAVE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// cat codes
|
|
|
|
//
|
|
|
|
if (t.cat() == catMath) {
|
|
|
|
if (mode == MATHTEXT_MODE) {
|
2009-03-30 01:43:40 +00:00
|
|
|
// we are inside some text mode thing, so opening new math is allowed
|
2003-04-23 15:14:43 +00:00
|
|
|
Token const & n = p.get_token();
|
2003-04-17 09:47:21 +00:00
|
|
|
if (n.cat() == catMath) {
|
|
|
|
// TeX's $$...$$ syntax for displayed math
|
|
|
|
os << "\\[";
|
|
|
|
parse_math(p, os, FLAG_SIMPLE, MATH_MODE);
|
|
|
|
os << "\\]";
|
2003-04-23 15:14:43 +00:00
|
|
|
p.get_token(); // skip the second '$' token
|
2003-04-17 09:47:21 +00:00
|
|
|
} else {
|
|
|
|
// simple $...$ stuff
|
|
|
|
p.putback();
|
|
|
|
os << '$';
|
|
|
|
parse_math(p, os, FLAG_SIMPLE, MATH_MODE);
|
|
|
|
os << '$';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (flags & FLAG_SIMPLE) {
|
|
|
|
// this is the end of the formula
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
else {
|
|
|
|
cerr << "\nmode: " << mode << endl;
|
|
|
|
p.error("something strange in the parser\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t.cat() == catLetter ||
|
|
|
|
t.cat() == catSuper ||
|
|
|
|
t.cat() == catSub ||
|
|
|
|
t.cat() == catOther ||
|
|
|
|
t.cat() == catAlign ||
|
|
|
|
t.cat() == catActive ||
|
|
|
|
t.cat() == catParameter)
|
2008-11-16 17:02:00 +00:00
|
|
|
os << t.cs();
|
2003-04-17 09:47:21 +00:00
|
|
|
|
|
|
|
else if (t.cat() == catBegin) {
|
|
|
|
os << '{';
|
|
|
|
parse_math(p, os, FLAG_BRACE_LAST, mode);
|
|
|
|
os << '}';
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t.cat() == catEnd) {
|
|
|
|
if (flags & FLAG_BRACE_LAST)
|
|
|
|
return;
|
|
|
|
os << "unexpected '}' in math\n";
|
|
|
|
}
|
|
|
|
|
2003-11-05 10:14:13 +00:00
|
|
|
else if (t.cat() == catComment) {
|
2003-11-19 10:35:50 +00:00
|
|
|
if (!t.cs().empty())
|
2012-05-06 16:46:06 +02:00
|
|
|
os << t.asInput();
|
2003-11-05 10:14:13 +00:00
|
|
|
else
|
|
|
|
// "%\n" combination
|
|
|
|
p.skip_spaces();
|
|
|
|
}
|
2003-04-17 09:47:21 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// control sequences
|
|
|
|
//
|
|
|
|
|
|
|
|
else if (t.cs() == "(") {
|
|
|
|
os << "\\(";
|
|
|
|
parse_math(p, os, FLAG_SIMPLE2, MATH_MODE);
|
|
|
|
os << "\\)";
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t.cs() == "[") {
|
2003-04-25 15:54:29 +00:00
|
|
|
// special handling of a few common SW user quirks
|
|
|
|
p.skip_spaces();
|
2003-09-09 18:27:24 +00:00
|
|
|
//if (p.next_token().cs() ==
|
2003-04-17 09:47:21 +00:00
|
|
|
os << "\\[";
|
|
|
|
parse_math(p, os, FLAG_EQUATION, MATH_MODE);
|
|
|
|
os << "\\]";
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t.cs() == "protect")
|
|
|
|
// ignore \\protect, will hopefully be re-added during output
|
|
|
|
;
|
|
|
|
|
|
|
|
else if (t.cs() == "begin") {
|
|
|
|
string const name = p.getArg('{', '}');
|
2003-04-23 15:14:43 +00:00
|
|
|
active_environments.push_back(name);
|
2003-04-17 09:47:21 +00:00
|
|
|
os << "\\begin{" << name << "}";
|
|
|
|
if (name == "tabular")
|
|
|
|
parse_math(p, os, FLAG_END, MATHTEXT_MODE);
|
|
|
|
else
|
|
|
|
parse_math(p, os, FLAG_END, mode);
|
|
|
|
os << "\\end{" << name << "}";
|
2003-10-23 11:46:33 +00:00
|
|
|
active_environments.pop_back();
|
2003-04-17 09:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
else if (t.cs() == "end") {
|
|
|
|
if (flags & FLAG_END) {
|
|
|
|
// eat environment name
|
|
|
|
string const name = p.getArg('{', '}');
|
2003-04-23 15:14:43 +00:00
|
|
|
if (name != active_environment())
|
2003-04-17 09:47:21 +00:00
|
|
|
p.error("\\end{" + name + "} does not match \\begin{"
|
2003-04-23 15:14:43 +00:00
|
|
|
+ active_environment() + "}");
|
2003-04-17 09:47:21 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
p.error("found 'end' unexpectedly");
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t.cs() == ")") {
|
|
|
|
if (flags & FLAG_SIMPLE2)
|
|
|
|
return;
|
|
|
|
p.error("found '\\)' unexpectedly");
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t.cs() == "]") {
|
|
|
|
if (flags & FLAG_EQUATION)
|
|
|
|
return;
|
|
|
|
p.error("found '\\]' unexpectedly");
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t.cs() == "textrm" || t.cs() == "textsf" || t.cs() == "textbf"
|
2003-10-23 11:46:33 +00:00
|
|
|
|| t.cs() == "texttt" || t.cs() == "textsc"
|
|
|
|
|| t.cs() == "text" || t.cs() == "intertext") {
|
2003-04-17 09:47:21 +00:00
|
|
|
os << '\\' << t.cs() << '{';
|
|
|
|
parse_math(p, os, FLAG_ITEM, MATHTEXT_MODE);
|
|
|
|
os << '}';
|
|
|
|
}
|
|
|
|
|
2006-03-16 15:05:47 +00:00
|
|
|
else if (t.cs() == "tag") {
|
|
|
|
os << '\\' << t.cs();
|
|
|
|
if (p.next_token().asInput() == "*") {
|
|
|
|
p.get_token();
|
|
|
|
os << '*';
|
|
|
|
}
|
|
|
|
os << '{';
|
|
|
|
parse_math(p, os, FLAG_ITEM, MATHTEXT_MODE);
|
|
|
|
os << '}';
|
|
|
|
}
|
|
|
|
|
2003-10-23 11:46:33 +00:00
|
|
|
else if (t.cs() == "mbox" || t.cs() == "fbox") {
|
|
|
|
os << "\\" << t.cs() << '{';
|
2003-04-17 09:47:21 +00:00
|
|
|
parse_math(p, os, FLAG_ITEM, MATHTEXT_MODE);
|
|
|
|
os << '}';
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t.cs() == "\"") {
|
2003-04-23 15:14:43 +00:00
|
|
|
string const name = p.verbatim_item();
|
2007-05-10 10:35:57 +00:00
|
|
|
if (name == "a") os << '\xe4';
|
|
|
|
else if (name == "o") os << '\xf6';
|
|
|
|
else if (name == "u") os << '\xfc';
|
|
|
|
else if (name == "A") os << '\xc4';
|
|
|
|
else if (name == "O") os << '\xd6';
|
|
|
|
else if (name == "U") os << '\xdc';
|
2003-04-17 09:47:21 +00:00
|
|
|
else os << "\"{" << name << "}";
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t.cs() == "ss")
|
2007-05-10 10:35:57 +00:00
|
|
|
os << "\xdf";
|
2003-04-17 09:47:21 +00:00
|
|
|
|
2004-07-23 18:47:57 +00:00
|
|
|
else if (t.cs() == "cr") {
|
|
|
|
// lyx can't handle \\cr
|
2022-12-28 16:29:38 +01:00
|
|
|
warning_message("Converting TeX '\\cr' to LaTeX '\\\\'.");
|
2004-07-23 18:47:57 +00:00
|
|
|
os << "\\\\";
|
|
|
|
}
|
|
|
|
|
2011-12-08 20:05:51 +00:00
|
|
|
else if (t.cs() == "vref" || t.cs() == "vpageref") {
|
|
|
|
os << t.asInput();
|
|
|
|
preamble.registerAutomaticallyLoadedPackage("varioref");
|
|
|
|
}
|
|
|
|
|
2013-02-19 00:07:24 +01:00
|
|
|
else if (t.cs() == "textipa") {
|
|
|
|
os << t.asInput();
|
|
|
|
preamble.registerAutomaticallyLoadedPackage("tipa");
|
|
|
|
}
|
|
|
|
|
2003-09-09 18:27:24 +00:00
|
|
|
else
|
2003-04-17 09:47:21 +00:00
|
|
|
os << t.asInput();
|
|
|
|
|
|
|
|
if (flags & FLAG_LEAVE) {
|
|
|
|
flags &= ~FLAG_LEAVE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// }])
|
2006-10-21 00:16:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
} // namespace lyx
|