-/*
-* File: formula.C
-* Purpose: Implementation of formula inset
-* Author: Alejandro Aguilar Sierra <asierra@servidor.unam.mx>
-* Created: January 1996
-* Description: Allows the edition of math paragraphs inside Lyx.
-*
-* Copyright: 1996-1998 Alejandro Aguilar Sierra
-*
-* Version: 0.4, Lyx project.
-*
-* You are free to use and modify this code under the terms of
-* the GNU General Public Licence version 2 or later.
-*/
-
-#ifdef __GNUG__
-#pragma implementation
-#endif
+/**
+ * \file formula.C
+ * This file is part of LyX, the document processor.
+ * Licence details can be found in the file COPYING.
+ *
+ * \author Alejandro Aguilar Sierra
+ * \author André Pönitz
+ *
+ * Full author contact details are available in file CREDITS.
+ */
#include <config.h>
#include "formula.h"
-#include "math_cursor.h"
-#include "math_parser.h"
-#include "math_charinset.h"
-#include "math_arrayinset.h"
-#include "math_metricsinfo.h"
-#include "math_deliminset.h"
+#include "formulamacro.h"
+#include "math_data.h"
#include "math_hullinset.h"
-#include "math_support.h"
#include "math_mathmlstream.h"
-#include "textpainter.h"
+#include "math_parser.h"
#include "BufferView.h"
-#include "gettext.h"
+#include "cursor.h"
+#include "dispatchresult.h"
#include "debug.h"
-#include "lyxrc.h"
#include "funcrequest.h"
-
-#include "support/LOstream.h"
-#include "support/LAssert.h"
-#include "support/lyxlib.h"
-#include "support/systemcall.h"
-#include "support/filetools.h"
+#include "gettext.h"
+#include "LaTeXFeatures.h"
+#include "LColor.h"
+#include "lyxrc.h"
+#include "lyxtext.h"
+#include "textpainter.h"
#include "frontends/Alert.h"
-#include "frontends/LyXView.h"
-#include "frontends/Painter.h"
-
-#include "graphics/PreviewedInset.h"
-#include "graphics/PreviewImage.h"
-#include <fstream>
+#include "support/std_sstream.h"
+using lyx::support::trim;
-using std::ostream;
-using std::ifstream;
-using std::istream;
-using std::pair;
using std::endl;
-using std::vector;
-using std::getline;
-
-
-class InsetFormula::PreviewImpl : public grfx::PreviewedInset {
-public:
- ///
- PreviewImpl(InsetFormula & p) : PreviewedInset(p) {}
-
-private:
- ///
- bool previewWanted() const;
- ///
- string const latexString() const;
- ///
- InsetFormula & parent() const
- {
- return *static_cast<InsetFormula*>(inset());
- }
-};
-
-
-
-InsetFormula::InsetFormula()
- : par_(MathAtom(new MathHullInset)),
- preview_(new PreviewImpl(*this))
-{}
+using std::max;
+using std::string;
+using std::auto_ptr;
+using std::istringstream;
+using std::ostringstream;
+using std::pair;
-InsetFormula::InsetFormula(InsetFormula const & other)
- : InsetFormulaBase(other),
- par_(other.par_),
- preview_(new PreviewImpl(*this))
-{}
+namespace {
-InsetFormula::InsetFormula(BufferView * bv)
- : 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\n";
-}
-
+} // namespace anon
-InsetFormula::~InsetFormula()
-{}
-Inset * InsetFormula::clone(Buffer const &, bool) const
+std::auto_ptr<InsetBase> InsetFormula::clone() const
{
- return new InsetFormula(*this);
+ 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);
-}
-
-
-int InsetFormula::latex(Buffer const *, ostream & os, bool fragile, bool) const
-{
- WriteStream wi(os, fragile, true);
- par_->write(wi);
- return wi.line();
-}
-
-
-int InsetFormula::ascii(Buffer const *, ostream & os, int) const
-{
- if (display()) {
- TextMetricsInfo mi;
- par()->metricsT(mi);
- TextPainter tpain(par()->width(), par()->height());
- par()->drawT(tpain, 0, par()->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();
- }
+ os << fileInsetLabel() << ' ';
+ MathHullInset::write(wi);
}
-int InsetFormula::linuxdoc(Buffer const * buf, ostream & os) const
+void InsetFormula::read(Buffer const &, LyXLex & lex)
{
- return docbook(buf, os, false);
+ MathAtom at;
+ mathed_parse_normal(at, lex);
+ MathHullInset::operator=(*at->asHullInset());
}
-int InsetFormula::docbook(Buffer const * buf, ostream & os, bool) const
-{
- 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;
-}
+/////////////////////////////////////////////
-void InsetFormula::read(Buffer const *, LyXLex & lex)
+void mathDispatchCreation(LCursor & cur, FuncRequest const & cmd,
+ bool display)
{
- mathed_parse_normal(par_, lex);
- metrics();
-}
-
-
-//ostream & operator<<(ostream & os, LyXCursor const & c)
-//{
-// os << '[' << c.x() << ' ' << c.y() << ' ' << c.pos() << ']';
-// return os;
-//}
+ // 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::draw(BufferView * bv, LyXFont const & font,
- int y, float & xx, bool) const
-{
- // This initiates the loading of the preview, so should come
- // before the metrics are computed.
- preview_->setView(bv);
- bool const use_preview = preview_->previewReady();
-
- int const x = int(xx);
- int const w = width(bv, font);
- int const d = descent(bv, font);
- int const a = ascent(bv, font);
- int const h = a + d;
-
- MathPainterInfo pi(bv->painter());
-
- if (use_preview) {
- pi.pain.image(x, y - a, w, h,
- *(preview_->pimage()->image(*this, *bv)));
- } else {
- pi.base.style = LM_ST_TEXT;
- pi.base.font = font;
- pi.base.font.setColor(LColor::math);
- if (lcolor.getX11Name(LColor::mathbg)
- != lcolor.getX11Name(LColor::background))
- pi.pain.fillRectangle(x, y - a, w, h, LColor::mathbg);
-
- if (mathcursor &&
- const_cast<InsetFormulaBase const *>(mathcursor->formula()) == this)
- {
- mathcursor->drawSelection(pi);
- //pi.pain.rectangle(x, y - a, w, h, LColor::mathframe);
+ 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));
}
-
- par_->draw(pi, x, y);
+ } else {
+ // 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);
}
-
- xx += w;
- xo_ = x;
- yo_ = y;
-
- setCursorVisible(false);
-}
-
-
-vector<string> const InsetFormula::getLabelList() const
-{
- vector<string> res;
- par()->getLabelList(res);
- return res;
-}
-
-
-Inset::Code InsetFormula::lyxCode() const
-{
- return Inset::MATH_CODE;
-}
-
-
-void InsetFormula::validate(LaTeXFeatures & features) const
-{
- par_->validate(features);
-}
-
-
-bool InsetFormula::insetAllowed(Inset::Code code) const
-{
- return
- (code == Inset::LABEL_CODE)
- || code == Inset::REF_CODE
- || code == Inset::ERT_CODE;
+ cmd.message(N_("Math editor mode"));
}
-int InsetFormula::ascent(BufferView *, LyXFont const &) const
+void mathDispatch(LCursor & cur, FuncRequest const & cmd)
{
- return preview_->previewReady() ?
- preview_->pimage()->ascent() : 1 + par_->ascent();
-}
-
-
-int InsetFormula::descent(BufferView *, LyXFont const &) const
-{
- if (!preview_->previewReady())
- return 1 + par_->descent();
+ if (!cur.bv().available())
+ return;
- int const descent = preview_->pimage()->descent();
- return display() ? descent + 12 : descent;
-}
+ switch (cmd.action) {
+ case LFUN_MATH_DISPLAY:
+ mathDispatchCreation(cur, cmd, true);
+ break;
-int InsetFormula::width(BufferView * bv, LyXFont const & font) const
-{
- metrics(bv, font);
- return preview_->previewReady() ?
- preview_->pimage()->width() : par_->width();
-}
+ case LFUN_MATH_MODE:
+ mathDispatchCreation(cur, cmd, false);
+ break;
+ case LFUN_MATH_IMPORT_SELECTION:
+ mathDispatchCreation(cur, cmd, false);
+ break;
/*
-void InsetFormula::mutate(string const & type)
-{
- par_.nucleus()->mutate(type);
-}
+ 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;
+ }
*/
-
-//
-// preview stuff
-//
-
-void InsetFormula::addPreview(grfx::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() const
-{
- ostringstream ls;
- WriteStream wi(ls, false, false);
- parent().par_->write(wi);
- return ls.str().c_str();
+ default:
+ break;
+ }
}