You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
511 lines
19 KiB
C++
511 lines
19 KiB
C++
/*
|
|
* Copyright (c) Contributors to the Open 3D Engine Project. For complete copyright and license terms please see the LICENSE at the root of this distribution.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0 OR MIT
|
|
*
|
|
*/
|
|
|
|
#include <ProjectUtils.h>
|
|
#include <ProjectManagerDefs.h>
|
|
#include <PythonBindingsInterface.h>
|
|
|
|
#include <QFileDialog>
|
|
#include <QDir>
|
|
#include <QtMath>
|
|
#include <QMessageBox>
|
|
#include <QFileInfo>
|
|
#include <QProcess>
|
|
#include <QProcessEnvironment>
|
|
#include <QGuiApplication>
|
|
#include <QProgressDialog>
|
|
#include <QSpacerItem>
|
|
#include <QGridLayout>
|
|
|
|
namespace O3DE::ProjectManager
|
|
{
|
|
namespace ProjectUtils
|
|
{
|
|
static bool WarnDirectoryOverwrite(const QString& path, QWidget* parent)
|
|
{
|
|
if (!QDir(path).isEmpty())
|
|
{
|
|
QMessageBox::StandardButton warningResult = QMessageBox::warning(
|
|
parent, QObject::tr("Overwrite Directory"),
|
|
QObject::tr("Directory is not empty! Are you sure you want to overwrite it?"), QMessageBox::No | QMessageBox::Yes);
|
|
|
|
if (warningResult != QMessageBox::Yes)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool IsDirectoryDescedent(const QString& possibleAncestorPath, const QString& possibleDecedentPath)
|
|
{
|
|
QDir ancestor(possibleAncestorPath);
|
|
QDir descendent(possibleDecedentPath);
|
|
|
|
do
|
|
{
|
|
if (ancestor == descendent)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
descendent.cdUp();
|
|
} while (!descendent.isRoot());
|
|
|
|
return false;
|
|
}
|
|
|
|
static bool SkipFilePaths(const QString& curPath, QStringList& skippedPaths, QStringList& deeperSkippedPaths)
|
|
{
|
|
bool skip = false;
|
|
for (const QString& skippedPath : skippedPaths)
|
|
{
|
|
QString nativeSkippedPath = QDir::toNativeSeparators(skippedPath);
|
|
QString firstSectionSkippedPath = nativeSkippedPath.section(QDir::separator(), 0, 0);
|
|
if (curPath == firstSectionSkippedPath)
|
|
{
|
|
// We are at the end of the path to skip, so skip it
|
|
if (nativeSkippedPath == firstSectionSkippedPath)
|
|
{
|
|
skippedPaths.removeAll(skippedPath);
|
|
skip = true;
|
|
break;
|
|
}
|
|
// Append the next section of the skipped path
|
|
else
|
|
{
|
|
deeperSkippedPaths.append(nativeSkippedPath.section(QDir::separator(), 1));
|
|
}
|
|
}
|
|
}
|
|
|
|
return skip;
|
|
}
|
|
|
|
typedef AZStd::function<void(/*fileCount=*/int, /*totalSizeInBytes=*/int)> StatusFunction;
|
|
static void RecursiveGetAllFiles(const QDir& directory, QStringList& skippedPaths, int& outFileCount, qint64& outTotalSizeInBytes, StatusFunction statusCallback)
|
|
{
|
|
const QStringList entries = directory.entryList(QDir::Dirs | QDir::Files | QDir::NoSymLinks | QDir::NoDotAndDotDot);
|
|
for (const QString& entryPath : entries)
|
|
{
|
|
const QString filePath = QDir::toNativeSeparators(QString("%1/%2").arg(directory.path()).arg(entryPath));
|
|
|
|
QStringList deeperSkippedPaths;
|
|
if (SkipFilePaths(entryPath, skippedPaths, deeperSkippedPaths))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
QFileInfo fileInfo(filePath);
|
|
|
|
if (fileInfo.isDir())
|
|
{
|
|
QDir subDirectory(filePath);
|
|
RecursiveGetAllFiles(subDirectory, deeperSkippedPaths, outFileCount, outTotalSizeInBytes, statusCallback);
|
|
}
|
|
else
|
|
{
|
|
++outFileCount;
|
|
outTotalSizeInBytes += fileInfo.size();
|
|
|
|
const int updateStatusEvery = 64;
|
|
if (outFileCount % updateStatusEvery == 0)
|
|
{
|
|
statusCallback(outFileCount, outTotalSizeInBytes);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static bool CopyDirectory(QProgressDialog* progressDialog,
|
|
const QString& origPath,
|
|
const QString& newPath,
|
|
QStringList& skippedPaths,
|
|
int filesToCopyCount,
|
|
int& outNumCopiedFiles,
|
|
qint64 totalSizeToCopy,
|
|
qint64& outCopiedFileSize,
|
|
bool& showIgnoreFileDialog)
|
|
{
|
|
QDir original(origPath);
|
|
if (!original.exists())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
for (const QString& directory : original.entryList(QDir::Dirs | QDir::NoDotAndDotDot))
|
|
{
|
|
if (progressDialog->wasCanceled())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
QStringList deeperSkippedPaths;
|
|
if (SkipFilePaths(directory, skippedPaths, deeperSkippedPaths))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
QString newDirectoryPath = newPath + QDir::separator() + directory;
|
|
original.mkpath(newDirectoryPath);
|
|
|
|
if (!CopyDirectory(progressDialog, origPath + QDir::separator() + directory, newDirectoryPath, deeperSkippedPaths,
|
|
filesToCopyCount, outNumCopiedFiles, totalSizeToCopy, outCopiedFileSize, showIgnoreFileDialog))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
QLocale locale;
|
|
const float progressDialogRangeHalf = qFabs(progressDialog->maximum() - progressDialog->minimum()) * 0.5f;
|
|
for (const QString& file : original.entryList(QDir::Files))
|
|
{
|
|
if (progressDialog->wasCanceled())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// Unused by this function but neccesary to pass in to SkipFilePaths
|
|
QStringList deeperSkippedPaths;
|
|
if (SkipFilePaths(file, skippedPaths, deeperSkippedPaths))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
// Progress window update
|
|
{
|
|
// Weight in the number of already copied files as well as the copied bytes to get a better progress indication
|
|
// for cases combining many small files and some really large files.
|
|
const float normalizedNumFiles = static_cast<float>(outNumCopiedFiles) / filesToCopyCount;
|
|
const float normalizedFileSize = static_cast<float>(outCopiedFileSize) / totalSizeToCopy;
|
|
const int progress = normalizedNumFiles * progressDialogRangeHalf + normalizedFileSize * progressDialogRangeHalf;
|
|
progressDialog->setValue(progress);
|
|
|
|
const QString copiedFileSizeString = locale.formattedDataSize(outCopiedFileSize);
|
|
const QString totalFileSizeString = locale.formattedDataSize(totalSizeToCopy);
|
|
progressDialog->setLabelText(QString("Coping file %1 of %2 (%3 of %4) ...").arg(QString::number(outNumCopiedFiles),
|
|
QString::number(filesToCopyCount),
|
|
copiedFileSizeString,
|
|
totalFileSizeString));
|
|
qApp->processEvents(QEventLoop::ExcludeUserInputEvents);
|
|
}
|
|
|
|
const QString toBeCopiedFilePath = origPath + QDir::separator() + file;
|
|
const QString copyToFilePath = newPath + QDir::separator() + file;
|
|
if (!QFile::copy(toBeCopiedFilePath, copyToFilePath))
|
|
{
|
|
// Let the user decide to ignore files that failed to copy or cancel the whole operation.
|
|
if (showIgnoreFileDialog)
|
|
{
|
|
QMessageBox ignoreFileMessageBox;
|
|
const QString text = QString("Cannot copy <b>%1</b>.<br><br>"
|
|
"Source: %2<br>"
|
|
"Destination: %3<br><br>"
|
|
"Press <b>Yes</b> to ignore the file, <b>YesToAll</b> to ignore all upcoming non-copyable files or "
|
|
"<b>Cancel</b> to abort duplicating the project.").arg(file, toBeCopiedFilePath, copyToFilePath);
|
|
|
|
ignoreFileMessageBox.setModal(true);
|
|
ignoreFileMessageBox.setWindowTitle("Cannot copy file");
|
|
ignoreFileMessageBox.setText(text);
|
|
ignoreFileMessageBox.setIcon(QMessageBox::Question);
|
|
ignoreFileMessageBox.setStandardButtons(QMessageBox::YesToAll | QMessageBox::Yes | QMessageBox::Cancel);
|
|
|
|
int ignoreFile = ignoreFileMessageBox.exec();
|
|
if (ignoreFile == QMessageBox::YesToAll)
|
|
{
|
|
showIgnoreFileDialog = false;
|
|
continue;
|
|
}
|
|
else if (ignoreFile == QMessageBox::Yes)
|
|
{
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
outNumCopiedFiles++;
|
|
|
|
QFileInfo fileInfo(toBeCopiedFilePath);
|
|
outCopiedFileSize += fileInfo.size();
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool ClearProjectBuildArtifactsAndCache(const QString& origPath, const QString& newPath, QWidget* parent)
|
|
{
|
|
QDir buildDirectory = QDir(newPath);
|
|
if ((!buildDirectory.cd(ProjectBuildDirectoryName) || !DeleteProjectFiles(buildDirectory.path(), true))
|
|
&& QDir(origPath).cd(ProjectBuildDirectoryName))
|
|
{
|
|
QMessageBox::warning(
|
|
parent,
|
|
QObject::tr("Clear Build Artifacts"),
|
|
QObject::tr("Build artifacts failed to delete for moved project. Please manually delete build directory at \"%1\"")
|
|
.arg(buildDirectory.path()),
|
|
QMessageBox::Close);
|
|
|
|
return false;
|
|
}
|
|
|
|
QDir cacheDirectory = QDir(newPath);
|
|
if ((!cacheDirectory.cd(ProjectCacheDirectoryName) || !DeleteProjectFiles(cacheDirectory.path(), true))
|
|
&& QDir(origPath).cd(ProjectCacheDirectoryName))
|
|
{
|
|
QMessageBox::warning(
|
|
parent,
|
|
QObject::tr("Clear Asset Cache"),
|
|
QObject::tr("Asset cache failed to delete for moved project. Please manually delete cache directory at \"%1\"")
|
|
.arg(cacheDirectory.path()),
|
|
QMessageBox::Close);
|
|
|
|
return false;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool AddProjectDialog(QWidget* parent)
|
|
{
|
|
QString path = QDir::toNativeSeparators(QFileDialog::getExistingDirectory(parent, QObject::tr("Select Project Directory")));
|
|
if (!path.isEmpty())
|
|
{
|
|
return RegisterProject(path);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool RegisterProject(const QString& path)
|
|
{
|
|
return PythonBindingsInterface::Get()->AddProject(path);
|
|
}
|
|
|
|
bool UnregisterProject(const QString& path)
|
|
{
|
|
return PythonBindingsInterface::Get()->RemoveProject(path);
|
|
}
|
|
|
|
bool CopyProjectDialog(const QString& origPath, ProjectInfo& newProjectInfo, QWidget* parent)
|
|
{
|
|
bool copyResult = false;
|
|
|
|
QDir parentOrigDir(origPath);
|
|
parentOrigDir.cdUp();
|
|
QString newPath = QDir::toNativeSeparators(
|
|
QFileDialog::getExistingDirectory(parent, QObject::tr("Select New Project Directory"), parentOrigDir.path()));
|
|
if (!newPath.isEmpty())
|
|
{
|
|
newProjectInfo.m_path = newPath;
|
|
|
|
if (!WarnDirectoryOverwrite(newPath, parent))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
copyResult = CopyProject(origPath, newPath, parent);
|
|
}
|
|
|
|
return copyResult;
|
|
}
|
|
|
|
bool CopyProject(const QString& origPath, const QString& newPath, QWidget* parent, bool skipRegister)
|
|
{
|
|
// Disallow copying from or into subdirectory
|
|
if (IsDirectoryDescedent(origPath, newPath) || IsDirectoryDescedent(newPath, origPath))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
int filesToCopyCount = 0;
|
|
qint64 totalSizeInBytes = 0;
|
|
QStringList skippedPaths
|
|
{
|
|
ProjectBuildDirectoryName,
|
|
ProjectCacheDirectoryName
|
|
};
|
|
|
|
QProgressDialog* progressDialog = new QProgressDialog(parent);
|
|
progressDialog->setAutoClose(true);
|
|
progressDialog->setValue(0);
|
|
progressDialog->setRange(0, 1000);
|
|
progressDialog->setModal(true);
|
|
progressDialog->setWindowTitle(QObject::tr("Copying project ..."));
|
|
progressDialog->show();
|
|
|
|
QLocale locale;
|
|
QStringList getFilesSkippedPaths(skippedPaths);
|
|
RecursiveGetAllFiles(origPath, getFilesSkippedPaths, filesToCopyCount, totalSizeInBytes, [=](int fileCount, int sizeInBytes)
|
|
{
|
|
// Create a human-readable version of the file size.
|
|
const QString fileSizeString = locale.formattedDataSize(sizeInBytes);
|
|
|
|
progressDialog->setLabelText(QString("%1 ... %2 %3, %4 %5.")
|
|
.arg(QObject::tr("Indexing files"))
|
|
.arg(QString::number(fileCount))
|
|
.arg(QObject::tr("files found"))
|
|
.arg(fileSizeString)
|
|
.arg(QObject::tr("to copy")));
|
|
qApp->processEvents(QEventLoop::ExcludeUserInputEvents);
|
|
});
|
|
|
|
int numFilesCopied = 0;
|
|
qint64 copiedFileSize = 0;
|
|
|
|
// Phase 1: Copy files
|
|
bool showIgnoreFileDialog = true;
|
|
QStringList copyFilesSkippedPaths(skippedPaths);
|
|
bool success = CopyDirectory(progressDialog, origPath, newPath, copyFilesSkippedPaths, filesToCopyCount, numFilesCopied,
|
|
totalSizeInBytes, copiedFileSize, showIgnoreFileDialog);
|
|
if (success && !skipRegister)
|
|
{
|
|
// Phase 2: Register project
|
|
success = RegisterProject(newPath);
|
|
}
|
|
|
|
if (!success)
|
|
{
|
|
progressDialog->setLabelText(QObject::tr("Duplicating project failed/cancelled, removing already copied files ..."));
|
|
qApp->processEvents(QEventLoop::ExcludeUserInputEvents);
|
|
|
|
DeleteProjectFiles(newPath, true);
|
|
}
|
|
|
|
progressDialog->deleteLater();
|
|
return success;
|
|
}
|
|
|
|
bool DeleteProjectFiles(const QString& path, bool force)
|
|
{
|
|
QDir projectDirectory(path);
|
|
if (projectDirectory.exists())
|
|
{
|
|
// Check if there is an actual project here or just force it
|
|
if (force || PythonBindingsInterface::Get()->GetProject(path).IsSuccess())
|
|
{
|
|
return projectDirectory.removeRecursively();
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool MoveProject(QString origPath, QString newPath, QWidget* parent, bool skipRegister)
|
|
{
|
|
origPath = QDir::toNativeSeparators(origPath);
|
|
newPath = QDir::toNativeSeparators(newPath);
|
|
|
|
if (!WarnDirectoryOverwrite(newPath, parent) || (!skipRegister && !UnregisterProject(origPath)))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
QDir newDirectory(newPath);
|
|
if (!newDirectory.removeRecursively())
|
|
{
|
|
return false;
|
|
}
|
|
if (!newDirectory.rename(origPath, newPath))
|
|
{
|
|
// Likely failed because trying to move to another partition, try copying
|
|
if (!CopyProject(origPath, newPath, parent))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
DeleteProjectFiles(origPath, true);
|
|
}
|
|
else
|
|
{
|
|
// If directoy rename succeeded then build and cache directories need to be deleted seperately
|
|
ClearProjectBuildArtifactsAndCache(origPath, newPath, parent);
|
|
}
|
|
|
|
if (!skipRegister && !RegisterProject(newPath))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ReplaceFile(const QString& origFile, const QString& newFile, QWidget* parent, bool interactive)
|
|
{
|
|
QFileInfo original(origFile);
|
|
if (original.exists())
|
|
{
|
|
if (interactive)
|
|
{
|
|
QMessageBox::StandardButton warningResult = QMessageBox::warning(
|
|
parent,
|
|
QObject::tr("Overwrite File?"),
|
|
QObject::tr("Replacing this will overwrite the current file on disk. Are you sure?"),
|
|
QMessageBox::No | QMessageBox::Yes);
|
|
|
|
if (warningResult == QMessageBox::No)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (!QFile::remove(origFile))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (!QFile::copy(newFile, origFile))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool FindSupportedCompiler(QWidget* parent)
|
|
{
|
|
auto findCompilerResult = FindSupportedCompilerForPlatform();
|
|
|
|
if (!findCompilerResult.IsSuccess())
|
|
{
|
|
QMessageBox vsWarningMessage(parent);
|
|
vsWarningMessage.setIcon(QMessageBox::Warning);
|
|
vsWarningMessage.setWindowTitle(QObject::tr("Create Project"));
|
|
// Makes link clickable
|
|
vsWarningMessage.setTextFormat(Qt::RichText);
|
|
vsWarningMessage.setText(findCompilerResult.GetError());
|
|
vsWarningMessage.setStandardButtons(QMessageBox::Close);
|
|
|
|
QSpacerItem* horizontalSpacer = new QSpacerItem(600, 0, QSizePolicy::Minimum, QSizePolicy::Expanding);
|
|
QGridLayout* layout = reinterpret_cast<QGridLayout*>(vsWarningMessage.layout());
|
|
layout->addItem(horizontalSpacer, layout->rowCount(), 0, 1, layout->columnCount());
|
|
vsWarningMessage.exec();
|
|
}
|
|
|
|
return findCompilerResult.IsSuccess();
|
|
}
|
|
|
|
ProjectManagerScreen GetProjectManagerScreen(const QString& screen)
|
|
{
|
|
auto iter = s_ProjectManagerStringNames.find(screen);
|
|
if (iter != s_ProjectManagerStringNames.end())
|
|
{
|
|
return iter.value();
|
|
}
|
|
|
|
return ProjectManagerScreen::Invalid;
|
|
}
|
|
} // namespace ProjectUtils
|
|
} // namespace O3DE::ProjectManager
|