/** * \file InsetMathMacro.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 #include "InsetMathMacro.h" #include "MathSupport.h" #include "MathExtern.h" #include "MathStream.h" #include "buffer.h" #include "cursor.h" #include "debug.h" #include "BufferView.h" #include "LaTeXFeatures.h" #include "frontends/Painter.h" namespace lyx { using std::string; using std::max; using std::auto_ptr; using std::endl; using std::vector; /// This class is the value of a macro argument, technically /// a wrapper of the cells of MathMacro. class MathMacroArgumentValue : public InsetMathDim { public: /// MathMacroArgumentValue(MathArray const * value) : value_(value) {} /// bool metrics(MetricsInfo & mi, Dimension & dim) const; /// void draw(PainterInfo &, int x, int y) const; private: std::auto_ptr doClone() const; MathArray const * value_; }; auto_ptr MathMacroArgumentValue::doClone() const { return auto_ptr(new MathMacroArgumentValue(*this)); } bool MathMacroArgumentValue::metrics(MetricsInfo & mi, Dimension & dim) const { value_->metrics(mi, dim); metricsMarkers2(dim); if (dim_ == dim) return false; dim_ = dim; return true; } void MathMacroArgumentValue::draw(PainterInfo & pi, int x, int y) const { value_->draw(pi, x, y); } MathMacro::MathMacro(docstring const & name) : InsetMathNest(0), name_(name) {} auto_ptr MathMacro::doClone() const { return auto_ptr(new MathMacro(*this)); } docstring MathMacro::name() const { return name_; } void MathMacro::cursorPos(BufferView const & bv, CursorSlice const & sl, bool boundary, int & x, int & y) const { // We may have 0 arguments, but InsetMathNest requires at least one. if (nargs() > 0) InsetMathNest::cursorPos(bv, sl, boundary, x, y); } bool MathMacro::metrics(MetricsInfo & mi, Dimension & dim) const { if (!MacroTable::globalMacros().has(name())) { mathed_string_dim(mi.base.font, "Unknown: " + name(), dim); } else if (editing(mi.base.bv)) { // FIXME UNICODE asArray(MacroTable::globalMacros().get(name()).def(), tmpl_); LyXFont font = mi.base.font; augmentFont(font, from_ascii("lyxtex")); tmpl_.metrics(mi, dim); // FIXME UNICODE dim.wid += mathed_string_width(font, name()) + 10; // FIXME UNICODE int ww = mathed_string_width(font, from_ascii("#1: ")); for (idx_type i = 0; i < nargs(); ++i) { MathArray const & c = cell(i); c.metrics(mi); dim.wid = max(dim.wid, c.width() + ww); dim.des += c.height() + 10; } } else { // create MathMacroArgumentValue object pointing to the cells of the macro MacroData const & macro = MacroTable::globalMacros().get(name()); vector values(nargs()); for (size_t i = 0; i != nargs(); ++i) values[i].insert(0, MathAtom(new MathMacroArgumentValue(&cells_[i]))); macro.expand(values, expanded_); expanded_.metrics(mi, dim); } metricsMarkers2(dim); if (dim_ == dim) return false; dim_ = dim; return true; } void MathMacro::draw(PainterInfo & pi, int x, int y) const { if (!MacroTable::globalMacros().has(name())) { // FIXME UNICODE drawStrRed(pi, x, y, "Unknown: " + name()); } else if (editing(pi.base.bv)) { LyXFont font = pi.base.font; augmentFont(font, from_ascii("lyxtex")); int h = y - dim_.ascent() + 2 + tmpl_.ascent(); pi.pain.text(x + 3, h, name(), font); int const w = mathed_string_width(font, name()); tmpl_.draw(pi, x + w + 12, h); h += tmpl_.descent(); Dimension ldim; docstring t = from_ascii("#1: "); mathed_string_dim(font, t, ldim); for (idx_type i = 0; i < nargs(); ++i) { MathArray const & c = cell(i); h += max(c.ascent(), ldim.asc) + 5; c.draw(pi, x + ldim.wid, h); char_type str[] = { '#', '1', ':', '\0' }; str[1] += static_cast(i); pi.pain.text(x + 3, h, str, font); h += max(c.descent(), ldim.des) + 5; } } else { expanded_.draw(pi, x, y); } drawMarkers2(pi, x, y); } void MathMacro::drawSelection(PainterInfo & pi, int x, int y) const { // We may have 0 arguments, but InsetMathNest requires at least one. if (nargs() > 0) InsetMathNest::drawSelection(pi, x, y); } void MathMacro::validate(LaTeXFeatures & features) const { if (name() == "binom" || name() == "mathcircumflex") features.require(to_utf8(name())); } InsetBase * MathMacro::editXY(LCursor & cur, int x, int y) { // We may have 0 arguments, but InsetMathNest requires at least one. if (nargs() > 0) return InsetMathNest::editXY(cur, x, y); else return this; } void MathMacro::detachArguments(std::vector &args) { args = cells_; cells_ = std::vector(); } void MathMacro::attachArguments(std::vector const &args) { cells_ = args; } bool MathMacro::idxFirst(LCursor & cur) const { cur.updateFlags(Update::Force); return InsetMathNest::idxFirst(cur); } bool MathMacro::idxLast(LCursor & cur) const { cur.updateFlags(Update::Force); return InsetMathNest::idxLast(cur); } bool MathMacro::notifyCursorLeaves(LCursor & cur) { cur.updateFlags(Update::Force); return InsetMathNest::notifyCursorLeaves(cur); } void MathMacro::maple(MapleStream & os) const { lyx::maple(expanded_, os); } void MathMacro::mathmlize(MathStream & os) const { lyx::mathmlize(expanded_, os); } void MathMacro::octave(OctaveStream & os) const { lyx::octave(expanded_, os); } void MathMacro::infoize(odocstream & os) const { os << "Macro: " << name(); } void MathMacro::infoize2(odocstream & os) const { os << "Macro: " << name(); } } // namespace lyx