#include <config.h>
#include "formula.h"
-#include "math_cursor.h"
-#include "math_parser.h"
+#include "formulamacro.h"
+#include "math_data.h"
#include "math_hullinset.h"
#include "math_mathmlstream.h"
-#include "textpainter.h"
+#include "math_parser.h"
+#include "BufferView.h"
+#include "cursor.h"
+#include "dispatchresult.h"
#include "debug.h"
-#include "latexrunparams.h"
+#include "funcrequest.h"
+#include "gettext.h"
+#include "LaTeXFeatures.h"
#include "LColor.h"
+#include "lyxrc.h"
+#include "lyxtext.h"
+#include "textpainter.h"
-#include "frontends/Painter.h"
-
-#include "graphics/PreviewedInset.h"
-#include "graphics/PreviewImage.h"
+#include "frontends/Alert.h"
#include "support/std_sstream.h"
+using lyx::support::trim;
-using std::ostream;
-using std::ostringstream;
-using std::vector;
-using std::auto_ptr;
using std::endl;
+using std::max;
+using std::string;
+using std::auto_ptr;
+using std::istringstream;
+using std::ostringstream;
+using std::pair;
-class InsetFormula::PreviewImpl : public lyx::graphics::PreviewedInset {
-public:
- ///
- PreviewImpl(InsetFormula & p) : PreviewedInset(p) {}
-
-private:
- ///
- bool previewWanted() const;
- ///
- string const latexString(Buffer const &) const;
- ///
- InsetFormula const & parent() const
- {
- return dynamic_cast<InsetFormula const &>(inset());
- }
-};
-
-
-
-InsetFormula::InsetFormula(bool chemistry)
- : par_(MathAtom(new MathHullInset)),
- preview_(new PreviewImpl(*this))
-{
- if (chemistry)
- mutate("chemistry");
-}
-
-
-InsetFormula::InsetFormula(InsetFormula const & other)
- : InsetFormulaBase(other),
- par_(other.par_),
- preview_(new PreviewImpl(*this))
-{}
+namespace {
-InsetFormula::InsetFormula(BufferView *)
- : par_(MathAtom(new MathHullInset)),
- preview_(new PreviewImpl(*this))
+bool openNewInset(LCursor & cur, InsetBase * inset)
{
- //view_ = bv->owner()->view();
+ if (!cur.bv().insertInset(inset)) {
+ delete inset;
+ return false;
+ }
+ inset->edit(cur, true);
+ return true;
}
-InsetFormula::InsetFormula(string const & data)
- : par_(MathAtom(new MathHullInset)),
- preview_(new PreviewImpl(*this))
-{
- if (!data.size())
- return;
- if (!mathed_parse_normal(par_, data))
- lyxerr << "cannot interpret '" << data << "' as math" << endl;
-}
+} // namespace anon
-InsetFormula::~InsetFormula()
-{}
-
-auto_ptr<InsetBase> InsetFormula::clone() const
+std::auto_ptr<InsetBase> InsetFormula::clone() const
{
return auto_ptr<InsetBase>(new InsetFormula(*this));
}
-void InsetFormula::write(Buffer const &, ostream & os) const
+void InsetFormula::write(Buffer const &, std::ostream & os) const
{
WriteStream wi(os, false, false);
- os << par_->fileInsetLabel() << ' ';
- par_->write(wi);
+ os << fileInsetLabel() << ' ';
+ MathHullInset::write(wi);
}
-int InsetFormula::latex(Buffer const &, ostream & os,
- LatexRunParams const & runparams) const
-{
- WriteStream wi(os, runparams.moving_arg, true);
- par_->write(wi);
- return wi.line();
-}
-
-
-int InsetFormula::ascii(Buffer const &, ostream & os, int) const
+void InsetFormula::read(Buffer const &, LyXLex & lex)
{
- if (0 && display()) {
- Dimension dim;
- TextMetricsInfo mi;
- par()->metricsT(mi, dim);
- TextPainter tpain(dim.width(), dim.height());
- par()->drawT(tpain, 0, dim.ascent());
- tpain.show(os, 3);
- // reset metrics cache to "real" values
- //metrics();
- return tpain.textheight();
- } else {
- WriteStream wi(os, false, true);
- wi << ' ' << (par_->asNestInset()->cell(0)) << ' ';
- return wi.line();
- }
+ MathAtom at;
+ mathed_parse_normal(at, lex);
+ MathHullInset::operator=(*at->asHullInset());
}
-int InsetFormula::linuxdoc(Buffer const & buf, ostream & os) const
-{
- return docbook(buf, os, false);
-}
+/////////////////////////////////////////////
-int InsetFormula::docbook(Buffer const & buf, ostream & os, bool) const
+void mathDispatchCreation(LCursor & cur, FuncRequest const & cmd,
+ bool display)
{
- MathMLStream ms(os);
- ms << MTag("equation");
- ms << MTag("alt");
- ms << "<[CDATA[";
- int res = ascii(buf, ms.os(), 0);
- ms << "]]>";
- ms << ETag("alt");
- ms << MTag("math");
- ms << par_;
- ms << ETag("math");
- ms << ETag("equation");
- return ms.line() + res;
-}
+ // use selection if available..
+ //string sel;
+ //if (action == LFUN_MATH_IMPORT_SELECTION)
+ // sel = "";
+ //else
+ string sel =
+ cur.bv().getLyXText()->selectionAsString(*cur.bv().buffer(), false);
-void InsetFormula::read(Buffer const &, LyXLex & lex)
-{
- mathed_parse_normal(par_, lex);
- // remove extra 'mathrm' for chemistry stuff.
- // will be re-added on write
- if (par_->asHullInset()->getType() =="chemistry") {
- lyxerr << "this is chemistry" << endl;
- if (par_->cell(0).size() == 1) {
- lyxerr << "this is size 1" << endl;
- if (par_->cell(0)[0]->asFontInset()) {
- lyxerr << "this is a font inset "
- << "replacing " << par_.nucleus()->cell(0) <<
- " with " << par_->cell(0)[0]->cell(0) << endl;
- }
+ if (sel.empty()) {
+ InsetBase * f = new MathHullInset;
+ if (openNewInset(cur, f)) {
+ cur.inset()->dispatch(cur, FuncRequest(LFUN_MATH_MUTATE, "simple"));
+ // don't do that also for LFUN_MATH_MODE unless you want end up with
+ // always changing to mathrm when opening an inlined inset
+ // -- I really hate "LyXfunc overloading"...
+ if (display)
+ f->dispatch(cur, FuncRequest(LFUN_MATH_DISPLAY));
+ f->dispatch(cur, FuncRequest(LFUN_INSERT_MATH, cmd.argument));
}
- }
- //metrics();
-}
-
-
-//ostream & operator<<(ostream & os, LyXCursor const & c)
-//{
-// os << '[' << c.x() << ' ' << c.y() << ' ' << c.pos() << ']';
-// return os;
-//}
-
-
-void InsetFormula::draw(PainterInfo & pi, int x, int y) const
-{
- cache(pi.base.bv);
- // This initiates the loading of the preview, so should come
- // before the metrics are computed.
- bool const use_preview = preview_->previewReady();
-
- int const w = dim_.wid;
- int const d = dim_.des;
- int const a = dim_.asc;
- int const h = a + d;
-
- if (use_preview) {
- pi.pain.image(x + 1, y - a, w, h, // one pixel gap in front
- *(preview_->pimage()->image()));
} else {
- PainterInfo p(pi.base.bv);
- p.base.style = LM_ST_TEXT;
- p.base.font = pi.base.font;
- p.base.font.setColor(LColor::math);
- if (lcolor.getX11Name(LColor::mathbg)
- != lcolor.getX11Name(LColor::background))
- p.pain.fillRectangle(x, y - a, w, h, LColor::mathbg);
-
- if (mathcursor &&
- const_cast<InsetFormulaBase const *>(mathcursor->formula()) == this)
- {
- mathcursor->drawSelection(pi);
- //p.pain.rectangle(x, y - a, w, h, LColor::mathframe);
- }
-
- par_->draw(p, x + offset_, y);
+ // create a macro if we see "\\newcommand" somewhere, and an ordinary
+ // formula otherwise
+ InsetBase * f;
+ if (sel.find("\\newcommand") == string::npos &&
+ sel.find("\\def") == string::npos)
+ f = new MathHullInset(sel);
+ else
+ f = new InsetFormulaMacro(sel);
+ cur.bv().getLyXText()->cutSelection(true, false);
+ openNewInset(cur, f);
}
-
- xo_ = x;
- yo_ = y;
+ cmd.message(N_("Math editor mode"));
}
-void InsetFormula::getLabelList(Buffer const & buffer,
- vector<string> & res) const
+void mathDispatch(LCursor & cur, FuncRequest const & cmd)
{
- par()->getLabelList(buffer, res);
-}
-
-
-InsetOld::Code InsetFormula::lyxCode() const
-{
- return InsetOld::MATH_CODE;
-}
-
-
-void InsetFormula::validate(LaTeXFeatures & features) const
-{
- par_->validate(features);
-}
-
-
-bool InsetFormula::insetAllowed(InsetOld::Code code) const
-{
- return
- code == InsetOld::LABEL_CODE
- || code == InsetOld::REF_CODE
- || code == InsetOld::ERT_CODE;
-}
-
+ if (!cur.bv().available())
+ return;
-void InsetFormula::metrics(MetricsInfo & m, Dimension & dim) const
-{
- view_ = m.base.bv;
- if (preview_->previewReady()) {
- dim.asc = preview_->pimage()->ascent();
- dim.des = preview_->pimage()->descent();
- // insert a one pixel gap in front of the formula
- dim.wid = 1 + preview_->pimage()->width();
- if (display())
- dim.des += 12;
- } else {
- MetricsInfo mi = m;
- mi.base.style = LM_ST_TEXT;
- mi.base.font.setColor(LColor::math);
- par()->metrics(mi, dim);
- dim.asc += 1;
- dim.des += 1;
- }
+ switch (cmd.action) {
+
+ case LFUN_MATH_DISPLAY:
+ mathDispatchCreation(cur, cmd, true);
+ break;
+
+ case LFUN_MATH_MODE:
+ mathDispatchCreation(cur, cmd, false);
+ break;
+
+ case LFUN_MATH_IMPORT_SELECTION:
+ mathDispatchCreation(cur, cmd, false);
+ break;
+
+/*
+ case LFUN_MATH_MACRO:
+ if (cmd.argument.empty())
+ cmd.errorMessage(N_("Missing argument"));
+ else {
+ string s = cmd.argument;
+ string const s1 = token(s, ' ', 1);
+ int const nargs = s1.empty() ? 0 : atoi(s1);
+ string const s2 = token(s, ' ', 2);
+ string const type = s2.empty() ? "newcommand" : s2;
+ openNewInset(cur, new InsetFormulaMacro(token(s, ' ', 0), nargs, s2));
+ }
+ break;
+
+ case LFUN_INSERT_MATH:
+ case LFUN_INSERT_MATRIX:
+ case LFUN_MATH_DELIM: {
+ MathHullInset * f = new MathHullInset;
+ if (openNewInset(cur, f)) {
+ cur.inset()->dispatch(cur, FuncRequest(LFUN_MATH_MUTATE, "simple"));
+ cur.inset()->dispatch(cur, cmd);
+ }
+ break;
+ }
+*/
- if (display()) {
- offset_ = (m.base.textwidth - dim.wid) / 2;
- dim.wid = m.base.textwidth;
- } else {
- offset_ = 0;
+ default:
+ break;
}
-
- dim_ = dim;
-}
-
-
-void InsetFormula::mutate(string const & type)
-{
- par_.nucleus()->mutate(type);
-}
-
-
-//
-// preview stuff
-//
-
-void InsetFormula::addPreview(lyx::graphics::PreviewLoader & ploader) const
-{
- preview_->addPreview(ploader);
-}
-
-
-void InsetFormula::generatePreview() const
-{
- preview_->generatePreview();
-}
-
-
-bool InsetFormula::PreviewImpl::previewWanted() const
-{
- return !parent().par_->asNestInset()->editing();
-}
-
-
-string const InsetFormula::PreviewImpl::latexString(Buffer const &) const
-{
- ostringstream ls;
- WriteStream wi(ls, false, false);
- parent().par_->write(wi);
- return ls.str();
}