3 * This file is part of LyX, the document processor.
4 * Licence details can be found in the file COPYING.
8 * Full author contact details are available in file CREDITS.
13 #include "InsetMathSqrt.h"
14 #include "MacroTable.h"
15 #include "MathMacroTemplate.h"
16 #include "MathMacroArgument.h"
17 #include "MathParser.h"
18 #include "MathStream.h"
19 #include "MathSupport.h"
20 #include "InsetMathNest.h"
23 #include "DocIterator.h"
24 #include "InsetList.h"
27 #include "support/debug.h"
28 #include "support/gettext.h"
29 #include "support/lassert.h"
38 /////////////////////////////////////////////////////////////////////
42 /////////////////////////////////////////////////////////////////////
44 MacroData::MacroData(Buffer * buf)
45 : buffer_(buf), queried_(true), numargs_(0), sym_(0), optionals_(0),
46 lockCount_(0), redefinition_(false), type_(MacroTypeNewcommand)
50 MacroData::MacroData(Buffer * buf, DocIterator const & pos)
51 : buffer_(buf), pos_(pos), queried_(false), numargs_(0), sym_(0),
52 optionals_(0), lockCount_(0), redefinition_(false),
53 type_(MacroTypeNewcommand)
58 MacroData::MacroData(Buffer * buf, MathMacroTemplate const & macro)
59 : buffer_(buf), queried_(false), numargs_(0), sym_(0), optionals_(0),
60 lockCount_(0), redefinition_(false), type_(MacroTypeNewcommand)
66 bool MacroData::expand(vector<MathData> const & args, MathData & to) const
70 // Hack. Any inset with a cell would do.
71 static InsetMathSqrt inset(0);
72 inset.setBuffer(const_cast<Buffer &>(*buffer_));
74 docstring const & definition(display_.empty() ? definition_ : display_);
75 asArray(definition, inset.cell(0));
76 //lyxerr << "MathData::expand: args: " << args << endl;
77 //LYXERR0("MathData::expand: ar: " << inset.cell(0));
78 for (DocIterator it = doc_iterator_begin(buffer_, &inset); it; it.forwardChar()) {
81 if (it.nextInset()->lyxCode() != MATH_MACROARG_CODE)
83 //it.cell().erase(it.pos());
84 //it.cell().insert(it.pos(), it.nextInset()->asInsetMath()
85 size_t n = static_cast<MathMacroArgument*>(it.nextInset())->number();
86 if (n <= args.size()) {
87 it.cell().erase(it.pos());
88 it.cell().insert(it.pos(), args[n - 1]);
91 //LYXERR0("MathData::expand: res: " << inset.cell(0));
93 // If the result is equal to the definition then we either have a
94 // recursive loop, or the definition did not contain any macro in the
96 return asString(to) != definition;
100 size_t MacroData::optionals() const
107 vector<docstring> const & MacroData::defaults() const
114 string const MacroData::requires() const
117 return sym_->requires;
122 bool MacroData::hidden() const
130 docstring const MacroData::xmlname() const
133 return sym_->xmlname;
138 char const * MacroData::MathMLtype() const
140 return sym_ ? sym_->MathMLtype() : 0;
144 void MacroData::unlock() const
147 LASSERT(lockCount_ >= 0, lockCount_ = 0);
151 void MacroData::queryData(MathMacroTemplate const & macro) const
157 definition_ = macro.definition();
158 numargs_ = macro.numArgs();
159 display_ = macro.displayDefinition();
160 redefinition_ = macro.redefinition();
161 type_ = macro.type();
162 optionals_ = macro.numOptionals();
164 macro.getDefaults(defaults_);
168 void MacroData::updateData() const
175 // Try to fix position DocIterator. Should not do anything in theory.
178 // find macro template
179 Inset * inset = pos_.nextInset();
180 if (inset == 0 || inset->lyxCode() != MATHMACRO_CODE) {
181 lyxerr << "BUG: No macro template found by MacroData" << endl;
185 // query the data from the macro template
186 queryData(static_cast<MathMacroTemplate const &>(*inset));
190 int MacroData::write(odocstream & os, bool overwriteRedefinition) const
194 // find macro template
195 Inset * inset = pos_.nextInset();
196 if (inset == 0 || inset->lyxCode() != MATHMACRO_CODE) {
197 lyxerr << "BUG: No macro template found by MacroData" << endl;
202 MathMacroTemplate const & tmpl =
203 static_cast<MathMacroTemplate const &>(*inset);
204 WriteStream wi(os, false, true, WriteStream::wsDefault);
205 return tmpl.write(wi, overwriteRedefinition);
209 /////////////////////////////////////////////////////////////////////
211 // The global table of macros
213 /////////////////////////////////////////////////////////////////////
215 MacroTable & MacroTable::globalMacros()
217 static MacroTable theGlobalMacros;
218 return theGlobalMacros;
222 MacroData const * MacroTable::get(docstring const & name) const
224 const_iterator it = find(name);
225 return it == end() ? 0 : &it->second;
230 MacroTable::insert(docstring const & name, MacroData const & data)
232 //lyxerr << "MacroTable::insert: " << to_utf8(name) << endl;
233 iterator it = find(name);
235 it = map<docstring, MacroData>::insert(
236 make_pair(name, data)).first;
244 MacroTable::insert(Buffer * buf, docstring const & def)
246 //lyxerr << "MacroTable::insert, def: " << to_utf8(def) << endl;
247 MathMacroTemplate mac(buf, def);
248 MacroData data(buf, mac);
249 return insert(mac.name(), data);
253 void MacroTable::getMacroNames(std::set<docstring> & names, bool gethidden) const
255 for (const_iterator it = begin(); it != end(); ++it)
256 if (gethidden || !it->second.hidden())
257 names.insert(it->first);
261 void MacroTable::dump()
263 lyxerr << "\n------------------------------------------" << endl;
264 for (const_iterator it = begin(); it != end(); ++it)
265 lyxerr << to_utf8(it->first)
266 << " [" << to_utf8(it->second.definition()) << "] : "
267 << " [" << to_utf8(it->second.display()) << "] : "
269 lyxerr << "------------------------------------------" << endl;
273 /////////////////////////////////////////////////////////////////////
277 /////////////////////////////////////////////////////////////////////
279 MacroContext::MacroContext(Buffer const * buf, DocIterator const & pos)
280 : buf_(buf), pos_(pos)
285 MacroData const * MacroContext::get(docstring const & name) const
287 return buf_->getMacro(name, pos_);