mirror of
https://git.lyx.org/repos/lyx.git
synced 2024-11-08 10:51:03 +00:00
226 lines
4.6 KiB
C++
226 lines
4.6 KiB
C++
|
/** The .tex to .lyx converter
|
|||
|
\author Andr<EFBFBD> P<EFBFBD>nitz (2003)
|
|||
|
*/
|
|||
|
|
|||
|
// {[(
|
|||
|
|
|||
|
#include <config.h>
|
|||
|
|
|||
|
#include "Lsstream.h"
|
|||
|
#include "tex2lyx.h"
|
|||
|
|
|||
|
#include <iostream>
|
|||
|
#include <string>
|
|||
|
#include <vector>
|
|||
|
|
|||
|
using std::cerr;
|
|||
|
using std::endl;
|
|||
|
using std::ostream;
|
|||
|
using std::string;
|
|||
|
using std::vector;
|
|||
|
|
|||
|
|
|||
|
bool is_math_env(string const & name)
|
|||
|
{
|
|||
|
static char const * known_math_envs[] = { "equation", "equation*",
|
|||
|
"eqnarray", "eqnarray*", "align", "align*", 0};
|
|||
|
|
|||
|
for (char const ** what = known_math_envs; *what; ++what)
|
|||
|
if (*what == name)
|
|||
|
return true;
|
|||
|
return false;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void parse_math(Parser & p, ostream & os, unsigned flags, const mode_type mode)
|
|||
|
{
|
|||
|
while (p.good()) {
|
|||
|
Token const & t = p.getToken();
|
|||
|
|
|||
|
#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) {
|
|||
|
// we are inside some text mode thingy, so opening new math is allowed
|
|||
|
Token const & n = p.getToken();
|
|||
|
if (n.cat() == catMath) {
|
|||
|
// TeX's $$...$$ syntax for displayed math
|
|||
|
os << "\\[";
|
|||
|
parse_math(p, os, FLAG_SIMPLE, MATH_MODE);
|
|||
|
os << "\\]";
|
|||
|
p.getToken(); // skip the second '$' token
|
|||
|
} 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() == catSpace ||
|
|||
|
t.cat() == catSuper ||
|
|||
|
t.cat() == catSub ||
|
|||
|
t.cat() == catOther ||
|
|||
|
t.cat() == catAlign ||
|
|||
|
t.cat() == catActive ||
|
|||
|
t.cat() == catParameter)
|
|||
|
os << t.character();
|
|||
|
|
|||
|
else if (t.cat() == catNewline) {
|
|||
|
//if (p.nextToken().cat() == catNewline) {
|
|||
|
// p.getToken();
|
|||
|
// handle_par(os);
|
|||
|
//} else {
|
|||
|
os << "\n "; // note the space
|
|||
|
//}
|
|||
|
}
|
|||
|
|
|||
|
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";
|
|||
|
}
|
|||
|
|
|||
|
else if (t.cat() == catComment)
|
|||
|
handle_comment(p);
|
|||
|
|
|||
|
//
|
|||
|
// control sequences
|
|||
|
//
|
|||
|
|
|||
|
else if (t.cs() == "(") {
|
|||
|
os << "\\(";
|
|||
|
parse_math(p, os, FLAG_SIMPLE2, MATH_MODE);
|
|||
|
os << "\\)";
|
|||
|
}
|
|||
|
|
|||
|
else if (t.cs() == "[") {
|
|||
|
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('{', '}');
|
|||
|
active_environments_push(name);
|
|||
|
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 << "}";
|
|||
|
}
|
|||
|
|
|||
|
else if (t.cs() == "end") {
|
|||
|
if (flags & FLAG_END) {
|
|||
|
// eat environment name
|
|||
|
string const name = p.getArg('{', '}');
|
|||
|
if (name != curr_env())
|
|||
|
p.error("\\end{" + name + "} does not match \\begin{"
|
|||
|
+ curr_env() + "}");
|
|||
|
active_environments_pop();
|
|||
|
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"
|
|||
|
|| t.cs() == "texttt" || t.cs() == "textsc") {
|
|||
|
os << '\\' << t.cs() << '{';
|
|||
|
parse_math(p, os, FLAG_ITEM, MATHTEXT_MODE);
|
|||
|
os << '}';
|
|||
|
}
|
|||
|
|
|||
|
else if (t.cs() == "mbox") {
|
|||
|
os << "\\mbox{";
|
|||
|
parse_math(p, os, FLAG_ITEM, MATHTEXT_MODE);
|
|||
|
os << '}';
|
|||
|
}
|
|||
|
|
|||
|
else if (t.cs() == "\"") {
|
|||
|
string const name = p.verbatimItem();
|
|||
|
if (name == "a") os << '<EFBFBD>';
|
|||
|
else if (name == "o") os << '<EFBFBD>';
|
|||
|
else if (name == "u") os << '<EFBFBD>';
|
|||
|
else if (name == "A") os << '<EFBFBD>';
|
|||
|
else if (name == "O") os << '<EFBFBD>';
|
|||
|
else if (name == "U") os << '<EFBFBD>';
|
|||
|
else os << "\"{" << name << "}";
|
|||
|
}
|
|||
|
|
|||
|
else if (t.cs() == "ss")
|
|||
|
os << "<EFBFBD>";
|
|||
|
|
|||
|
else
|
|||
|
os << t.asInput();
|
|||
|
|
|||
|
if (flags & FLAG_LEAVE) {
|
|||
|
flags &= ~FLAG_LEAVE;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
// }])
|