#include "converter.h"
+#include "ConverterCache.h"
#include "buffer.h"
#include "buffer_funcs.h"
#include "bufferparams.h"
#include "support/path.h"
#include "support/systemcall.h"
-using lyx::support::addName;
-using lyx::support::bformat;
-using lyx::support::changeExtension;
-using lyx::support::compare_ascii_no_case;
-using lyx::support::contains;
-using lyx::support::dirList;
-using lyx::support::getExtension;
-using lyx::support::isFileReadable;
-using lyx::support::libFileSearch;
-using lyx::support::libScriptSearch;
-using lyx::support::makeRelPath;
-using lyx::support::onlyFilename;
-using lyx::support::onlyPath;
-using lyx::support::Path;
-using lyx::support::prefixIs;
-using lyx::support::quoteName;
-using lyx::support::split;
-using lyx::support::subst;
-using lyx::support::Systemcall;
+
+namespace lyx {
+
+using support::addName;
+using support::bformat;
+using support::changeExtension;
+using support::compare_ascii_no_case;
+using support::contains;
+using support::dirList;
+using support::FileName;
+using support::getExtension;
+using support::isFileReadable;
+using support::libFileSearch;
+using support::libScriptSearch;
+using support::makeAbsPath;
+using support::makeRelPath;
+using support::onlyFilename;
+using support::onlyPath;
+using support::prefixIs;
+using support::quoteName;
+using support::removeExtension;
+using support::split;
+using support::subst;
+using support::Systemcall;
using std::endl;
using std::find_if;
using std::vector;
using std::distance;
+namespace Alert = lyx::frontend::Alert;
+
namespace {
bool Converters::convert(Buffer const * buffer,
- string const & from_file, string const & to_file_base,
- string const & from_format, string const & to_format,
- string & to_file, ErrorList & errorList, bool try_default)
+ FileName const & from_file, FileName const & to_file,
+ FileName const & orig_from,
+ string const & from_format, string const & to_format,
+ ErrorList & errorList, int conversionflags)
{
- string const to_ext = formats.extension(to_format);
- to_file = changeExtension(to_file_base, to_ext);
-
if (from_format == to_format)
return move(from_format, from_file, to_file, false);
+ if ((conversionflags & try_cache) &&
+ ConverterCache::get().inCache(orig_from, to_format))
+ return ConverterCache::get().copy(orig_from, to_format, to_file);
+
Graph::EdgePath edgepath = getPath(from_format, to_format);
if (edgepath.empty()) {
- if (try_default) {
+ if (conversionflags & try_default) {
// if no special converter defined, then we take the
// default one from ImageMagic.
string const from_ext = from_format.empty() ?
- getExtension(from_file) :
+ getExtension(from_file.absFilename()) :
formats.extension(from_format);
+ string const to_ext = formats.extension(to_format);
string const command =
- lyx::support::os::python() + ' ' +
- quoteName(libFileSearch("scripts", "convertDefault.py")) +
+ support::os::python() + ' ' +
+ quoteName(libFileSearch("scripts", "convertDefault.py").toFilesystemEncoding()) +
' ' +
- quoteName(from_ext + ':' + from_file) +
+ quoteName(from_ext + ':' + from_file.toFilesystemEncoding()) +
' ' +
- quoteName(to_ext + ':' + to_file);
- lyxerr[Debug::FILES]
+ quoteName(to_ext + ':' + to_file.toFilesystemEncoding());
+ LYXERR(Debug::FILES)
<< "No converter defined! "
"I use convertDefault.py:\n\t"
<< command << endl;
Systemcall one;
one.startscript(Systemcall::Wait, command);
if (isFileReadable(to_file)) {
+ if (conversionflags & try_cache)
+ ConverterCache::get().add(orig_from,
+ to_format, to_file);
return true;
}
}
- Alert::error(lyx::to_utf8(_("Cannot convert file")),
- bformat(lyx::to_utf8(_("No information for converting %1$s "
+ Alert::error(_("Cannot convert file"),
+ bformat(_("No information for converting %1$s "
"format files to %2$s.\n"
- "Define a convertor in the preferences.")),
- from_format, to_format));
+ "Define a converter in the preferences."),
+ from_ascii(from_format), from_ascii(to_format)));
return false;
}
- OutputParams runparams;
+
+ // buffer is only invalid for importing, and then runparams is not
+ // used anyway.
+ OutputParams runparams(buffer ? &buffer->params().encoding() : 0);
runparams.flavor = getFlavor(edgepath);
- string path = onlyPath(from_file);
- Path p(path);
+
+ // Some converters (e.g. lilypond) can only output files to the
+ // current directory, so we need to change the current directory.
+ // This has the added benefit that all other files that may be
+ // generated by the converter are deleted when LyX closes and do not
+ // clutter the real working directory.
+ string const path(onlyPath(from_file.absFilename()));
+ // Prevent the compiler from optimizing away p
+ FileName pp(path);
+ support::Path p(pp);
+
// empty the error list before any new conversion takes place.
errorList.clear();
bool run_latex = false;
- string from_base = changeExtension(from_file, "");
- string to_base = changeExtension(to_file, "");
- string infile;
- string outfile = from_file;
+ string from_base = changeExtension(from_file.absFilename(), "");
+ string to_base = changeExtension(to_file.absFilename(), "");
+ FileName infile;
+ FileName outfile = from_file;
for (Graph::EdgePath::const_iterator cit = edgepath.begin();
cit != edgepath.end(); ++cit) {
Converter const & conv = converterlist_[*cit];
bool dummy = conv.To->dummy() && conv.to != "program";
if (!dummy)
- lyxerr[Debug::FILES] << "Converting from "
+ LYXERR(Debug::FILES) << "Converting from "
<< conv.from << " to " << conv.to << endl;
infile = outfile;
- outfile = conv.result_dir.empty()
- ? changeExtension(from_file, conv.To->extension())
+ outfile = FileName(conv.result_dir.empty()
+ ? changeExtension(from_file.absFilename(), conv.To->extension())
: addName(subst(conv.result_dir,
token_base, from_base),
subst(conv.result_file,
- token_base, onlyFilename(from_base)));
+ token_base, onlyFilename(from_base))));
// if input and output files are equal, we use a
// temporary file as intermediary (JMarc)
- string real_outfile;
+ FileName real_outfile;
if (outfile == infile) {
real_outfile = infile;
- outfile = addName(buffer->temppath(), "tmpfile.out");
+ outfile = FileName(addName(buffer->temppath(), "tmpfile.out"));
}
if (conv.latex) {
run_latex = true;
string const command = subst(conv.command, token_from, "");
- lyxerr[Debug::FILES] << "Running " << command << endl;
+ LYXERR(Debug::FILES) << "Running " << command << endl;
if (!runLaTeX(*buffer, command, runparams, errorList))
return false;
} else {
if (conv.need_aux && !run_latex
&& !latex_command_.empty()) {
- lyxerr[Debug::FILES]
+ LYXERR(Debug::FILES)
<< "Running " << latex_command_
<< " to update aux file"<< endl;
runLaTeX(*buffer, latex_command_, runparams, errorList);
}
+ // FIXME UNICODE
string const infile2 = (conv.original_dir)
- ? infile : makeRelPath(infile, path);
+ ? infile.absFilename() : to_utf8(makeRelPath(from_utf8(infile.absFilename()),
+ from_utf8(path)));
string const outfile2 = (conv.original_dir)
- ? outfile : makeRelPath(outfile, path);
+ ? outfile.absFilename() : to_utf8(makeRelPath(from_utf8(outfile.absFilename()),
+ from_utf8(path)));
string command = conv.command;
command = subst(command, token_from, quoteName(infile2));
command = add_options(command,
dvipdfm_options(buffer->params()));
- lyxerr[Debug::FILES] << "Calling " << command << endl;
+ LYXERR(Debug::FILES) << "Calling " << command << endl;
if (buffer)
- buffer->message(lyx::to_utf8(_("Executing command: "))
- + command);
+ buffer->message(_("Executing command: ")
+ + from_utf8(command));
Systemcall::Starttype const type = (dummy)
? Systemcall::DontWait : Systemcall::Wait;
Systemcall one;
int res;
if (conv.original_dir) {
- Path p(buffer->filePath());
- res = one.startscript(type, command);
+ FileName path(buffer->filePath());
+ support::Path p(path);
+ res = one.startscript(type,
+ to_filesystem8bit(from_utf8(command)));
} else
- res = one.startscript(type, command);
+ res = one.startscript(type,
+ to_filesystem8bit(from_utf8(command)));
if (!real_outfile.empty()) {
- Mover const & mover = movers(conv.to);
+ Mover const & mover = getMover(conv.to);
if (!mover.rename(outfile, real_outfile))
res = -1;
else
- lyxerr[Debug::FILES]
+ LYXERR(Debug::FILES)
<< "renaming file " << outfile
<< " to " << real_outfile
<< endl;
string const command2 = script +
" < " + quoteName(infile2 + ".out") +
" > " + quoteName(logfile);
- one.startscript(Systemcall::Wait, command2);
- if (!scanLog(*buffer, command, logfile, errorList))
+ one.startscript(Systemcall::Wait,
+ to_filesystem8bit(from_utf8(command2)));
+ if (!scanLog(*buffer, command, makeAbsPath(logfile, path), errorList))
return false;
}
if (res) {
if (conv.to == "program") {
- Alert::error(lyx::to_utf8(_("Build errors")),
- lyx::to_utf8(_("There were errors during the build process.")));
+ Alert::error(_("Build errors"),
+ _("There were errors during the build process."));
} else {
// FIXME: this should go out of here. For example, here we cannot say if
// it is a document (.lyx) or something else. Same goes for elsewhere.
- Alert::error(lyx::to_utf8(_("Cannot convert file")),
- bformat(lyx::to_utf8(_("An error occurred whilst running %1$s")),
- command.substr(0, 50)));
+ Alert::error(_("Cannot convert file"),
+ bformat(_("An error occurred whilst running %1$s"),
+ from_utf8(command.substr(0, 50))));
}
return false;
}
return true;
if (!conv.result_dir.empty()) {
- to_file = addName(subst(conv.result_dir, token_base, to_base),
- subst(conv.result_file,
- token_base, onlyFilename(to_base)));
+ // The converter has put the file(s) in a directory.
+ // In this case we ignore the given to_file.
if (from_base != to_base) {
string const from = subst(conv.result_dir,
token_base, from_base);
string const to = subst(conv.result_dir,
token_base, to_base);
- Mover const & mover = movers(conv.from);
- if (!mover.rename(from, to)) {
- Alert::error(lyx::to_utf8(_("Cannot convert file")),
- bformat(lyx::to_utf8(_("Could not move a temporary file from %1$s to %2$s.")),
- from, to));
+ Mover const & mover = getMover(conv.from);
+ if (!mover.rename(FileName(from), FileName(to))) {
+ Alert::error(_("Cannot convert file"),
+ bformat(_("Could not move a temporary directory from %1$s to %2$s."),
+ from_utf8(from), from_utf8(to)));
return false;
}
}
return true;
- } else
+ } else {
+ if (conversionflags & try_cache)
+ ConverterCache::get().add(orig_from, to_format, outfile);
return move(conv.to, outfile, to_file, conv.latex);
+ }
}
bool Converters::move(string const & fmt,
- string const & from, string const & to, bool copy)
+ FileName const & from, FileName const & to, bool copy)
{
if (from == to)
return true;
bool no_errors = true;
- string const path = onlyPath(from);
- string const base = onlyFilename(changeExtension(from, ""));
- string const to_base = changeExtension(to, "");
- string const to_extension = getExtension(to);
-
- vector<string> files = dirList(onlyPath(from), getExtension(from));
- for (vector<string>::const_iterator it = files.begin();
- it != files.end(); ++it)
- if (prefixIs(*it, base)) {
- string const from2 = path + *it;
- string to2 = to_base + it->substr(base.length());
- to2 = changeExtension(to2, to_extension);
- lyxerr[Debug::FILES] << "moving " << from2
+ string const path = onlyPath(from.absFilename());
+ string const base = onlyFilename(removeExtension(from.absFilename()));
+ string const to_base = removeExtension(to.absFilename());
+ string const to_extension = getExtension(to.absFilename());
+
+ vector<FileName> const files = dirList(FileName(path),
+ getExtension(from.absFilename()));
+ for (vector<FileName>::const_iterator it = files.begin();
+ it != files.end(); ++it) {
+ string const from2 = it->absFilename();
+ string const file2 = onlyFilename(from2);
+ if (prefixIs(file2, base)) {
+ string const to2 = changeExtension(
+ to_base + file2.substr(base.length()),
+ to_extension);
+ LYXERR(Debug::FILES) << "moving " << from2
<< " to " << to2 << endl;
- Mover const & mover = movers(fmt);
+ Mover const & mover = getMover(fmt);
bool const moved = copy
- ? mover.copy(from2, to2)
- : mover.rename(from2, to2);
+ ? mover.copy(*it, FileName(to2))
+ : mover.rename(*it, FileName(to2));
if (!moved && no_errors) {
- Alert::error(lyx::to_utf8(_("Cannot convert file")),
+ Alert::error(_("Cannot convert file"),
bformat(copy ?
- lyx::to_utf8(_("Could not copy a temporary file from %1$s to %2$s.")) :
- lyx::to_utf8(_("Could not move a temporary file from %1$s to %2$s.")),
- from2, to2));
+ _("Could not copy a temporary file from %1$s to %2$s.") :
+ _("Could not move a temporary file from %1$s to %2$s."),
+ from_utf8(from2), from_utf8(to2)));
no_errors = false;
}
}
+ }
return no_errors;
}
-bool Converters::convert(Buffer const * buffer,
- string const & from_file, string const & to_file_base,
- string const & from_format, string const & to_format,
- ErrorList & errorList, bool try_default)
-{
- string to_file;
- return convert(buffer, from_file, to_file_base, from_format, to_format,
- to_file, errorList, try_default);
-}
-
-
bool Converters::formatIsUsed(string const & format)
{
ConverterList::const_iterator cit = converterlist_.begin();
bool Converters::scanLog(Buffer const & buffer, string const & /*command*/,
- string const & filename, ErrorList & errorList)
+ FileName const & filename, ErrorList & errorList)
{
- OutputParams runparams;
+ OutputParams runparams(0);
runparams.flavor = OutputParams::LATEX;
- LaTeX latex("", runparams, filename, "");
+ LaTeX latex("", runparams, filename);
TeXErrors terr;
int const result = latex.scanLogFile(terr);
namespace {
-class showMessage : public std::unary_function<string, void>, public boost::signals::trackable {
+class showMessage : public std::unary_function<docstring, void>, public boost::signals::trackable {
public:
showMessage(Buffer const & b) : buffer_(b) {};
- void operator()(string const & m) const
+ void operator()(docstring const & m) const
{
buffer_.message(m);
}
OutputParams const & runparams, ErrorList & errorList)
{
buffer.busy(true);
- buffer.message(lyx::to_utf8(_("Running LaTeX...")));
+ buffer.message(_("Running LaTeX..."));
runparams.document_language = buffer.params().language->babel();
// do the LaTeX run(s)
string const name = buffer.getLatexName();
- LaTeX latex(command, runparams, name, buffer.filePath());
+ LaTeX latex(command, runparams, FileName(makeAbsPath(name)));
TeXErrors terr;
showMessage show(buffer);
latex.message.connect(show);
// check return value from latex.run().
if ((result & LaTeX::NO_LOGFILE)) {
- string const str =
- bformat(lyx::to_utf8(_("LaTeX did not run successfully. "
+ docstring const str =
+ bformat(_("LaTeX did not run successfully. "
"Additionally, LyX could not locate "
- "the LaTeX log %1$s.")), name);
- Alert::error(lyx::to_utf8(_("LaTeX failed")), str);
+ "the LaTeX log %1$s."), from_utf8(name));
+ Alert::error(_("LaTeX failed"), str);
} else if (result & LaTeX::NO_OUTPUT) {
- Alert::warning(lyx::to_utf8(_("Output is empty")),
- lyx::to_utf8(_("An empty output file was generated.")));
+ Alert::warning(_("Output is empty"),
+ _("An empty output file was generated."));
}
formats.getNumber(to));
}
-
-/// The global instance
-Converters converters;
-
-// The global copy after reading lyxrc.defaults
-Converters system_converters;
+} // namespace lyx