diff --git a/src/graphics/ChangeLog b/src/graphics/ChangeLog index 4210985328..a9864a1716 100644 --- a/src/graphics/ChangeLog +++ b/src/graphics/ChangeLog @@ -1,3 +1,8 @@ +2002-07-06 Angus Leeming + + * PreviewLoader.C: ensure that the metrics data is used by the + correct image! + 2002-07-05 Angus Leeming * PreviewImage.h: diff --git a/src/graphics/PreviewLoader.C b/src/graphics/PreviewLoader.C index 6a18ebd988..9519bf49fd 100644 --- a/src/graphics/PreviewLoader.C +++ b/src/graphics/PreviewLoader.C @@ -40,14 +40,39 @@ #include using std::endl; -using std::ofstream; -using std::ostream; +using std::find_if; using std::setfill; using std::setw; +using std::sort; +using std::map; +using std::ofstream; +using std::ostream; +using std::pair; +using std::vector; namespace { +typedef pair StrPair; + +struct CompSecond { + bool operator()(StrPair const & lhs, StrPair const & rhs) + { + return lhs.second < rhs.second; + } +}; + +struct FindFirst { + FindFirst(string const & comp) : comp_(comp) {} + bool operator()(StrPair const & sp) + { + return sp.first < comp_; + } +private: + string const comp_; +}; + + string const unique_filename() { @@ -84,54 +109,63 @@ struct PreviewLoader::Impl : public boost::signals::trackable { /// void startLoading(); + /// + typedef pair StrPair; + /// + typedef map PendingMap; + private: /// Called by the Forkedcall process that generated the bitmap files. void finishedGenerating(string const &, pid_t, int); /// void dumpPreamble(ostream &) const; /// - void dumpData(ostream &) const; + void dumpData(ostream &, vector const &) const; /// static void setConverter(); /// We don't own this static Converter const * pconverter_; - /** The cache allows easy retrieval of already-generated images + /** cache_ allows easy retrieval of already-generated images * using the LaTeX snippet as the identifier. */ typedef boost::shared_ptr PreviewImagePtr; /// - typedef std::map Cache; + typedef map Cache; /// Cache cache_; - /** The map stores the LaTeX snippet and the name of the generated - * bitmap image file. + /** pending_ stores the LaTeX snippet and the name of the generated + * bitmap image file in anticipation of them being sent to the + * converter. */ - typedef std::map PendingMap; - /// PendingMap pending_; /// Store info on a currently executing, forked process. struct InProgress { - /// - typedef std::map PendingMap; /// InProgress() {} /// - InProgress(string const & mf, PendingMap const & s) - : metrics_file(mf), snippets(s) - {} + InProgress(string const & f, PendingMap const & m) + : metrics_file(f), snippets(m.begin(), m.end()) + { + sort(snippets.begin(), snippets.end(), CompSecond()); + } + /// string metrics_file; - /// - PendingMap snippets; + + /** Store the info in the PendingMap as a vector. + Ensures that the data is output in the order + file001, file002 etc, as we expect, which is /not/ what + happens when we iterate through the map. + */ + vector snippets; }; - friend class InProgress; /// Store all forked processes so that we can proceed thereafter. - typedef std::map InProgressMap; + typedef map InProgressMap; /// InProgressMap in_progress_; @@ -143,7 +177,6 @@ private: Buffer const & buffer_; }; - Converter const * PreviewLoader::Impl::pconverter_; @@ -245,13 +278,16 @@ PreviewLoader::Impl::status(string const & latex_snippet) const if (pit != pending_.end()) return PreviewLoader::InQueue; - InProgressMap::const_iterator git = in_progress_.begin(); - InProgressMap::const_iterator gend = in_progress_.end(); + InProgressMap::const_iterator ipit = in_progress_.begin(); + InProgressMap::const_iterator ipend = in_progress_.end(); - for (; git != gend; ++git) { - PendingMap::const_iterator pit = - git->second.snippets.find(latex_snippet); - if (pit != git->second.snippets.end()) + for (; ipit != ipend; ++ipit) { + vector const & snippets = ipit->second.snippets; + vector::const_iterator vit = snippets.begin(); + vector::const_iterator vend = snippets.end(); + vit = find_if(vit, vend, FindFirst(latex_snippet)); + + if (vit != vend) return PreviewLoader::Processing; } @@ -296,19 +332,22 @@ void PreviewLoader::Impl::remove(string const & latex_snippet) if (pit != pending_.end()) pending_.erase(pit); - InProgressMap::iterator git = in_progress_.begin(); - InProgressMap::iterator gend = in_progress_.end(); + InProgressMap::iterator ipit = in_progress_.begin(); + InProgressMap::iterator ipend = in_progress_.end(); - while (git != gend) { - InProgressMap::iterator curr = git; - ++git; + while (ipit != ipend) { + InProgressMap::iterator curr = ipit; + ++ipit; - PendingMap::iterator pit = - curr->second.snippets.find(latex_snippet); - if (pit != curr->second.snippets.end()) - curr->second.snippets.erase(pit); + vector & snippets = curr->second.snippets; + vector::iterator vit = snippets.begin(); + vector::iterator vend = snippets.end(); + vit = find_if(vit, vend, FindFirst(latex_snippet)); + + if (vit != vend) + snippets.erase(vit, vit+1); - if (curr->second.snippets.empty()) + if (snippets.empty()) in_progress_.erase(curr); } } @@ -327,31 +366,32 @@ void PreviewLoader::Impl::startLoading() lyxerr[Debug::GRAPHICS] << "PreviewLoader::startLoading()" << endl; + // Create an InProgress instance to place in the map of all + // such processes if it starts correctly. + string const metrics_file = filename_base_ + ".metrics"; + InProgress inprogress(metrics_file, pending_); + // Output the LaTeX file. string const latexfile = filename_base_ + ".tex"; ofstream of(latexfile.c_str()); dumpPreamble(of); of << "\n\\begin{document}\n"; - dumpData(of); + dumpData(of, inprogress.snippets); of << "\n\\end{document}\n"; of.close(); + // Reset the filename and clear pending_, so we're ready to + // start afresh. + pending_.clear(); + filename_base_ = unique_filename(); + // The conversion command. ostringstream cs; cs << pconverter_->command << " " << latexfile << " " << tostr(0.01 * lyxrc.dpi * lyxrc.zoom); - // Store the generation process in a list of all generating processes - // (I anticipate that this will be small!) string const command = cs.str().c_str(); - string const metrics_file = filename_base_ + ".metrics"; - in_progress_[command] = InProgress(metrics_file, pending_); - - // Reset the filename and clear the data, so we're ready to - // start afresh. - pending_.clear(); - filename_base_ = unique_filename(); // Initiate the conversion from LaTeX to bitmap images files. Forkedcall::SignalTypePtr convert_ptr; @@ -364,14 +404,15 @@ void PreviewLoader::Impl::startLoading() int ret = call.startscript(command, convert_ptr); if (ret != 0) { - InProgressMap::iterator it = in_progress_.find(command); - if (it != in_progress_.end()) - in_progress_.erase(it); - lyxerr[Debug::GRAPHICS] << "PreviewLoader::startLoading()\n" << "Unable to start process \n" << command << endl; + return; } + + // Store the generation process in a list of all generating processes + // (I anticipate that this will be small!) + in_progress_[command] = inprogress; } @@ -398,8 +439,8 @@ void PreviewLoader::Impl::finishedGenerating(string const & command, // Add these newly generated bitmap files to the cache and // start loading them into LyX. - PendingMap::const_iterator it = git->second.snippets.begin(); - PendingMap::const_iterator end = git->second.snippets.end(); + vector::const_iterator it = git->second.snippets.begin(); + vector::const_iterator end = git->second.snippets.end(); int metrics_counter = 0; for (; it != end; ++it) { @@ -469,32 +510,14 @@ void PreviewLoader::Impl::dumpPreamble(ostream & os) const } -namespace { - -typedef std::pair StrPair; - -struct CompSecond { - bool operator()(StrPair const & lhs, StrPair const & rhs) - { - return lhs.second < rhs.second; - } -}; - -} // namespace anon - - -void PreviewLoader::Impl::dumpData(ostream & os) const +void PreviewLoader::Impl::dumpData(ostream & os, + vector const & vec) const { - if (pending_.empty()) + if (vec.empty()) return; - // Sorting by image filename ensures that the snippets are put into - // the LaTeX file in the expected order. - std::vector vec(pending_.begin(), pending_.end()); - std::sort(vec.begin(), vec.end(), CompSecond()); - - std::vector::const_iterator it = vec.begin(); - std::vector::const_iterator end = vec.end(); + vector::const_iterator it = vec.begin(); + vector::const_iterator end = vec.end(); for (; it != end; ++it) { os << "\\begin{preview}\n"