/** * \file LayoutFileList.cpp * This file is part of LyX, the document processor. * Licence details can be found in the file COPYING. * * \author Lars Gullik Bjønnes * \author John Levon * * Full author contact details are available in file CREDITS. */ #include #include "LayoutFile.h" #include "Counters.h" #include "Floating.h" #include "FloatList.h" #include "Lexer.h" #include "TextClass.h" #include "support/assert.h" #include "support/debug.h" #include "support/FileName.h" #include "support/filetools.h" #include "support/gettext.h" #include #include #include using namespace std; using namespace lyx::support; namespace lyx { using boost::bind; using boost::regex; using boost::smatch; LayoutFile::LayoutFile(string const & fn, string const & cln, string const & desc, bool texClassAvail ) { name_ = fn; latexname_ = cln; description_ = desc; texClassAvail_ = texClassAvail; } LayoutFileList::~LayoutFileList() { ClassMap::const_iterator it = classmap_.begin(); ClassMap::const_iterator en = classmap_.end(); for (; it != en; ++it) { delete it->second; } } LayoutFileList & LayoutFileList::get() { static LayoutFileList baseclasslist; return baseclasslist; } bool LayoutFileList::haveClass(string const & classname) const { ClassMap::const_iterator it = classmap_.begin(); ClassMap::const_iterator en = classmap_.end(); for (; it != en; ++it) { if (it->first == classname) return true; } return false; } LayoutFile const & LayoutFileList::operator[](string const & classname) const { LASSERT(haveClass(classname), /**/); return *classmap_[classname]; } LayoutFile & LayoutFileList::operator[](string const & classname) { LASSERT(haveClass(classname), /**/); return *classmap_[classname]; } // Reads LyX textclass definitions according to textclass config file bool LayoutFileList::read() { Lexer lex; FileName const real_file = libFileSearch("", "textclass.lst"); LYXERR(Debug::TCLASS, "Reading textclasses from `" << real_file << '\''); if (real_file.empty()) { lyxerr << "LayoutFileList::Read: unable to find " "textclass file `" << to_utf8(makeDisplayPath(real_file.absFilename(), 1000)) << "'. Exiting." << endl; return false; // This causes LyX to end... Not a desirable behaviour. Lgb // What do you propose? That the user gets a file dialog // and is allowed to hunt for the file? (Asger) // more that we have a layout for minimal.cls statically // compiled in... (Lgb) } if (!lex.setFile(real_file)) { lyxerr << "LayoutFileList::Read: " "lyxlex was not able to set file: " << real_file << endl; } if (!lex.isOK()) { lyxerr << "LayoutFileList::Read: unable to open " "textclass file `" << to_utf8(makeDisplayPath(real_file.absFilename(), 1000)) << "'\nCheck your installation. LyX can't continue." << endl; return false; } bool finished = false; // Parse config-file LYXERR(Debug::TCLASS, "Starting parsing of textclass.lst"); while (lex.isOK() && !finished) { LYXERR(Debug::TCLASS, "\tline by line"); switch (lex.lex()) { case Lexer::LEX_FEOF: finished = true; break; default: string const fname = lex.getString(); LYXERR(Debug::TCLASS, "Fname: " << fname); if (lex.next()) { string const clname = lex.getString(); LYXERR(Debug::TCLASS, "Clname: " << clname); if (lex.next()) { string const desc = lex.getString(); LYXERR(Debug::TCLASS, "Desc: " << desc); if (lex.next()) { bool avail = lex.getBool(); LYXERR(Debug::TCLASS, "Avail: " << avail); // This code is run when we have // fname, clname, desc, and avail LayoutFile * tmpl = new LayoutFile(fname, clname, desc, avail); if (lyxerr.debugging(Debug::TCLASS)) { // only system layout files are loaded here so no // buffer path is needed. tmpl->load(); } classmap_[fname] = tmpl; } } } } } LYXERR(Debug::TCLASS, "End of parsing of textclass.lst"); // lyx will start with an empty classmap_, but only reconfigure is allowed // in this case. This gives users a second chance to configure lyx if // initial configuration fails. (c.f. bug 2829) if (classmap_.empty()) lyxerr << "LayoutFileList::Read: no textclasses found!" << endl; return true; } std::vector LayoutFileList::classList() const { std::vector cl; ClassMap::const_iterator it = classmap_.begin(); ClassMap::const_iterator en = classmap_.end(); for (; it != en; ++it) cl.push_back(it->first); return cl; } void LayoutFileList::reset(LayoutFileIndex const & classname) { LASSERT(haveClass(classname), /**/); LayoutFile * tc = classmap_[classname]; LayoutFile * tmpl = new LayoutFile(tc->name(), tc->latexname(), tc->description(), tc->isTeXClassAvailable()); classmap_[classname] = tmpl; delete tc; } string const LayoutFileList::localPrefix = "LOCAL:"; LayoutFileIndex LayoutFileList::addLayoutFile(string const & textclass, string const & path, Layout_Type type) { // FIXME There is a bug here: 4593 // // only check for textclass.layout file, .cls can be anywhere in $TEXINPUTS // NOTE: latex class name is defined in textclass.layout, which can be // different from textclass string fullName = addName(path, textclass + ".layout"); string localIndex; if (type == Local) localIndex = localPrefix + fullName; // if the local file has already been loaded, return it if (haveClass(localIndex)) return localIndex; FileName const layout_file(fullName); if (layout_file.exists()) { LYXERR(Debug::TCLASS, "Adding class " << textclass << " from directory " << path); // Read .layout file and get description, real latex classname etc // // This is a C++ version of function processLayoutFile in configure.py, // which uses the following regex // \Declare(LaTeX|DocBook)Class\s*(\[([^,]*)(,.*)*\])*\s*{(.*)} ifstream ifs(layout_file.toFilesystemEncoding().c_str()); static regex const reg("^#\\s*\\\\Declare(LaTeX|DocBook)Class\\s*" "(?:\\[([^,]*)(?:,.*)*\\])*\\s*\\{(.*)\\}\\s*"); string line; while (getline(ifs, line)) { // look for the \DeclareXXXClass line smatch sub; if (regex_match(line, sub, reg)) { // returns: whole string, classtype (not used here), class name, description LASSERT(sub.size() == 4, /**/); // now, create a TextClass with description containing path information string className(sub.str(2) == "" ? textclass : sub.str(2)); LayoutFile * tmpl = new LayoutFile(textclass, className, localIndex, true); // This textclass is added on request so it will definitely be // used. Load it now because other load() calls may fail if they // are called in a context without buffer path information. tmpl->load(path); classmap_[localIndex] = tmpl; return localIndex; } } } // If .layout is not in local directory, or an invalid layout is found, return null return string(); } LayoutFileIndex defaultBaseclass() { if (LayoutFileList::get().haveClass("article")) return string("article"); if (LayoutFileList::get().empty()) return string(); return LayoutFileList::get().classList().front(); } // Reads the style files bool LyXSetStyle() { LYXERR(Debug::TCLASS, "LyXSetStyle: parsing configuration..."); if (!LayoutFileList::get().read()) { LYXERR(Debug::TCLASS, "LyXSetStyle: an error occured " "during parsing.\n Exiting."); return false; } LYXERR(Debug::TCLASS, "LyXSetStyle: configuration parsed."); return true; } } // namespace lyx