mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-03 08:28:25 +00:00
Update paths of included files when saving to a different folder.
Fixes #9528 and #5115.
This commit is contained in:
parent
23297a3108
commit
8c351e32be
@ -249,6 +249,9 @@ public:
|
|||||||
/// map from children inclusion positions to their scope and their buffer
|
/// map from children inclusion positions to their scope and their buffer
|
||||||
PositionScopeBufferMap position_to_children;
|
PositionScopeBufferMap position_to_children;
|
||||||
|
|
||||||
|
/// Keeps track of old buffer filePath() for save-as operations
|
||||||
|
string old_position;
|
||||||
|
|
||||||
/// Container for all sort of Buffer dependant errors.
|
/// Container for all sort of Buffer dependant errors.
|
||||||
map<string, ErrorList> errorLists;
|
map<string, ErrorList> errorLists;
|
||||||
|
|
||||||
@ -4926,6 +4929,8 @@ bool Buffer::saveAs(FileName const & fn)
|
|||||||
FileName const old_name = fileName();
|
FileName const old_name = fileName();
|
||||||
FileName const old_auto = getAutosaveFileName();
|
FileName const old_auto = getAutosaveFileName();
|
||||||
bool const old_unnamed = isUnnamed();
|
bool const old_unnamed = isUnnamed();
|
||||||
|
bool success = true;
|
||||||
|
d->old_position = filePath();
|
||||||
|
|
||||||
setFileName(fn);
|
setFileName(fn);
|
||||||
markDirty();
|
markDirty();
|
||||||
@ -4943,22 +4948,19 @@ bool Buffer::saveAs(FileName const & fn)
|
|||||||
// are still valid.
|
// are still valid.
|
||||||
checkChildBuffers();
|
checkChildBuffers();
|
||||||
checkMasterBuffer();
|
checkMasterBuffer();
|
||||||
return true;
|
|
||||||
} else {
|
} else {
|
||||||
// save failed
|
// save failed
|
||||||
// reset the old filename and unnamed state
|
// reset the old filename and unnamed state
|
||||||
setFileName(old_name);
|
setFileName(old_name);
|
||||||
setUnnamed(old_unnamed);
|
setUnnamed(old_unnamed);
|
||||||
return false;
|
success = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
d->old_position.clear();
|
||||||
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// FIXME We could do better here, but it is complicated. What would be
|
|
||||||
// nice is to offer either (a) to save the child buffer to an appropriate
|
|
||||||
// location, so that it would "move with the master", or else (b) to update
|
|
||||||
// the InsetInclude so that it pointed to the same file. But (a) is a bit
|
|
||||||
// complicated, because the code for this lives in GuiView.
|
|
||||||
void Buffer::checkChildBuffers()
|
void Buffer::checkChildBuffers()
|
||||||
{
|
{
|
||||||
Impl::BufferPositionMap::iterator it = d->children_positions.begin();
|
Impl::BufferPositionMap::iterator it = d->children_positions.begin();
|
||||||
@ -4978,11 +4980,6 @@ void Buffer::checkChildBuffers()
|
|||||||
if (oldloc == newloc)
|
if (oldloc == newloc)
|
||||||
continue;
|
continue;
|
||||||
// the location of the child file is incorrect.
|
// the location of the child file is incorrect.
|
||||||
Alert::warning(_("Included File Invalid"),
|
|
||||||
bformat(_("Saving this document to a new location has made the file:\n"
|
|
||||||
" %1$s\n"
|
|
||||||
"inaccessible. You will need to update the included filename."),
|
|
||||||
from_utf8(oldloc)));
|
|
||||||
cbuf->setParent(0);
|
cbuf->setParent(0);
|
||||||
inset_inc->setChildBuffer(0);
|
inset_inc->setChildBuffer(0);
|
||||||
}
|
}
|
||||||
@ -5012,4 +5009,18 @@ void Buffer::checkMasterBuffer()
|
|||||||
setParent(0);
|
setParent(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
string Buffer::includedFilePath(string const & name) const
|
||||||
|
{
|
||||||
|
if (d->old_position.empty() || d->old_position == filePath())
|
||||||
|
return name;
|
||||||
|
|
||||||
|
if (FileName::isAbsolute(name))
|
||||||
|
return to_utf8(makeRelPath(from_utf8(name), from_utf8(filePath())));
|
||||||
|
|
||||||
|
// old_position already contains a trailing path separator
|
||||||
|
string const cleanpath = FileName(d->old_position + name).realPath();
|
||||||
|
return to_utf8(makeRelPath(from_utf8(cleanpath), from_utf8(filePath())));
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace lyx
|
} // namespace lyx
|
||||||
|
@ -710,6 +710,12 @@ public:
|
|||||||
///
|
///
|
||||||
void checkMasterBuffer();
|
void checkMasterBuffer();
|
||||||
|
|
||||||
|
/// If the document is being saved to a new location, return the
|
||||||
|
/// updated path of an included file relative to the new buffer path
|
||||||
|
/// if possible, otherwise return its absolute path.
|
||||||
|
/// In all other cases, this is a no-op and name is returned unchanged.
|
||||||
|
std::string includedFilePath(std::string const & name) const;
|
||||||
|
|
||||||
/// compute statistics between \p from and \p to
|
/// compute statistics between \p from and \p to
|
||||||
/// \p from initial position
|
/// \p from initial position
|
||||||
/// \p to points to the end position
|
/// \p to points to the end position
|
||||||
|
@ -1025,6 +1025,12 @@ docstring InsetBibtex::xhtml(XHTMLStream & xs, OutputParams const &) const
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void InsetBibtex::write(ostream & os) const
|
||||||
|
{
|
||||||
|
params().Write(os, &buffer());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
string InsetBibtex::contextMenuName() const
|
string InsetBibtex::contextMenuName() const
|
||||||
{
|
{
|
||||||
return "context-bibtex";
|
return "context-bibtex";
|
||||||
|
@ -38,6 +38,8 @@ public:
|
|||||||
bool addDatabase(docstring const &);
|
bool addDatabase(docstring const &);
|
||||||
///
|
///
|
||||||
bool delDatabase(docstring const &);
|
bool delDatabase(docstring const &);
|
||||||
|
///
|
||||||
|
void write(std::ostream &) const;
|
||||||
|
|
||||||
/// \name Public functions inherited from Inset class
|
/// \name Public functions inherited from Inset class
|
||||||
//@{
|
//@{
|
||||||
|
@ -28,6 +28,7 @@
|
|||||||
#include "InsetRef.h"
|
#include "InsetRef.h"
|
||||||
#include "InsetTOC.h"
|
#include "InsetTOC.h"
|
||||||
|
|
||||||
|
#include "Buffer.h"
|
||||||
#include "Encoding.h"
|
#include "Encoding.h"
|
||||||
#include "Lexer.h"
|
#include "Lexer.h"
|
||||||
#include "OutputParams.h"
|
#include "OutputParams.h"
|
||||||
@ -320,6 +321,12 @@ void InsetCommandParams::read(Lexer & lex)
|
|||||||
|
|
||||||
|
|
||||||
void InsetCommandParams::write(ostream & os) const
|
void InsetCommandParams::write(ostream & os) const
|
||||||
|
{
|
||||||
|
Write(os, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void InsetCommandParams::Write(ostream & os, Buffer const * buffer) const
|
||||||
{
|
{
|
||||||
os << "CommandInset " << insetType() << '\n';
|
os << "CommandInset " << insetType() << '\n';
|
||||||
os << "LatexCommand " << cmdName_ << '\n';
|
os << "LatexCommand " << cmdName_ << '\n';
|
||||||
@ -328,12 +335,27 @@ void InsetCommandParams::write(ostream & os) const
|
|||||||
ParamInfo::const_iterator it = info_.begin();
|
ParamInfo::const_iterator it = info_.begin();
|
||||||
ParamInfo::const_iterator end = info_.end();
|
ParamInfo::const_iterator end = info_.end();
|
||||||
for (; it != end; ++it) {
|
for (; it != end; ++it) {
|
||||||
std::string const & name = it->name();
|
string const & name = it->name();
|
||||||
docstring const & data = (*this)[name];
|
string data = to_utf8((*this)[name]);
|
||||||
if (!data.empty()) {
|
if (!data.empty()) {
|
||||||
// FIXME UNICODE
|
// Adjust path of files if document was moved
|
||||||
|
if (buffer && name == "filename") {
|
||||||
|
data = buffer->includedFilePath(data);
|
||||||
|
} else if (buffer && name == "bibfiles") {
|
||||||
|
int i = 0;
|
||||||
|
string newdata;
|
||||||
|
string bib = token(data, ',', i);
|
||||||
|
while (!bib.empty()) {
|
||||||
|
bib = buffer->includedFilePath(bib);
|
||||||
|
if (!newdata.empty())
|
||||||
|
newdata.append(1, ',');
|
||||||
|
newdata.append(bib);
|
||||||
|
bib = token(data, ',', ++i);
|
||||||
|
}
|
||||||
|
data = newdata;
|
||||||
|
}
|
||||||
os << name << ' '
|
os << name << ' '
|
||||||
<< Lexer::quoteString(to_utf8(data))
|
<< Lexer::quoteString(data)
|
||||||
<< '\n';
|
<< '\n';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -28,6 +28,7 @@
|
|||||||
namespace lyx {
|
namespace lyx {
|
||||||
|
|
||||||
class Lexer;
|
class Lexer;
|
||||||
|
class Buffer;
|
||||||
|
|
||||||
class ParamInfo {
|
class ParamInfo {
|
||||||
public:
|
public:
|
||||||
@ -117,6 +118,8 @@ public:
|
|||||||
/// Parse the command
|
/// Parse the command
|
||||||
///
|
///
|
||||||
void write(std::ostream &) const;
|
void write(std::ostream &) const;
|
||||||
|
///
|
||||||
|
void Write(std::ostream & os, Buffer const * buf) const;
|
||||||
/// Build the complete LaTeX command
|
/// Build the complete LaTeX command
|
||||||
docstring getCommand(OutputParams const &) const;
|
docstring getCommand(OutputParams const &) const;
|
||||||
/// Return the command name
|
/// Return the command name
|
||||||
|
@ -1029,6 +1029,12 @@ void InsetInclude::draw(PainterInfo & pi, int x, int y) const
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void InsetInclude::write(ostream & os) const
|
||||||
|
{
|
||||||
|
params().Write(os, &buffer());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
string InsetInclude::contextMenuName() const
|
string InsetInclude::contextMenuName() const
|
||||||
{
|
{
|
||||||
return "context-include";
|
return "context-include";
|
||||||
|
@ -64,6 +64,8 @@ public:
|
|||||||
|
|
||||||
///
|
///
|
||||||
void updateCommand();
|
void updateCommand();
|
||||||
|
///
|
||||||
|
void write(std::ostream &) const;
|
||||||
|
|
||||||
/// \name Public functions inherited from Inset class
|
/// \name Public functions inherited from Inset class
|
||||||
//@{
|
//@{
|
||||||
|
Loading…
Reference in New Issue
Block a user