]> git.lyx.org Git - lyx.git/blobdiff - src/Buffer.cpp
cosmetics
[lyx.git] / src / Buffer.cpp
index b9b7e4bf65fb625cc6489019f0134c3270d71994..edb0684b739a678a8623e1d82cbe9d42e892f26e 100644 (file)
@@ -4,6 +4,7 @@
  * Licence details can be found in the file COPYING.
  *
  * \author Lars Gullik Bjønnes
+ * \author Stefan Schimanski
  *
  * Full author contact details are available in file CREDITS.
  */
@@ -68,8 +69,8 @@
 #include "insets/InsetInclude.h"
 #include "insets/InsetText.h"
 
-#include "mathed/MathMacroTemplate.h"
 #include "mathed/MacroTable.h"
+#include "mathed/MathMacroTemplate.h"
 #include "mathed/MathSupport.h"
 
 #include "frontends/alert.h"
@@ -84,7 +85,6 @@
 #include "support/FileFilterList.h"
 #include "support/filetools.h"
 #include "support/Forkedcall.h"
-#include "support/fs_extras.h"
 #include "support/gzstream.h"
 #include "support/lyxlib.h"
 #include "support/os.h"
@@ -97,8 +97,6 @@
 #endif
 
 #include <boost/bind.hpp>
-#include <boost/filesystem/exception.hpp>
-#include <boost/filesystem/operations.hpp>
 #include <boost/shared_ptr.hpp>
 
 #include <algorithm>
@@ -152,11 +150,10 @@ using support::suffixIs;
 
 namespace Alert = frontend::Alert;
 namespace os = support::os;
-namespace fs = boost::filesystem;
 
 namespace {
 
-int const LYX_FORMAT = 298; //Schimmi: Optional parameters in macros
+int const LYX_FORMAT = 299; // Uwe: Hyperlink types
 
 } // namespace anon
 
@@ -200,12 +197,14 @@ public:
        /// our Text that should be wrapped in an InsetText
        InsetText inset;
 
-       ///
-       MacroTable macros;
-
        ///
        TocBackend toc_backend;
 
+       /// macro table
+       typedef std::map<unsigned int, MacroData, std::greater<int> > PositionToMacroMap;
+       typedef std::map<docstring, PositionToMacroMap> NameToPositionMacroMap;
+       NameToPositionMacroMap macros;
+
        /// Container for all sort of Buffer dependant errors.
        map<string, ErrorList> errorLists;
 
@@ -398,12 +397,15 @@ string Buffer::latexName(bool const no_path) const
 }
 
 
-pair<Buffer::LogType, string> Buffer::logName() const
+string Buffer::logName(LogType * type) const
 {
        string const filename = latexName(false);
 
-       if (filename.empty())
-               return make_pair(Buffer::latexlog, string());
+       if (filename.empty()) {
+               if (type)
+                       *type = latexlog;
+               return string();
+       }
 
        string const path = temppath();
 
@@ -420,10 +422,14 @@ pair<Buffer::LogType, string> Buffer::logName() const
        if (bname.exists() &&
            (!fname.exists() || fname.lastModified() < bname.lastModified())) {
                LYXERR(Debug::FILES) << "Log name calculated as: " << bname << endl;
-               return make_pair(Buffer::buildlog, bname.absFilename());
+               if (type)
+                       *type = buildlog;
+               return bname.absFilename();
        }
        LYXERR(Debug::FILES) << "Log name calculated as: " << fname << endl;
-       return make_pair(Buffer::latexlog, fname.absFilename());
+       if (type)
+                       *type = latexlog;
+       return fname.absFilename();
 }
 
 
@@ -823,10 +829,6 @@ Buffer::ReadStatus Buffer::readFile(Lexer & lex, FileName const & filename,
                                       from_utf8(filename.absFilename())));
        }
 
-       //lyxerr << "removing " << MacroTable::localMacros().size()
-       //      << " temporary macro entries" << endl;
-       //MacroTable::localMacros().clear();
-
        pimpl_->file_fully_loaded = true;
        return success;
 }
@@ -844,7 +846,7 @@ bool Buffer::save() const
        bool madeBackup = false;
 
        // make a backup if the file already exists
-       if (lyxrc.make_backup && fs::exists(encodedFilename)) {
+       if (lyxrc.make_backup && fileName().exists()) {
                backupName = FileName(absFileName() + '~');
                if (!lyxrc.backupdir_path.empty()) {
                        string const mangledName =
@@ -853,20 +855,19 @@ bool Buffer::save() const
                        backupName = FileName(addName(lyxrc.backupdir_path,
                                                      mangledName));
                }
-               try {
-                       fs::copy_file(encodedFilename, backupName.toFilesystemEncoding(), false);
+               if (fileName().copyTo(backupName, false)) {
                        madeBackup = true;
-               } catch (fs::filesystem_error const & fe) {
+               } else {
                        Alert::error(_("Backup failure"),
                                     bformat(_("Cannot create backup file %1$s.\n"
                                               "Please check whether the directory exists and is writeable."),
                                             from_utf8(backupName.absFilename())));
-                       LYXERR(Debug::DEBUG) << "Fs error: " << fe.what() << endl;
+                       //LYXERR(Debug::DEBUG) << "Fs error: " << fe.what() << endl;
                }
        }
 
        // ask if the disk file has been externally modified (use checksum method)
-       if (fs::exists(encodedFilename) && isExternallyModified(checksum_method)) {
+       if (fileName().exists() && isExternallyModified(checksum_method)) {
                docstring const file = makeDisplayPath(absFileName(), 20);
                docstring text = bformat(_("Document %1$s has been externally modified. Are you sure "
                                                             "you want to overwrite this file?"), file);
@@ -948,7 +949,7 @@ bool Buffer::write(ostream & ofs) const
        AuthorList::Authors::const_iterator a_it = params().authors().begin();
        AuthorList::Authors::const_iterator a_end = params().authors().end();
        for (; a_it != a_end; ++a_it)
-               a_it->second.used(false);
+               a_it->second.setUsed(false);
 
        ParIterator const end = par_iterator_end();
        ParIterator it = par_iterator_begin();
@@ -1312,7 +1313,7 @@ int Buffer::runChktex()
        string const name = addName(path.absFilename(), latexName());
        string const org_path = filePath();
 
-       support::Path p(path); // path to LaTeX file
+       support::PathChanger p(path); // path to LaTeX file
        message(_("Running chktex..."));
 
        // Generate the LaTeX file if neccessary
@@ -1694,6 +1695,12 @@ void Buffer::markDirty()
 }
 
 
+FileName Buffer::fileName() const
+{
+       return pimpl_->filename;
+}
+
+
 string Buffer::absFileName() const
 {
        return pimpl_->filename.absFilename();
@@ -1725,7 +1732,7 @@ void Buffer::setParentName(string const & name)
 Buffer const * Buffer::masterBuffer() const
 {
        if (!params().parentname.empty()
-           && theBufferList().exists(params().parentname)) {
+               && theBufferList().exists(params().parentname)) {
                Buffer const * buf = theBufferList().getBuffer(params().parentname);
                //We need to check if the parent is us...
                //FIXME RECURSIVE INCLUDE
@@ -1754,44 +1761,104 @@ Buffer * Buffer::masterBuffer()
 }
 
 
-MacroData const & Buffer::getMacro(docstring const & name) const
+bool Buffer::hasMacro(docstring const & name, Paragraph const & par) const
 {
-       return pimpl_->macros.get(name);
+       Impl::PositionToMacroMap::iterator it;
+       it = pimpl_->macros[name].upper_bound(par.macrocontextPosition());
+       if (it != pimpl_->macros[name].end())
+               return true;
+
+       // If there is a master buffer, query that
+       const Buffer * master = masterBuffer();
+       if (master && master != this)
+               return master->hasMacro(name);
+
+       return MacroTable::globalMacros().has(name);
 }
 
 
 bool Buffer::hasMacro(docstring const & name) const
 {
-       return pimpl_->macros.has(name);
+       if( !pimpl_->macros[name].empty() )
+               return true;
+
+       // If there is a master buffer, query that
+       const Buffer * master = masterBuffer();
+       if (master && master != this)
+               return master->hasMacro(name);
+
+       return MacroTable::globalMacros().has(name);
 }
 
 
-void Buffer::insertMacro(docstring const & name, MacroData const & data)
+MacroData const & Buffer::getMacro(docstring const & name,
+       Paragraph const & par) const
 {
-       MacroTable::globalMacros().insert(name, data);
-       pimpl_->macros.insert(name, data);
+       Impl::PositionToMacroMap::iterator it;
+       it = pimpl_->macros[name].upper_bound(par.macrocontextPosition());
+       if( it != pimpl_->macros[name].end() )
+               return it->second;
+
+       // If there is a master buffer, query that
+       const Buffer * master = masterBuffer();
+       if (master && master != this)
+               return master->getMacro(name);
+
+       return MacroTable::globalMacros().get(name);
 }
 
 
-void Buffer::buildMacros()
+MacroData const & Buffer::getMacro(docstring const & name) const
 {
-       // Start with global table.
-       pimpl_->macros = MacroTable::globalMacros();
+       Impl::PositionToMacroMap::iterator it;
+       it = pimpl_->macros[name].begin();
+       if( it != pimpl_->macros[name].end() )
+               return it->second;
+
+       // If there is a master buffer, query that
+       const Buffer * master = masterBuffer();
+       if (master && master != this)
+               return master->getMacro(name);
 
-       // Now add our own.
-       ParagraphList const & pars = text().paragraphs();
+       return MacroTable::globalMacros().get(name);
+}
+
+
+void Buffer::updateMacros()
+{
+       // start with empty table
+       pimpl_->macros = Impl::NameToPositionMacroMap();
+
+       // Iterate over buffer
+       ParagraphList & pars = text().paragraphs();
        for (size_t i = 0, n = pars.size(); i != n; ++i) {
+               // set position again
+               pars[i].setMacrocontextPosition(i);
+
                //lyxerr << "searching main par " << i
                //      << " for macro definitions" << std::endl;
                InsetList const & insets = pars[i].insetList();
                InsetList::const_iterator it = insets.begin();
                InsetList::const_iterator end = insets.end();
                for ( ; it != end; ++it) {
-                       //lyxerr << "found inset code " << it->inset->lyxCode() << std::endl;
-                       if (it->inset->lyxCode() == MATHMACRO_CODE) {
-                               MathMacroTemplate const & mac
-                                       = static_cast<MathMacroTemplate const &>(*it->inset);
-                               insertMacro(mac.name(), mac.asMacroData());
+                       if (it->inset->lyxCode() != MATHMACRO_CODE)
+                               continue;
+                       
+                       // get macro data
+                       MathMacroTemplate const & macroTemplate
+                       = static_cast<MathMacroTemplate const &>(*it->inset);
+
+                       // valid?
+                       if (macroTemplate.validMacro()) {
+                               MacroData macro = macroTemplate.asMacroData();
+
+                               // redefinition?
+                               // call hasMacro here instead of directly querying mc to
+                               // also take the master document into consideration
+                               macro.setRedefinition(hasMacro(macroTemplate.name()));
+
+                               // register macro (possibly overwrite the previous one of this paragraph)
+                               pimpl_->macros[macroTemplate.name()][i] = macro;
                        }
                }
        }