]> git.lyx.org Git - lyx.git/blobdiff - src/mathed/MacroTable.cpp
Merge branch 'master' of git.lyx.org:lyx
[lyx.git] / src / mathed / MacroTable.cpp
index 17e177b4228beff368d249b43bb17959d4ce662f..20a713a513e15e1d3af377282e4cfe63d359cfe5 100644 (file)
@@ -3,59 +3,81 @@
  * 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 "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>
 
+using namespace std;
 
 namespace lyx {
 
-using std::endl;
-using std::istringstream;
-using std::map;
-using std::pair;
-using std::string;
-using std::vector;
-using std::size_t;
 
+/////////////////////////////////////////////////////////////////////
+//
+// MacroData
+//
+/////////////////////////////////////////////////////////////////////
 
-MacroData::MacroData()
-       : numargs_(0), lockCount_(0)
+MacroData::MacroData(Buffer * buf)
+       : buffer_(buf), queried_(true), numargs_(0), optionals_(0), lockCount_(0),
+         redefinition_(false), type_(MacroTypeNewcommand)
 {}
 
-
-MacroData::MacroData(docstring const & def, int numargs, docstring const & disp, string const & requires)
-       : def_(def), numargs_(numargs), disp_(disp), requires_(requires), lockCount_(0)
-{}
+       
+       
+MacroData::MacroData(Buffer * buf, DocIterator const & pos)
+       : buffer_(buf), pos_(pos), queried_(false), numargs_(0),
+         optionals_(0), lockCount_(0), redefinition_(false),
+         type_(MacroTypeNewcommand)
+{
+}
+       
+       
+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(disp_.empty() ? def_ : disp_, inset.cell(0));
+       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()
@@ -70,25 +92,102 @@ void MacroData::expand(vector<MathData> const & args, MathData & to) const
 }
 
 
-// The global table.
-MacroTable & MacroTable::globalMacros()
+size_t MacroData::optionals() const
 {
-       static MacroTable theGlobalMacros;
-       return theGlobalMacros;
+       updateData();
+       return optionals_;
+}
+
+
+vector<docstring> const &  MacroData::defaults() const
+{
+       updateData();
+       return defaults_;
+}
+
+
+void MacroData::unlock() const
+{
+       --lockCount_;
+       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();
 
-bool MacroTable::has(docstring const & name) const
+       // 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);
+}
+
+
+/////////////////////////////////////////////////////////////////////
+//
+// The global table of macros
+//
+/////////////////////////////////////////////////////////////////////
+
+MacroTable & MacroTable::globalMacros()
 {
-       return find(name) != end();
+       static MacroTable theGlobalMacros;
+       return theGlobalMacros;
 }
 
 
-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;
 }
 
 
@@ -99,26 +198,50 @@ void MacroTable::insert(docstring const & name, MacroData const & data)
 }
 
 
-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;
        for (const_iterator it = begin(); it != end(); ++it)
                lyxerr << to_utf8(it->first)
-                       << " [" << to_utf8(it->second.def()) << "] : "
-                       << " [" << to_utf8(it->second.disp()) << "] : "
+                       << " [" << to_utf8(it->second.definition()) << "] : "
+                       << " [" << to_utf8(it->second.display()) << "] : "
                        << endl;
        lyxerr << "------------------------------------------" << endl;
 }
 
 
+/////////////////////////////////////////////////////////////////////
+//
+// MacroContext
+//
+/////////////////////////////////////////////////////////////////////
+
+MacroContext::MacroContext(Buffer const * buf, DocIterator const & pos)
+       : buf_(buf), pos_(pos)
+{
+}
+
+
+MacroData const * MacroContext::get(docstring const & name) const
+{
+       return buf_->getMacro(name, pos_);
+}
+
 } // namespace lyx