]> git.lyx.org Git - lyx.git/blobdiff - src/BufferParams.cpp
* fix crashing special menu items on Mac after preferences have been changed.
[lyx.git] / src / BufferParams.cpp
index af14a55aa4c2f2f4cb617ccaf09acf091b1186d4..b545bf88f3b79d86a70f0255bec0fa6548a2fbd8 100644 (file)
@@ -335,6 +335,7 @@ BufferParams::BufferParams()
        use_bibtopic = false;
        trackChanges = false;
        outputChanges = false;
+       use_default_options = true;
        secnumdepth = 3;
        tocdepth = 3;
        language = default_language;
@@ -495,6 +496,8 @@ string BufferParams::readToken(Lexer & lex, string const & token,
        } else if (token == "\\options") {
                lex.eatLine();
                options = lex.getString();
+       } else if (token == "\\use_default_options") {
+               lex >> use_default_options;
        } else if (token == "\\master") {
                lex.eatLine();
                master = lex.getString();
@@ -692,6 +695,10 @@ void BufferParams::writeFile(ostream & os) const
                os << "\\options " << options << '\n';
        }
 
+       // use the class options defined in the layout?
+       os << "\\use_default_options " 
+          << convert<string>(use_default_options) << "\n";
+
        // the master document
        if (!master.empty()) {
                os << "\\master " << master << '\n';
@@ -1025,27 +1032,32 @@ bool BufferParams::writeLaTeX(odocstream & os, LaTeXFeatures & features,
                                language_options << ',';
                        language_options << language->babel();
                }
-               // FIXME: don't hardcode this! 
                // if Vietnamese is used, babel must directly be loaded
                // with language options, not in the class options, see
                // http://www.mail-archive.com/lyx-devel@lists.lyx.org/msg129417.html
-               //
+               size_t viet = language_options.str().find("vietnam");
+               // viet = string::npos when not found
                // the same is for all other languages that are not directly supported by
                // babel, but where LaTeX-packages add babel support.
                // this is currently the case for Latvian, Lithuanian, and Mongolian
-               //
+               size_t latvian = language_options.str().find("latvian");
+               size_t lithu = language_options.str().find("lithuanian");
+               size_t mongo = language_options.str().find("mongolian");
                // if Japanese is used, babel must directly be loaded
                // with language options, not in the class options, see
                // http://bugzilla.lyx.org/show_bug.cgi?id=4597#c4
+               size_t japan = language_options.str().find("japanese");
                if (lyxrc.language_global_options && !language_options.str().empty()
-                && !features.hasLanguage("vietnam")
-                && !features.hasLanguage("latvian")
-                && !features.hasLanguage("lithuanian")
-                && !features.hasLanguage("mongolian")
-                && !features.hasLanguage("japanese"))
+                       && viet == string::npos && japan == string::npos
+                       && latvian == string::npos && lithu == string::npos
+                       && mongo == string::npos)
                        clsoptions << language_options.str() << ',';
        }
 
+       // the predefined options from the layout
+       if (use_default_options && !tclass.options().empty())
+               clsoptions << tclass.options() << ',';
+
        // the user-defined options
        if (!options.empty()) {
                clsoptions << options << ',';
@@ -1417,7 +1429,7 @@ void BufferParams::useClassDefaults()
        sides = tclass.sides();
        columns = tclass.columns();
        pagestyle = tclass.pagestyle();
-       options = tclass.options();
+       use_default_options = true;
        // Only if class has a ToC hierarchy
        if (tclass.hasTocLevels()) {
                secnumdepth = tclass.secnumdepth();
@@ -1433,7 +1445,7 @@ bool BufferParams::hasClassDefaults() const
        return sides == tclass.sides()
                && columns == tclass.columns()
                && pagestyle == tclass.pagestyle()
-               && options == tclass.options()
+               && use_default_options
                && secnumdepth == tclass.secnumdepth()
                && tocdepth == tclass.tocdepth();
 }
@@ -1456,6 +1468,41 @@ void BufferParams::setDocumentClass(DocumentClass const * const tc) {
 }
 
 
+void BufferParams::addDefaultModules()
+{
+       // add any default modules not already in use
+       list<string> const & mods = baseClass()->defaultModules();
+       list<string>::const_iterator mit = mods.begin();
+       list<string>::const_iterator men = mods.end();
+
+       // we want to add these to the front, but in the right order,
+       // so we collect them here first.
+       list<string> modulesToAdd;
+
+       for (; mit != men; mit++) {
+               string const & modName = *mit;
+               // make sure the user hasn't removed it
+               if (find(removedModules_.begin(), removedModules_.end(), modName) !=
+                   removedModules_.end()) {
+                       LYXERR(Debug::TCLASS, "Default module `" << modName << 
+                                       "' not added because removed by user.");
+                       continue;
+               }
+
+               if (moduleCanBeAdded(modName)) {
+                       LYXERR(Debug::TCLASS, "Default module `" << modName << "' added.");
+                       modulesToAdd.push_back(modName);
+               } else
+                       LYXERR(Debug::TCLASS, 
+                                       "Default module `" << modName << "' could not be added.");
+       }
+
+       // OK, now we can add the default modules.
+       layoutModules_.insert(
+                       layoutModules_.begin(), modulesToAdd.begin(), modulesToAdd.end());
+}
+
+
 bool BufferParams::setBaseClass(string const & classname)
 {
        LYXERR(Debug::TCLASS, "setBaseClass: " << classname);
@@ -1472,7 +1519,7 @@ bool BufferParams::setBaseClass(string const & classname)
        }
 
        bool const success = bcl[classname].load();
-       if (!success) { 
+       if (!success) {
                docstring s = 
                        bformat(_("The document class %1$s could not be loaded."),
                        from_utf8(classname));
@@ -1481,52 +1528,8 @@ bool BufferParams::setBaseClass(string const & classname)
        }
 
        pimpl_->baseClass_ = classname;
+       addDefaultModules();
 
-       // add any required modules not already in use
-       list<string> const & mods = baseClass()->defaultModules();
-       list<string>::const_iterator mit = mods.begin();
-       list<string>::const_iterator men = mods.end();
-       for (; mit != men; mit++) {
-               string const & modName = *mit;
-               // see if we're already in use
-               if (find(layoutModules_.begin(), layoutModules_.end(), modName) != 
-                   layoutModules_.end()) {
-                       LYXERR(Debug::TCLASS, "Default module `" << modName << 
-                                       "' not added because already used.");
-                       continue;
-               }
-               // make sure the user hasn't removed it
-               if (find(removedModules_.begin(), removedModules_.end(), modName) != 
-                   removedModules_.end()) {
-                       LYXERR(Debug::TCLASS, "Default module `" << modName << 
-                                       "' not added because removed by user.");
-                       continue;
-               }
-               // Now we want to check the list of selected modules to see if any of them
-               // exclude this one.
-               bool foundit = false;
-               // so iterate over the selected modules...
-               LayoutModuleList::const_iterator lit = layoutModules_.begin();
-               LayoutModuleList::const_iterator len = layoutModules_.end();
-               for (; lit != len; lit++) {
-                       LyXModule * lm = moduleList[*lit];
-                       if (!lm)
-                               continue;
-                       vector<string> const & exc = lm->getExcludedModules();
-                       // ...and see if this one excludes us.
-                       if (find(exc.begin(), exc.end(), modName) != exc.end()) {
-                               foundit = true;
-                               LYXERR(Debug::TCLASS, "Default module `" << modName << 
-                                               "' not added because excluded by loaded module `" << 
-                                               *lit << "'.");
-                               break;
-                       }
-               }
-               if (!foundit) {
-                       LYXERR(Debug::TCLASS, "Default module `" << modName << "' added.");
-                       layoutModules_.push_back(modName);
-               }
-       }
        return true;
 }
 
@@ -1595,7 +1598,47 @@ void BufferParams::makeDocumentClass()
 }
 
 
-bool BufferParams::addLayoutModule(string const & modName) 
+bool BufferParams::moduleCanBeAdded(string const & modName) const
+{
+       // Is the module already present?
+       LayoutModuleList::const_iterator it = layoutModules_.begin();
+       LayoutModuleList::const_iterator end = layoutModules_.end();
+       for (; it != end; it++)
+               if (*it == modName) 
+                       return false;
+
+       LyXModule const * const lm = moduleList[modName];
+       if (!lm)
+               return true;
+
+       LayoutModuleList::const_iterator mit = getModules().begin();
+       LayoutModuleList::const_iterator const men = getModules().end();
+       // Check for conflicts with used modules
+       for (; mit != men; ++mit)
+               if (!LyXModule::areCompatible(modName, *mit))
+                       return false;
+
+       // Check whether some required module is available
+       vector<string> const reqs = lm->getRequiredModules();
+       if (reqs.empty())
+               return true;
+
+       mit = getModules().begin(); // reset
+       vector<string>::const_iterator rit = reqs.begin();
+       vector<string>::const_iterator ren = reqs.end();
+       bool foundone = false;
+       for (; rit != ren; ++rit) {
+               if (find(mit, men, *rit) != men) {
+                       foundone = true;
+                       break;
+               }
+       }
+
+       return foundone;
+}
+
+
+bool BufferParams::addLayoutModule(string const & modName)
 {
        LayoutModuleList::const_iterator it = layoutModules_.begin();
        LayoutModuleList::const_iterator end = layoutModules_.end();