18#include "filemanager.h"
22#include <QVersionNumber>
24#include "fileformat.h"
26#include "layercamera.h"
31 srand(
static_cast<uint
>(time(
nullptr)));
37 dd <<
"\n[Project LOAD diagnostics]\n";
41 handleOpenProjectError(Status::FILE_NOT_FOUND, dd);
47 std::unique_ptr<Object> obj(
new Object);
48 obj->setFilePath(sFileName);
49 obj->createWorkingDir();
55 bool isArchive = isArchiveFormat(sFileName);
57 QString fileFormat =
"Project format: %1";
60 dd << fileFormat.
arg(
".pcl");
64 dd <<
"Failed to copy main xml file";
66 Status st =
copyDir(sFileName +
"." + PFF_OLD_DATA_DIR, strDataFolder);
69 dd.collect(st.details());
74 QString workingDirPath = obj->workingDir();
76 dd <<
QString(
"Working dir: %1").
arg(workingDirPath);
77 dd << fileFormat.
arg(
".pclx");
79 Status sanityCheck = MiniZ::sanityCheck(sFileName);
82 if (!sanityCheck.ok()) {
83 dd.collect(sanityCheck.details());
84 dd <<
"\nError: Unable to extract project, miniz sanity check failed.";
85 handleOpenProjectError(Status::ERROR_INVALID_XML_FILE, dd);
88 Status unzipStatus = unzip(sFileName, workingDirPath);
89 dd.collect(unzipStatus.details());
91 if(unzipStatus.ok()) {
92 dd <<
QString(
"Unzipped at: %1 ").
arg(workingDirPath);
94 dd <<
QString(
"Error: Unzipping failed: %1 ").
arg(workingDirPath);
95 handleOpenProjectError(Status::ERROR_INVALID_XML_FILE, dd);
101 obj->setDataDir(strDataFolder);
102 obj->setMainXMLFile(strMainXMLFile);
105 mMaxProgressValue = totalFileCount;
106 emit progressRangeChanged(mMaxProgressValue);
108 QFile file(strMainXMLFile);
111 dd <<
"Error: No main XML exists!";
112 handleOpenProjectError(Status::ERROR_INVALID_XML_FILE, dd);
117 dd <<
"Error: Main XML file is read only!";
118 handleOpenProjectError(Status::ERROR_FILE_CANNOT_OPEN, dd);
122 dd <<
"Main XML exists: Yes";
127 FILEMANAGER_LOG(
"Couldn't open the main XML file");
128 dd <<
"Error: Unable to parse or open the main XML file";
129 handleOpenProjectError(Status::ERROR_INVALID_XML_FILE, dd);
134 if (!(type.
name() ==
"PencilDocument" || type.
name() ==
"MyObject"))
136 FILEMANAGER_LOG(
"Invalid main XML doctype");
138 handleOpenProjectError(Status::ERROR_INVALID_PENCIL_FILE, dd);
145 dd <<
"Error: Main XML root node is null";
146 handleOpenProjectError(Status::ERROR_INVALID_PENCIL_FILE, dd);
150 loadPalette(obj.get());
154 if (root.
tagName() ==
"document")
156 ok = loadObject(obj.get(), root);
158 else if (root.
tagName() ==
"object" || root.
tagName() ==
"MyOject")
160 ok = loadObjectOldWay(obj.get(), root);
166 dd <<
"Error: Issue occurred during object loading";
167 handleOpenProjectError(Status::ERROR_INVALID_PENCIL_FILE, dd);
171 verifyObject(obj.get());
173 return obj.release();
191 if (element.
tagName() ==
"object")
193 ok =
object->loadXML(element, [
this]{ progressForward(); });
194 if (!ok) FILEMANAGER_LOG(
"Failed to Load object");
197 else if (element.
tagName() ==
"editor" || element.
tagName() ==
"projectdata")
199 object->setData(loadProjectData(element));
201 else if (element.
tagName() ==
"version")
206 if (!fileVersion.
isNull())
208 if (appVersion < fileVersion)
210 qWarning() <<
"You are opening a newer project file in an older version of Pencil2D!";
224 return object->loadXML(root, [
this] { progressForward(); });
227bool FileManager::isArchiveFormat(
const QString& fileName)
const
238 dd <<
"\n[Project SAVE diagnostics]\n";
239 dd << (
"file name:" + sFileName);
241 if (
object ==
nullptr)
243 dd <<
"Error: Object parameter is null";
244 return Status(Status::INVALID_ARGUMENT, dd);
247 dd <<
"Error: File name is empty, unable to save.";
248 return Status(Status::INVALID_ARGUMENT, dd,
249 tr(
"Invalid Save Path"),
250 tr(
"The path is empty."));
253 const int totalCount =
object->totalKeyFrameCount();
254 mMaxProgressValue = totalCount + 5;
255 emit progressRangeChanged(mMaxProgressValue);
260 if (fileInfo.isDir())
262 dd <<
"Error: File name must point to a file, not a directory.";
263 return Status(Status::INVALID_ARGUMENT, dd,
264 tr(
"Invalid Save Path"),
265 tr(
"The path (\"%1\") points to a directory.").arg(fileInfo.absoluteFilePath()));
267 QFileInfo parentDirInfo(fileInfo.dir().absolutePath());
268 if (!parentDirInfo.exists())
270 dd <<
QString(
"Error: The parent directory of %1 does not exist").
arg(sFileName);
271 return Status(Status::INVALID_ARGUMENT, dd,
272 tr(
"Invalid Save Path"),
273 tr(
"The directory (\"%1\") does not exist.").arg(parentDirInfo.absoluteFilePath()));
275 if ((fileInfo.exists() && !fileInfo.isWritable()) || !parentDirInfo.isWritable())
277 dd <<
"Error: File name points to a location that is not writable";
278 return Status(Status::INVALID_ARGUMENT, dd,
279 tr(
"Invalid Save Path"),
280 tr(
"The path (\"%1\") is not writable.").arg(fileInfo.absoluteFilePath()));
288 bool isArchive = isArchiveFormat(sFileName);
291 dd << fileFormat.
arg(
".pcl");
293 sMainXMLFile = sFileName;
294 sDataFolder = sMainXMLFile +
"." + PFF_OLD_DATA_DIR;
298 sTempWorkingFolder =
object->workingDir();
300 dd <<
QString(
"Working dir: %1").
arg(sTempWorkingFolder);
301 dd << fileFormat.
arg(
".pclx");
303 Q_ASSERT(
QDir(sTempWorkingFolder).exists());
305 sMainXMLFile =
QDir(sTempWorkingFolder).
filePath(PFF_XML_FILE_NAME);
306 sDataFolder =
QDir(sTempWorkingFolder).
filePath(PFF_OLD_DATA_DIR);
310 if (!dataInfo.exists())
312 QDir dir(sDataFolder);
314 if (!dir.mkpath(sDataFolder))
316 dd <<
QString(
"Error: Unable to create data directory, tried to save to: %1").
arg(dir.absolutePath());
317 return Status(Status::FAIL, dd,
318 tr(
"Cannot Create Data Directory"),
319 tr(
"Failed to create directory \"%1\". Please make sure you have sufficient permissions.").arg(sDataFolder));
322 if (!dataInfo.isDir())
324 dd <<
QString(
"Error: Expected data to be a directory but found %1 instead").
arg(dataInfo.absoluteFilePath());
325 return Status(Status::FAIL,
327 tr(
"Cannot Create Data Directory"),
328 tr(
"\"%1\" is a file. Please delete the file and try again.").arg(dataInfo.absoluteFilePath()));
332 Status stKeyFrames = writeKeyFrameFiles(
object, sDataFolder, filesToZip);
333 dd.collect(stKeyFrames.details());
335 Status stMainXml = writeMainXml(
object, sMainXMLFile, filesToZip);
336 dd.collect(stMainXml.details());
338 Status stPalette = writePalette(
object, sDataFolder, filesToZip);
339 dd.collect(stPalette.details());
341 const bool saveOk = stKeyFrames.ok() && stMainXml.ok() && stPalette.ok();
347 dd <<
"\n[Archiving diagnostics]\n";
348 QString sBackupFile = backupPreviousFile(sFileName);
351 dd <<
QString(
"\nNote: A backup has been made here: %1").
arg(sBackupFile);
355 return Status(Status::FAIL, dd,
356 tr(
"Internal Error"),
357 tr(
"An internal error occurred. The project could not be saved."));
360 dd <<
"Miniz: Zipping...";
361 Status stMiniz = MiniZ::compressFolder(sFileName, sTempWorkingFolder, filesToZip,
"application/x-pencil2d-pclx");
364 dd.collect(stMiniz.details());
365 dd <<
"\nError: Miniz failed to zip project";
366 return Status(Status::ERROR_MINIZ_FAIL, dd,
368 tr(
"An internal error occurred. The project may not have been saved successfully."));
370 dd <<
"Miniz: Zip file saved successfully";
371 Q_ASSERT(stMiniz.ok());
374 dd <<
"Project saved successfully, deleting backup";
375 deleteBackupFile(sBackupFile);
383 return Status(Status::FAIL, dd,
384 tr(
"Internal Error"),
385 tr(
"An internal error occurred. The project may not have been saved successfully."));
391Status FileManager::writeToWorkingFolder(
const Object*
object)
397 const QString dataFolder =
object->dataDir();
398 const QString mainXml =
object->mainXMLFile();
400 Status stKeyFrames = writeKeyFrameFiles(
object, dataFolder, filesWritten);
401 dd.collect(stKeyFrames.details());
403 Status stMainXml = writeMainXml(
object, mainXml, filesWritten);
404 dd.collect(stMainXml.details());
406 Status stPalette = writePalette(
object, dataFolder, filesWritten);
407 dd.collect(stPalette.details());
409 const bool saveOk = stKeyFrames.ok() && stMainXml.ok() && stPalette.ok();
410 const auto errorCode = (saveOk) ? Status::OK : Status::FAIL;
411 return Status(errorCode, dd);
432 extractProjectData(element, data);
445 currentFrameTag.
setAttribute(
"value", data->getCurrentFrame());
450 QColor color = data->getCurrentColor();
459 currentLayerTag.
setAttribute(
"value", data->getCurrentLayer());
480 tagIsLoop.
setAttribute(
"value", data->isLooping() ?
"true" :
"false");
484 tagRangedPlayback.
setAttribute(
"value", data->isRangedPlayback() ?
"true" :
"false");
488 tagMarkInFrame.
setAttribute(
"value", data->getMarkInFrameNumber());
492 tagMarkOutFrame.
setAttribute(
"value", data->getMarkOutFrameNumber());
501 if (strName ==
"currentFrame")
505 else if (strName ==
"currentColor")
512 data.setCurrentColor(
QColor(r, g, b, a));
514 else if (strName ==
"currentLayer")
518 else if (strName ==
"currentView")
527 data.setCurrentView(
QTransform(m11, m12, m21, m22, dx, dy));
529 else if (strName ==
"fps" || strName ==
"currentFps")
533 else if (strName ==
"isLoop")
535 data.setLooping(element.
attribute(
"value",
"false") ==
"true");
537 else if (strName ==
"isRangedPlayback")
539 data.setRangedPlayback((element.
attribute(
"value",
"false") ==
"true"));
541 else if (strName ==
"markInFrame")
543 data.setMarkInFrameNumber(element.
attribute(
"value",
"0").
toInt());
545 else if (strName ==
"markOutFrame")
547 data.setMarkOutFrameNumber(element.
attribute(
"value",
"15").
toInt());
551void FileManager::handleOpenProjectError(Status::ErrorCode error,
const DebugDetails& dd)
553 QString title =
tr(
"Could not open file");
556 "<li><a href=\"https://discuss.pencil2d.org/c/bugs\">Pencil2D Forum</a></li>"
557 "<li><a href=\"https://github.com/pencil2d/pencil/issues/new\">Github</a></li>"
558 "<li><a href=\"https://discord.gg/8FxdV2g\">Discord<\a></li>"
561 if (error == Status::FILE_NOT_FOUND)
563 errorDesc =
tr(
"The file does not exist, so we are unable to open it."
564 "Please check to make sure the path is correct and try again.");
566 else if (error == Status::ERROR_FILE_CANNOT_OPEN)
568 errorDesc =
tr(
"No permission to read the file. "
569 "Please check you have read permissions for this file and try again.");
574 errorDesc =
tr(
"There was an error processing your file. "
575 "This usually means that your project has been at least partially corrupted. "
576 "Try again with a newer version of Pencil2D, "
577 "or try to use a backup file if you have one. "
578 "If you contact us through one of our official channels we may be able to help you."
579 "For reporting issues, the best places to reach us are:");
582 mError =
Status(error, dd, title, errorDesc + contactLinks);
583 removePFFTmpDirectory(mstrLastTempFolder);
586int FileManager::countExistingBackups(
const QString& fileName)
const
589 QDir directory(fileInfo.absoluteDir());
590 const QString& baseFileName = fileInfo.completeBaseName();
593 for (
const QFileInfo &dirFileInfo : directory.entryInfoList(QDir::Filter::Files)) {
594 QString searchFileAbsPath = dirFileInfo.absoluteFilePath();
595 QString searchFileName = dirFileInfo.baseName();
597 bool sameBaseName = baseFileName.
compare(searchFileName) == 0;
598 if (sameBaseName && searchFileAbsPath.
contains(PFF_BACKUP_IDENTIFIER)) {
612 QString baseName = fileInfo.completeBaseName();
614 int backupCount = countExistingBackups(fileName) + 1;
617 QString sBackupFile = baseName +
"." + PFF_BACKUP_IDENTIFIER + countStr +
"." + fileInfo.suffix();
620 bool ok =
QFile::copy(fileInfo.absoluteFilePath(), sBackupFileFullPath);
623 FILEMANAGER_LOG(
"Cannot backup the previous file");
626 return sBackupFileFullPath;
629void FileManager::deleteBackupFile(
const QString& fileName)
637void FileManager::progressForward()
640 emit progressChanged(mCurrentProgress);
643bool FileManager::loadPalette(
Object* obj)
645 FILEMANAGER_LOG(
"Load Palette..");
647 QString paletteFilePath = validateDataPath(PFF_PALETTE_FILE, obj->dataDir());
648 if (paletteFilePath.
isEmpty() || !obj->importPalette(paletteFilePath))
650 obj->loadDefaultPalette();
658 dd <<
"\n[Keyframes WRITE diagnostics]\n";
660 const int numLayers =
object->getLayerCount();
661 dd <<
QString(
"Total layer count: %1").
arg(numLayers);
663 for (
int i = 0; i < numLayers; ++i)
665 Layer* layer =
object->getLayer(i);
666 layer->presave(dataFolder);
669 bool saveLayersOK =
true;
670 for (
int i = 0; i < numLayers; ++i)
672 Layer* layer =
object->getLayer(i);
674 dd <<
QString(
"Layer[%1] = [id=%2, type=%3, name=%4]").
arg(i).
arg(layer->id()).
arg(layer->type()).
arg(layer->name());
676 Status st = layer->save(dataFolder, filesFlushed, [
this] { progressForward(); });
679 saveLayersOK =
false;
680 dd.collect(st.details());
681 dd <<
QString(
"\nError: Failed to save Layer[%1] %2").
arg(i).
arg(layer->name());
687 auto errorCode = (saveLayersOK) ? Status::OK : Status::FAIL;
690 dd <<
"\nAll Layers saved";
692 dd <<
"\nError: Unable to save all layers";
695 return Status(errorCode, dd);
701 dd <<
"\n[XML WRITE diagnostics]\n";
703 QFile file(mainXmlPath);
706 dd <<
QString(
"Error: Failed to open Main XML at: %1, \nReason: %2").
arg(mainXmlPath).
arg(file.errorString());
707 return Status(Status::ERROR_FILE_CANNOT_OPEN, dd);
722 QDomElement projDataXml = saveProjectData(object->data(), xmlDoc);
726 QDomElement objectElement =
object->saveXML(xmlDoc);
734 dd <<
"Writing main xml file...";
736 const int indentSize = 2;
739 xmlDoc.
save(out, indentSize);
742 dd <<
"Done writing main xml file: " << mainXmlPath;
744 filesWritten.
append(mainXmlPath);
745 return Status(Status::OK, dd);
750 const QString paletteFile =
object->savePalette(dataFolder);
754 dd <<
"\nError: Failed to save palette";
755 return Status(Status::FAIL, dd);
757 filesWritten.
append(paletteFile);
776 if (!src.
exists())
return Status::FILE_NOT_FOUND;
783 dd << (
"Failed to create target directory: " + dst.
absolutePath());
784 return Status(Status::FAIL, dd);
793 dd.collect(st.details());
802 dd <<
"Failed to copy file"
803 << (
"Source path: " + src.
filePath(fileName))
804 << (
"Destination path: " + dst.
filePath(fileName));
814 return Status(Status::FAIL, dd);
821 removePFFTmpDirectory(strUnzipTarget);
823 Status s = MiniZ::uncompressFolder(strZipFile, strUnzipTarget);
826 mstrLastTempFolder = strUnzipTarget;
833 if (!fileInfo.exists())
845 int curLayer = obj->data()->getCurrentLayer();
846 int maxLayer = obj->getLayerCount();
847 if (curLayer >= maxLayer)
849 obj->data()->setCurrentLayer(maxLayer - 1);
853 std::vector<LayerCamera*> camLayers = obj->getLayersByType<
LayerCamera>();
854 if (camLayers.empty())
856 obj->addNewCameraLayer();
864 bool folderExists = pencil2DTempDir.
cd(
"Pencil2D");
870 const QStringList nameFilter(
"*_" PFF_TMP_DECOMPRESS_EXT
"_*");
874 for (
const QString& path : entries)
877 if (isProjectRecoverable(fullPath))
879 qDebug() <<
"Found debris at" << fullPath;
880 recoverables.
append(fullPath);
886bool FileManager::isProjectRecoverable(
const QString& projectFolder)
888 QDir dir(projectFolder);
889 if (!dir.exists()) {
return false; }
892 if (!dir.exists(
"data")) {
return false; }
894 bool ok = dir.cd(
"data");
898 nameFiler <<
"*.png" <<
"*.vec" <<
"*.xml";
901 return (entries.
size() > 0);
904Object* FileManager::recoverUnsavedProject(
QString intermeidatePath)
906 qDebug() <<
"TODO: recover project" << intermeidatePath;
908 QDir projectDir(intermeidatePath);
909 const QString mainXMLPath = projectDir.filePath(PFF_XML_FILE_NAME);
910 const QString dataFolder = projectDir.filePath(PFF_DATA_DIR);
912 std::unique_ptr<Object> object(
new Object);
913 object->setWorkingDir(intermeidatePath);
914 object->setMainXMLFile(mainXMLPath);
915 object->setDataDir(dataFolder);
917 Status st = recoverObject(
object.get());
924 return object.release();
930 bool mainXmlOK =
true;
932 QFile file(object->mainXMLFile());
933 mainXmlOK &= file.exists();
941 mainXmlOK &= (type.
name() ==
"PencilDocument" || type.
name() ==
"MyObject");
944 mainXmlOK &= (!root.
isNull());
947 mainXmlOK &= (!objectTag.
isNull());
949 if (mainXmlOK ==
false)
955 QFile file(object->mainXMLFile());
963 bool ok = loadObject(
object, root);
964 verifyObject(
object);
966 return ok ? Status::OK : Status::FAIL;
972 QDir dataDir(object->dataDir());
975 nameFiler <<
"*.png" <<
"*.vec";
981 for (
const QString& s : entries)
983 int layerIndex = layerIndexFromFilename(s);
986 keyFrameGroups[layerIndex].append(s);
991 const QString mainXMLPath =
object->mainXMLFile();
992 QFile file(mainXMLPath);
995 return Status::ERROR_FILE_CANNOT_OPEN;
1005 QDomElement projDataXml = saveProjectData(object->data(), xmlDoc);
1012 for (
const int layerIndex : keyFrameGroups.
keys())
1019 xmlDoc.
save(fout, 2);
1034 const int layerIndex,
1037 Q_ASSERT(frames.
length() > 0);
1039 Layer::LAYER_TYPE type = frames[0].
endsWith(
".png") ? Layer::BITMAP : Layer::VECTOR;
1043 elemLayer.
setAttribute(
"name", recoverLayerName(type, layerIndex));
1048 for (
const QString& s : frames)
1050 const int framePos = framePosFromFilename(s);
1051 if (framePos < 0) {
continue; }
1057 if (type == Layer::BITMAP)
1069QString FileManager::recoverLayerName(Layer::LAYER_TYPE type,
int index)
1074 return tr(
"Bitmap Layer %1").
arg(index);
1076 return tr(
"Vector Layer %1").
arg(index);
1078 return tr(
"Sound Layer %1").
arg(index);
1085int FileManager::layerIndexFromFilename(
const QString& filename)
1088 if (tokens.
length() >= 3)
1090 return tokens[0].toInt();
1095int FileManager::framePosFromFilename(
const QString& filename)
1098 if (tokens.
length() >= 3)
1100 return tokens[1].toInt();
Status rebuildMainXML(Object *object)
Create a new main.xml based on the png/vec filenames left in the data folder.
Status copyDir(const QDir src, const QDir dst)
Copy a directory to another directory recursively (depth-first).
Status rebuildLayerXmlTag(QDomDocument &doc, QDomElement &elemObject, const int layerIndex, const QStringList &frames)
Rebuild a layer xml tag.
QString absolutePath() const const
bool cd(const QString &dirName)
QStringList entryList(QDir::Filters filters, QDir::SortFlags sort) const const
bool exists() const const
QString filePath(const QString &fileName) const const
bool mkpath(const QString &dirPath) const const
QDomElement createElement(const QString &tagName)
QDomProcessingInstruction createProcessingInstruction(const QString &target, const QString &data)
QDomText createTextNode(const QString &value)
QDomDocumentType doctype() const const
QDomElement documentElement() const const
bool setContent(const QByteArray &data, bool namespaceProcessing, QString *errorMsg, int *errorLine, int *errorColumn)
QString name() const const
QString attribute(const QString &name, const QString &defValue) const const
void setAttribute(const QString &name, const QString &value)
QString tagName() const const
QString text() const const
QDomNode appendChild(const QDomNode &newChild)
QDomNode firstChild() const const
QDomElement firstChildElement(const QString &tagName) const const
bool isNull() const const
QDomNode nextSibling() const const
void save(QTextStream &stream, int indent, QDomNode::EncodingPolicy encodingPolicy) const const
QDomElement toElement() const const
bool copy(const QString &newName)
bool exists() const const
virtual bool open(QIODevice::OpenMode mode) override
void append(const T &value)
bool endsWith(const T &value) const const
QList< Key > keys() const const
const T value(const Key &key, const T &defaultValue) const const
QString tr(const char *sourceText, const char *disambiguation, int n)
QStringList split(const QString &sep, QString::SplitBehavior behavior, Qt::CaseSensitivity cs) const const
QString & append(QChar ch)
QString arg(qlonglong a, int fieldWidth, int base, QChar fillChar) const const
int compare(const QString &other, Qt::CaseSensitivity cs) const const
bool contains(QChar ch, Qt::CaseSensitivity cs) const const
bool isEmpty() const const
QString number(int n, int base)
double toDouble(bool *ok) const const
int toInt(bool *ok, int base) const const
QVersionNumber fromString(const QString &string, int *suffixIndex)
bool isNull() const const