* This file is part of LyX, the document processor.
* Licence details can be found in the file COPYING.
*
- * \author André Pönitz
+ * \author André Pönitz
*
* Full author contact details are available in file CREDITS.
*/
#include <config.h>
+#include "InsetMathSqrt.h"
#include "MacroTable.h"
#include "MathMacroTemplate.h"
#include "MathMacroArgument.h"
+#include "MathStream.h"
#include "MathSupport.h"
-#include "InsetMathSqrt.h"
-
#include "InsetMathNest.h"
-#include "Buffer.h"
-#include "support/debug.h"
+#include "Buffer.h"
#include "DocIterator.h"
+#include "InsetList.h"
+#include "Text.h"
-#include <boost/assert.hpp>
+#include "support/debug.h"
+#include "support/lassert.h"
#include <sstream>
//
/////////////////////////////////////////////////////////////////////
-MacroData::MacroData()
- : numargs_(0), lockCount_(0), redefinition_(false)
+MacroData::MacroData(Buffer * buf)
+ : buffer_(buf), queried_(true), numargs_(0), optionals_(0), lockCount_(0),
+ redefinition_(false), type_(MacroTypeNewcommand)
{}
-
-MacroData::MacroData(docstring const & definition,
- std::vector<docstring> const & defaults,
- int numargs, int optionals, docstring const & display,
- string const & requires)
- : definition_(definition), numargs_(numargs), display_(display),
- requires_(requires), lockCount_(0), redefinition_(false),
- optionals_(optionals), defaults_(defaults)
+
+
+MacroData::MacroData(Buffer * buf, DocIterator const & pos)
+ : buffer_(buf), pos_(pos), queried_(false), numargs_(0),
+ optionals_(0), lockCount_(0), redefinition_(false),
+ type_(MacroTypeNewcommand)
{
- defaults_.resize(optionals);
+}
+
+
+MacroData::MacroData(Buffer * buf, MathMacroTemplate const & macro)
+ : buffer_(buf), queried_(false), numargs_(0), optionals_(0), lockCount_(0),
+ redefinition_(false), type_(MacroTypeNewcommand)
+{
+ queryData(macro);
}
void MacroData::expand(vector<MathData> const & args, MathData & to) const
{
- InsetMathSqrt inset; // Hack. Any inset with a cell would do.
+ updateData();
+
+ // Hack. Any inset with a cell would do.
+ static InsetMathSqrt inset(0);
+ inset.setBuffer(const_cast<Buffer &>(*buffer_));
+
// FIXME UNICODE
asArray(display_.empty() ? definition_ : display_, inset.cell(0));
//lyxerr << "MathData::expand: args: " << args << endl;
//lyxerr << "MathData::expand: ar: " << inset.cell(0) << endl;
- for (DocIterator it = doc_iterator_begin(inset); it; it.forwardChar()) {
+ for (DocIterator it = doc_iterator_begin(buffer_, &inset); it; it.forwardChar()) {
if (!it.nextInset())
continue;
- if (it.nextInset()->lyxCode() != MATHMACROARG_CODE)
+ if (it.nextInset()->lyxCode() != MATH_MACROARG_CODE)
continue;
//it.cell().erase(it.pos());
//it.cell().insert(it.pos(), it.nextInset()->asInsetMath()
size_t MacroData::optionals() const
{
+ updateData();
return optionals_;
}
-std::vector<docstring> const & MacroData::defaults() const
+vector<docstring> const & MacroData::defaults() const
{
+ updateData();
return defaults_;
}
void MacroData::unlock() const
{
--lockCount_;
- BOOST_ASSERT(lockCount_ >= 0);
+ LASSERT(lockCount_ >= 0, /**/);
+}
+
+
+void MacroData::queryData(MathMacroTemplate const & macro) const
+{
+ if (queried_)
+ return;
+
+ queried_ = true;
+ definition_ = macro.definition();
+ numargs_ = macro.numArgs();
+ display_ = macro.displayDefinition();
+ redefinition_ = macro.redefinition();
+ type_ = macro.type();
+ optionals_ = macro.numOptionals();
+
+ macro.getDefaults(defaults_);
+}
+
+
+void MacroData::updateData() const
+{
+ if (queried_)
+ return;
+
+ LASSERT(buffer_ != 0, /**/);
+
+ // Try to fix position DocIterator. Should not do anything in theory.
+ pos_.fixIfBroken();
+
+ // find macro template
+ Inset * inset = pos_.nextInset();
+ if (inset == 0 || inset->lyxCode() != MATHMACRO_CODE) {
+ lyxerr << "BUG: No macro template found by MacroData" << endl;
+ return;
+ }
+
+ // query the data from the macro template
+ queryData(static_cast<MathMacroTemplate const &>(*inset));
+}
+
+
+int MacroData::write(odocstream & os, bool overwriteRedefinition) const
+{
+ updateData();
+
+ // find macro template
+ Inset * inset = pos_.nextInset();
+ if (inset == 0 || inset->lyxCode() != MATHMACRO_CODE) {
+ lyxerr << "BUG: No macro template found by MacroData" << endl;
+ return 0;
+ }
+
+ // output template
+ MathMacroTemplate const & tmpl =
+ static_cast<MathMacroTemplate const &>(*inset);
+ WriteStream wi(os, false, true, WriteStream::wsDefault);
+ return tmpl.write(wi, overwriteRedefinition);
}
}
-bool MacroTable::has(docstring const & name) const
-{
- return find(name) != end();
-}
-
-
-MacroData const & MacroTable::get(docstring const & name) const
+MacroData const * MacroTable::get(docstring const & name) const
{
const_iterator it = find(name);
- BOOST_ASSERT(it != end());
- return it->second;
+ return it == end() ? 0 : &it->second;
}
}
-void MacroTable::insert(docstring const & def, string const & requires)
+void MacroTable::insert(Buffer * buf, docstring const & def, string const & requires)
{
//lyxerr << "MacroTable::insert, def: " << to_utf8(def) << endl;
- MathMacroTemplate mac(def);
- MacroData data = mac.asMacroData();
+ MathMacroTemplate mac(buf, def);
+ MacroData data(buf, mac);
data.requires() = requires;
insert(mac.name(), data);
}
+void MacroTable::getMacroNames(std::set<docstring> & names) const
+{
+ for (const_iterator it = begin(); it != end(); ++it)
+ names.insert(it->first);
+}
+
+
void MacroTable::dump()
{
lyxerr << "\n------------------------------------------" << endl;
}
-MacroContext::MacroContext(Buffer const & buf, Paragraph const & par)
- : buf_(buf), par_(par)
-{
-}
-
-
-bool MacroContext::has(docstring const & name) const
-{
- // check if it's a local macro
- if (macros_.has(name))
- return true;
-
- // otherwise ask the buffer
- return buf_.hasMacro(name, par_);
-}
-
+/////////////////////////////////////////////////////////////////////
+//
+// MacroContext
+//
+/////////////////////////////////////////////////////////////////////
-MacroData const & MacroContext::get(docstring const & name) const
+MacroContext::MacroContext(Buffer const * buf, DocIterator const & pos)
+ : buf_(buf), pos_(pos)
{
- // check if it's a local macro
- if (macros_.has(name))
- return macros_.get(name);
-
- // ask the buffer for its macros
- return buf_.getMacro(name, par_);
}
-void MacroContext::insert(docstring const & name, MacroData const & data)
+MacroData const * MacroContext::get(docstring const & name) const
{
- macros_.insert(name, data);
+ return buf_->getMacro(name, pos_);
}
-
} // namespace lyx