// -*- C++ -*- /** * \file LayoutModuleList.cpp * This file is part of LyX, the document processor. * Licence details can be found in the file COPYING. * * \author Richard Kimberly Heck * * Full author contact details are available in file CREDITS. */ #include #include "LayoutModuleList.h" #include "LayoutFile.h" #include "ModuleList.h" #include "support/debug.h" #include #include #include using namespace std; namespace lyx { // the previous document class may have loaded some modules that the // new one excludes, and the new class may provide, etc, some that // conflict with ones that were already loaded. So we need to go // through the list and fix everything. I suppose there are various // ways this could be done, but the following seems to work at the // moment. (Thanks to Philippe Charpentier for helping work out all // the bugs---rgh.) bool LayoutModuleList::adaptToBaseClass(LayoutFile const * const lay, std::list const & removedModules) { // first, we remove any modules the new document class itself provides, // those it excludes, and those that conflict with ones it excludes. // this has to be done first because, otherwise, a module we're about // to remove could prevent a default module from being added. bool retval = removeBadModules(lay); // next, we add any default modules the new class provides. addDefaultModules(lay, removedModules); // finally, we perform a general consistency check on the set of // loaded modules. it's a hassle that we have to do this now, since // we just went through them a bit ago, but things might have changed // with the loading of the default modules. retval = (checkModuleConsistency(lay) || retval); return retval; } string LayoutModuleList::asString() const { string mods; for (auto const & mod : lml_) mods += mod + ','; // remove trailing comma if (!mods.empty()) mods.pop_back(); return mods; } bool LayoutModuleList::moduleCanBeAdded(string const & modName, LayoutFile const * const lay) const { // Is the module already present? const_iterator it = begin(); const_iterator const en = end(); for (; it != en; ++it) if (*it == modName) return false; LyXModule const * const lm = theModuleList[modName]; if (!lm) return true; // Does this module conflict with the document class or any loaded modules? if (moduleConflicts(modName, lay)) return false; // Is this module already provided by the document class? const_iterator const provmodstart = lay->providedModules().begin(); const_iterator const provmodend = lay->providedModules().end(); if (find(provmodstart, provmodend, modName) != provmodend) return false; // Check whether some required module is available vector const reqs = lm->getRequiredModules(); if (reqs.empty()) return true; const_iterator mit = begin(); const_iterator const men = end(); vector::const_iterator rit = reqs.begin(); vector::const_iterator ren = reqs.end(); bool foundone = false; for (; rit != ren; ++rit) { if (find(mit, men, *rit) != men || find(provmodstart, provmodend, *rit) != provmodend) { foundone = true; break; } } return foundone; } bool LayoutModuleList::moduleConflicts(string const & modName, LayoutFile const * const lay) const { // Is this module explicitly excluded by the document class? const_iterator const exclmodstart = lay->excludedModules().begin(); const_iterator const exclmodend = lay->excludedModules().end(); if (find(exclmodstart, exclmodend, modName) != exclmodend) return true; // Check for conflicts with used modules // first the provided modules... const_iterator provmodit = lay->providedModules().begin(); const_iterator const provmodend = lay->providedModules().end(); for (; provmodit != provmodend; ++provmodit) { if (!LyXModule::areCompatible(modName, *provmodit)) return true; } // and then the selected modules const_iterator mit = begin(); const_iterator const men = end(); for (; mit != men; ++mit) if (!LyXModule::areCompatible(modName, *mit)) return true; return false; } void LayoutModuleList::addDefaultModules(LayoutFile const * const lay, std::list removedModules) { LayoutModuleList mods = lay->defaultModules(); const_iterator mit = mods.begin(); const_iterator const men = mods.end(); // We want to insert the default modules at the beginning of // the list, but also to insert them in the correct order. // The obvious thing to do would be to collect them and then // insert them, but that doesn't work because a later default // module may require an earlier one, and then the test below // moduleCanBeAdded(modname) // will fail. So we have to do it a more complicated way. iterator insertpos = begin(); int numinserts = 0; 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, lay)) { // FIXME This could be because it's already present, so we should // probably return something indicating that. LYXERR(Debug::TCLASS, "Default module `" << modName << "' could not be added."); continue; } LYXERR(Debug::TCLASS, "Default module `" << modName << "' added."); insert(insertpos, modName); // now we reset insertpos ++numinserts; insertpos = begin(); advance(insertpos, numinserts); } } bool LayoutModuleList::removeBadModules(LayoutFile const * const lay) { // we'll write a new list of modules, since we can't just remove them, // as that would invalidate our iterators LayoutModuleList oldModules = *this; clear(); LayoutModuleList const & provmods = lay->providedModules(); LayoutModuleList const & exclmods = lay->excludedModules(); bool consistent = true; // set to false if we have to do anything const_iterator oit = oldModules.begin(); const_iterator const oen = oldModules.end(); for (; oit != oen; ++oit) { string const & modname = *oit; // skip modules that the class provides if (find(provmods.begin(), provmods.end(), modname) != provmods.end()) { LYXERR0("Module `" << modname << "' dropped because provided by document class."); consistent = false; continue; } // are we excluded by the document class? if (find(exclmods.begin(), exclmods.end(), modname) != exclmods.end()) { LYXERR0("Module `" << modname << "' dropped because excluded by document class."); consistent = false; continue; } // determine whether some provided module excludes us or we exclude it const_iterator pit = provmods.begin(); const_iterator const pen = provmods.end(); bool excluded = false; for (; !excluded && pit != pen; ++pit) { if (!LyXModule::areCompatible(modname, *pit)) { LYXERR0("Module " << modname << " dropped because it conflicts with provided module `" << *pit << "'."); consistent = false; excluded = true; } } if (excluded) continue; push_back(modname); } return consistent; } // Perform a consistency check on the set of modules. We need to make // sure that none of the modules exclude each other and that requires // are satisfied. bool LayoutModuleList::checkModuleConsistency(LayoutFile const * const lay) { bool consistent = true; LayoutModuleList oldModules = *this; clear(); const_iterator oit = oldModules.begin(); const_iterator const oen = oldModules.end(); LayoutModuleList const & provmods = lay->providedModules(); for (; oit != oen; ++oit) { string const & modname = *oit; bool excluded = false; // Determine whether some prior module excludes us, or we exclude it const_iterator lit = begin(); const_iterator const len = end(); for (; !excluded && lit != len; ++lit) { if (!LyXModule::areCompatible(modname, *lit)) { consistent = false; LYXERR0("Module " << modname << " dropped because it is excluded by prior module " << *lit); excluded = true; } } if (excluded) continue; // determine whether some provided module or some prior module // satisfies our requirements LyXModule const * const oldmod = theModuleList[modname]; if (!oldmod) { LYXERR0("Default module " << modname << " added although it is unavailable and can't check requirements."); continue; } vector const & reqs = oldmod->getRequiredModules(); if (!reqs.empty()) { // we now set excluded to true, meaning that we haven't // yet found a required module. excluded = true; vector::const_iterator rit = reqs.begin(); vector::const_iterator const ren = reqs.end(); for (; rit != ren; ++rit) { string const reqmod = *rit; if (find(provmods.begin(), provmods.end(), reqmod) != provmods.end()) { excluded = false; break; } if (find(begin(), end(), reqmod) != end()) { excluded = false; break; } } } if (excluded) { consistent = false; LYXERR0("Module " << modname << " dropped because requirements not met."); } else { LYXERR(Debug::TCLASS, "Module " << modname << " passed consistency check."); push_back(modname); } } return consistent; } } // namespace lyx