tageditor/renamingutility/renamingengine.cpp

386 lines
13 KiB
C++
Raw Normal View History

2015-09-06 20:20:00 +02:00
#include "./renamingengine.h"
#include "./filesystemitemmodel.h"
#include "./filteredfilesystemitemmodel.h"
#include "./tageditorobject.h"
2015-04-22 19:33:53 +02:00
#include <QDir>
#include <QStringBuilder>
2015-04-22 19:33:53 +02:00
2017-02-05 21:04:27 +01:00
#include <memory>
2015-04-22 19:33:53 +02:00
using namespace std;
namespace RenamingUtility {
2018-03-07 01:18:01 +01:00
RenamingEngine::RenamingEngine(QObject *parent)
: QObject(parent)
,
#ifndef TAGEDITOR_NO_JSENGINE
2018-03-07 01:18:01 +01:00
m_tagEditorQObj(new TagEditorObject(&m_engine))
, m_tagEditorJsObj(TAGEDITOR_JS_QOBJECT(m_engine, m_tagEditorQObj))
,
#endif
2018-03-07 01:18:01 +01:00
m_itemsProcessed(0)
, m_errorsOccured(0)
, m_aborted(false)
, m_includeSubdirs(false)
, m_isBusy(false)
, m_model(nullptr)
, m_currentModel(nullptr)
, m_previewModel(nullptr)
2015-04-22 19:33:53 +02:00
{
#ifndef TAGEDITOR_NO_JSENGINE
m_engine.globalObject().setProperty(QStringLiteral("tageditor"), m_tagEditorJsObj);
#endif
2017-01-06 21:29:43 +01:00
connect(this, &RenamingEngine::previewGenerated, this, &RenamingEngine::processPreviewGenerated);
connect(this, &RenamingEngine::changingsApplied, this, &RenamingEngine::processChangingsApplied);
2015-04-22 19:33:53 +02:00
}
#ifndef TAGEDITOR_NO_JSENGINE
2017-01-06 21:29:43 +01:00
bool RenamingEngine::setProgram(const TAGEDITOR_JS_VALUE &program)
{
if (program.isError()) {
m_errorMessage = program.property(QStringLiteral("message")).toString();
m_errorLineNumber = TAGEDITOR_JS_INT(program.property(QStringLiteral("lineNumber")));
return false;
} else if (!TAGEDITOR_JS_IS_VALID_PROG(program)) {
m_errorMessage = tr("Program is not callable. Please don't close a function you didn't open.");
m_errorLineNumber = 0;
return false;
}
m_errorMessage.clear();
m_errorLineNumber = 0;
m_program = program;
return true;
}
#endif
2015-04-22 19:33:53 +02:00
2017-01-06 21:29:43 +01:00
bool RenamingEngine::setProgram(const QString &program)
{
#ifndef TAGEDITOR_NO_JSENGINE
return setProgram(m_engine.evaluate(QStringLiteral("(function(){") % program % QStringLiteral("})")));
#else
m_errorLineNumber = 0;
m_errorMessage = tr("Not compiled with ECMA support.");
return false;
#endif
}
2017-01-06 21:29:43 +01:00
bool RenamingEngine::generatePreview(const QDir &rootDirectory, bool includeSubdirs)
2015-04-22 19:33:53 +02:00
{
#ifndef TAGEDITOR_NO_JSENGINE
2018-03-07 01:18:01 +01:00
if (m_isBusy) {
2015-04-22 19:33:53 +02:00
return false;
}
setRootItem();
m_includeSubdirs = includeSubdirs;
m_dir = rootDirectory;
(new PreviewGenerator(this))->start();
return m_isBusy = true;
#else
return false;
#endif
2015-04-22 19:33:53 +02:00
}
2017-01-06 21:29:43 +01:00
bool RenamingEngine::applyChangings()
2015-04-22 19:33:53 +02:00
{
2018-03-07 01:18:01 +01:00
if (!m_rootItem || m_isBusy) {
2015-04-22 19:33:53 +02:00
return false;
}
(new RenamingThing(this))->start();
return m_isBusy = true;
2015-04-22 19:33:53 +02:00
}
2017-01-06 21:29:43 +01:00
bool RenamingEngine::clearPreview()
2015-04-22 19:33:53 +02:00
{
2018-03-07 01:18:01 +01:00
if (m_isBusy) {
2015-04-22 19:33:53 +02:00
return false;
}
updateModel(nullptr);
m_rootItem.reset();
return true;
2015-04-22 19:33:53 +02:00
}
2017-01-06 21:29:43 +01:00
FileSystemItemModel *RenamingEngine::model()
2015-04-22 19:33:53 +02:00
{
2018-03-07 01:18:01 +01:00
if (!m_model) {
2015-04-22 19:33:53 +02:00
m_model = new FileSystemItemModel(m_rootItem.get(), this);
}
return m_model;
}
2017-01-06 21:29:43 +01:00
FilteredFileSystemItemModel *RenamingEngine::currentModel()
2015-04-22 19:33:53 +02:00
{
2018-03-07 01:18:01 +01:00
if (!m_currentModel) {
2015-04-22 19:33:53 +02:00
m_currentModel = new FilteredFileSystemItemModel(ItemStatus::Current, this);
m_currentModel->setSourceModel(model());
}
return m_currentModel;
}
2017-01-06 21:29:43 +01:00
FilteredFileSystemItemModel *RenamingEngine::previewModel()
2015-04-22 19:33:53 +02:00
{
2018-03-07 01:18:01 +01:00
if (!m_previewModel) {
2015-04-22 19:33:53 +02:00
m_previewModel = new FilteredFileSystemItemModel(ItemStatus::New, this);
m_previewModel->setSourceModel(model());
}
return m_previewModel;
}
2017-01-06 21:29:43 +01:00
void RenamingEngine::processPreviewGenerated()
2015-04-22 19:33:53 +02:00
{
finalizeTaskCompletion();
2015-10-13 20:12:00 +02:00
setRootItem(move(m_newlyGeneratedRootItem));
2015-04-22 19:33:53 +02:00
}
2017-01-06 21:29:43 +01:00
void RenamingEngine::processChangingsApplied()
2015-04-22 19:33:53 +02:00
{
finalizeTaskCompletion();
2015-04-22 19:33:53 +02:00
updateModel(nullptr);
updateModel(m_rootItem.get());
}
void RenamingEngine::resetStatus()
{
m_aborted.store(false);
m_itemsProcessed = 0;
m_errorsOccured = 0;
}
void RenamingEngine::finalizeTaskCompletion()
{
m_engine.moveToThread(thread());
m_isBusy = false;
}
2017-01-06 21:29:43 +01:00
inline void RenamingEngine::setRootItem(unique_ptr<FileSystemItem> &&rootItem)
2015-04-22 19:33:53 +02:00
{
2015-10-13 20:12:00 +02:00
updateModel(rootItem.get());
m_rootItem = move(rootItem);
2015-04-22 19:33:53 +02:00
}
2017-01-06 21:29:43 +01:00
void RenamingEngine::updateModel(FileSystemItem *rootItem)
2015-04-22 19:33:53 +02:00
{
2018-03-07 01:18:01 +01:00
if (m_model) {
2015-04-22 19:33:53 +02:00
m_model->setRootItem(rootItem);
}
}
#ifndef TAGEDITOR_NO_JSENGINE
2017-01-06 21:29:43 +01:00
unique_ptr<FileSystemItem> RenamingEngine::generatePreview(const QDir &dir, FileSystemItem *parent)
2015-04-22 19:33:53 +02:00
{
2015-10-13 20:12:00 +02:00
auto item = make_unique<FileSystemItem>(ItemStatus::Current, ItemType::Dir, dir.dirName(), parent);
2015-04-22 19:33:53 +02:00
item->setApplied(false);
2018-03-07 01:18:01 +01:00
for (const QFileInfo &entry : dir.entryInfoList()) {
if (entry.fileName() == QLatin1String("..") || entry.fileName() == QLatin1String(".")) {
2015-04-22 19:33:53 +02:00
continue;
}
2015-10-13 20:12:00 +02:00
FileSystemItem *subItem; // will be deleted by parent
2018-03-07 01:18:01 +01:00
if (entry.isDir() && m_includeSubdirs) {
2015-10-13 20:12:00 +02:00
subItem = generatePreview(QDir(entry.absoluteFilePath()), item.get()).release();
2018-03-07 01:18:01 +01:00
} else if (entry.isFile()) {
2015-10-13 20:12:00 +02:00
subItem = new FileSystemItem(ItemStatus::Current, ItemType::File, entry.fileName(), item.get());
2015-04-22 19:33:53 +02:00
subItem->setApplied(false);
} else {
subItem = nullptr;
}
2018-03-07 01:18:01 +01:00
if (subItem) {
2015-04-22 19:33:53 +02:00
executeScriptForItem(entry, subItem);
2018-03-07 01:18:01 +01:00
if (subItem->errorOccured()) {
2015-04-22 19:33:53 +02:00
++m_errorsOccured;
}
}
++m_itemsProcessed;
2018-03-07 01:18:01 +01:00
if (isAborted()) {
2015-04-22 19:33:53 +02:00
return item;
}
}
emit progress(m_itemsProcessed, m_errorsOccured);
return item;
}
#endif
2015-04-22 19:33:53 +02:00
2017-01-06 21:29:43 +01:00
void RenamingEngine::applyChangings(FileSystemItem *parentItem)
2015-04-22 19:33:53 +02:00
{
for (auto *const item : parentItem->children()) {
2018-03-07 01:18:01 +01:00
if (!item->applied() && !item->errorOccured()) {
switch (item->status()) {
2015-04-22 19:33:53 +02:00
case ItemStatus::New: {
const FileSystemItem *counterpartItem = item->counterpart(); // holds current name
const QString currentPath = counterpartItem ? counterpartItem->relativePath() : QString();
const QString newPath = item->relativePath();
2018-03-07 01:18:01 +01:00
if (item->name().isEmpty()) {
2015-04-22 19:33:53 +02:00
// new item name mustn't be empty
item->setNote(tr("generated name is empty"));
item->setErrorOccured(true);
2018-03-07 01:18:01 +01:00
} else if (counterpartItem && !counterpartItem->name().isEmpty()) {
2015-04-22 19:33:53 +02:00
// rename current item
2018-03-07 01:18:01 +01:00
if (item->parent() != counterpartItem->parent() || item->name() != counterpartItem->name()) {
if (m_dir.exists(newPath)) {
if (item->parent() == counterpartItem->parent()) {
2015-04-22 19:33:53 +02:00
item->setNote(tr("unable to rename, there is already an entry with the same name"));
} else {
item->setNote(tr("unable to move, there is already an entry with the same name"));
}
item->setErrorOccured(true);
2018-03-07 01:18:01 +01:00
} else if (m_dir.rename(currentPath, newPath)) {
if (item->parent() == counterpartItem->parent()) {
2015-04-22 19:33:53 +02:00
item->setNote(tr("renamed"));
} else {
item->setNote(tr("moved"));
}
item->setApplied(true);
} else {
item->setNote(tr("unable to rename"));
item->setErrorOccured(true);
}
} else {
item->setNote(tr("nothing to be changed"));
item->setApplied(true);
}
2018-03-07 01:18:01 +01:00
} else if (item->type() == ItemType::Dir) {
2015-04-22 19:33:53 +02:00
// create new item, but only if its a dir
2018-03-07 01:18:01 +01:00
if (m_dir.exists(newPath)) {
2015-04-22 19:33:53 +02:00
item->setNote(tr("directory already existed"));
item->setApplied(true);
2018-03-07 01:18:01 +01:00
} else if (m_dir.mkpath(newPath)) {
2015-04-22 19:33:53 +02:00
item->setNote(tr("directory created"));
item->setApplied(true);
} else {
item->setNote(tr("unable to create directory"));
item->setErrorOccured(true);
}
} else {
// can not create new file
item->setNote(tr("unable to create file"));
item->setErrorOccured(true);
}
break;
2018-03-07 01:18:01 +01:00
}
case ItemStatus::Current:
2015-04-22 19:33:53 +02:00
break;
}
}
2018-03-07 01:18:01 +01:00
if (item->errorOccured()) {
2015-04-22 19:33:53 +02:00
++m_errorsOccured;
}
// apply changings for child items as well
2018-03-07 01:18:01 +01:00
if (item->type() == ItemType::Dir) {
2015-04-22 19:33:53 +02:00
applyChangings(item);
}
}
m_itemsProcessed += parentItem->children().size();
emit progress(m_itemsProcessed, m_errorsOccured);
}
2017-01-06 21:29:43 +01:00
void RenamingEngine::setError(const QList<FileSystemItem *> items)
2015-04-22 19:33:53 +02:00
{
for (auto *const item : items) {
2015-04-22 19:33:53 +02:00
item->setErrorOccured(true);
item->setNote(tr("skipped due to error of superior item"));
}
}
#ifndef TAGEDITOR_NO_JSENGINE
2017-01-06 21:29:43 +01:00
void RenamingEngine::executeScriptForItem(const QFileInfo &fileInfo, FileSystemItem *item)
2015-04-22 19:33:53 +02:00
{
// make file info for the specified item available in the script
m_tagEditorQObj->setFileInfo(fileInfo, item);
2015-10-13 20:12:00 +02:00
// execute script
const auto scriptResult(m_program.call());
2018-03-07 01:18:01 +01:00
if (scriptResult.isError()) {
2015-10-13 20:12:00 +02:00
// handle error
2015-04-22 19:33:53 +02:00
item->setErrorOccured(true);
item->setNote(scriptResult.toString());
return;
}
// create preview for action
const QString &newName = m_tagEditorQObj->newName();
const QString &newRelativeDirectory = m_tagEditorQObj->newRelativeDirectory();
switch (m_tagEditorQObj->action()) {
case ActionType::None:
item->setNote(tr("no action specified"));
break;
case ActionType::Rename:
if (!newRelativeDirectory.isEmpty()) {
2018-11-15 21:32:49 +01:00
FileSystemItem *const counterpartParent = item->root()->makeChildAvailable(newRelativeDirectory);
const QString &counterpartName = newName.isEmpty() ? item->name() : newName;
if (const auto *const conflictingItem = counterpartParent->findChild(counterpartName, item)) {
QString conflictingName;
if (const auto *const conflictingCounterpart = conflictingItem->counterpart()) {
conflictingCounterpart->relativePath(conflictingName);
} else {
conflictingName = conflictingItem->currentName();
}
item->setNote(tr("name is already used at new location by '%1'").arg(conflictingName));
item->setErrorOccured(true);
} else {
2018-11-15 21:32:49 +01:00
auto *const counterpart = new FileSystemItem(ItemStatus::New, item->type(), counterpartName, counterpartParent);
item->setCounterpart(counterpart);
counterpart->setCheckable(true);
counterpart->setChecked(true);
2015-04-22 19:33:53 +02:00
}
} else if (!newName.isEmpty()) {
item->setNewName(newName);
}
2018-11-15 21:32:49 +01:00
if (FileSystemItem *const newItem = item->counterpart()) {
if ((newItem->name().isEmpty() || newItem->name() == item->name()) && (newItem->parent() == item->parent())) {
item->setNote(tr("name doesn't change"));
} else if (newItem->parent() && newItem->parent()->findChild(newItem->name(), newItem)) {
item->setNote(tr("generated name is already used"));
item->setErrorOccured(true);
} else if (newItem->parent() == item->parent()) {
item->setNote(tr("will be renamed"));
newItem->setCheckable(true);
newItem->setChecked(true);
2015-04-22 19:33:53 +02:00
} else {
item->setNote(tr("will be moved"));
2015-04-22 19:33:53 +02:00
}
} else if (item->note().isEmpty()) {
item->setNote(tr("can not be renamed"));
2015-04-22 19:33:53 +02:00
}
break;
default:
item->setNote(m_tagEditorQObj->note().isEmpty() ? tr("skipped") : m_tagEditorQObj->note());
2015-04-22 19:33:53 +02:00
}
}
PreviewGenerator::PreviewGenerator(RenamingEngine *engine)
: QThread(engine)
, m_engine(engine)
{
m_engine->m_engine.moveToThread(this);
connect(this, &PreviewGenerator::finished, m_engine, &RenamingEngine::previewGenerated, Qt::QueuedConnection);
connect(this, &PreviewGenerator::finished, this, &PreviewGenerator::deleteLater);
}
void PreviewGenerator::run()
{
m_engine->resetStatus();
m_engine->m_newlyGeneratedRootItem = m_engine->generatePreview(m_engine->m_dir);
}
RenamingThing::RenamingThing(RenamingEngine *engine)
: QThread(engine)
, m_engine(engine)
{
m_engine->m_engine.moveToThread(this);
connect(this, &RenamingThing::finished, m_engine, &RenamingEngine::changingsApplied, Qt::QueuedConnection);
connect(this, &RenamingThing::finished, this, &RenamingThing::deleteLater);
}
void RenamingThing::run()
{
m_engine->resetStatus();
m_engine->applyChangings(m_engine->m_rootItem.get());
}
#endif
2015-04-22 19:33:53 +02:00
} // namespace RenamingUtility