Merge pull request #380 from flowln/task-progress
This commit is contained in:
commit
fa870bc026
@ -1,7 +1,23 @@
|
|||||||
#include "SequentialTask.h"
|
#include "SequentialTask.h"
|
||||||
|
|
||||||
SequentialTask::SequentialTask(QObject *parent) : Task(parent), m_currentIndex(-1)
|
SequentialTask::SequentialTask(QObject* parent, const QString& task_name) : Task(parent), m_name(task_name), m_currentIndex(-1) {}
|
||||||
|
|
||||||
|
SequentialTask::~SequentialTask()
|
||||||
{
|
{
|
||||||
|
for(auto task : m_queue){
|
||||||
|
if(task)
|
||||||
|
task->deleteLater();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
auto SequentialTask::getStepProgress() const -> qint64
|
||||||
|
{
|
||||||
|
return m_stepProgress;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto SequentialTask::getStepTotalProgress() const -> qint64
|
||||||
|
{
|
||||||
|
return m_stepTotalProgress;
|
||||||
}
|
}
|
||||||
|
|
||||||
void SequentialTask::addTask(Task::Ptr task)
|
void SequentialTask::addTask(Task::Ptr task)
|
||||||
@ -15,16 +31,24 @@ void SequentialTask::executeTask()
|
|||||||
startNext();
|
startNext();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool SequentialTask::abort()
|
||||||
|
{
|
||||||
|
bool succeeded = true;
|
||||||
|
for (auto& task : m_queue) {
|
||||||
|
if (!task->abort()) succeeded = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return succeeded;
|
||||||
|
}
|
||||||
|
|
||||||
void SequentialTask::startNext()
|
void SequentialTask::startNext()
|
||||||
{
|
{
|
||||||
if (m_currentIndex != -1)
|
if (m_currentIndex != -1) {
|
||||||
{
|
|
||||||
Task::Ptr previous = m_queue[m_currentIndex];
|
Task::Ptr previous = m_queue[m_currentIndex];
|
||||||
disconnect(previous.get(), 0, this, 0);
|
disconnect(previous.get(), 0, this, 0);
|
||||||
}
|
}
|
||||||
m_currentIndex++;
|
m_currentIndex++;
|
||||||
if (m_queue.isEmpty() || m_currentIndex >= m_queue.size())
|
if (m_queue.isEmpty() || m_currentIndex >= m_queue.size()) {
|
||||||
{
|
|
||||||
emitSucceeded();
|
emitSucceeded();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -33,23 +57,27 @@ void SequentialTask::startNext()
|
|||||||
connect(next.get(), SIGNAL(status(QString)), this, SLOT(subTaskStatus(QString)));
|
connect(next.get(), SIGNAL(status(QString)), this, SLOT(subTaskStatus(QString)));
|
||||||
connect(next.get(), SIGNAL(progress(qint64, qint64)), this, SLOT(subTaskProgress(qint64, qint64)));
|
connect(next.get(), SIGNAL(progress(qint64, qint64)), this, SLOT(subTaskProgress(qint64, qint64)));
|
||||||
connect(next.get(), SIGNAL(succeeded()), this, SLOT(startNext()));
|
connect(next.get(), SIGNAL(succeeded()), this, SLOT(startNext()));
|
||||||
|
|
||||||
|
setStatus(tr("Executing task %1 out of %2").arg(m_currentIndex + 1).arg(m_queue.size()));
|
||||||
next->start();
|
next->start();
|
||||||
}
|
}
|
||||||
|
|
||||||
void SequentialTask::subTaskFailed(const QString &msg)
|
void SequentialTask::subTaskFailed(const QString& msg)
|
||||||
{
|
{
|
||||||
emitFailed(msg);
|
emitFailed(msg);
|
||||||
}
|
}
|
||||||
void SequentialTask::subTaskStatus(const QString &msg)
|
void SequentialTask::subTaskStatus(const QString& msg)
|
||||||
{
|
{
|
||||||
setStatus(msg);
|
setStepStatus(m_queue[m_currentIndex]->getStatus());
|
||||||
}
|
}
|
||||||
void SequentialTask::subTaskProgress(qint64 current, qint64 total)
|
void SequentialTask::subTaskProgress(qint64 current, qint64 total)
|
||||||
{
|
{
|
||||||
if(total == 0)
|
if (total == 0) {
|
||||||
{
|
|
||||||
setProgress(0, 100);
|
setProgress(0, 100);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
setProgress(current, total);
|
setProgress(m_currentIndex, m_queue.count());
|
||||||
|
|
||||||
|
m_stepProgress = current;
|
||||||
|
m_stepTotalProgress = total;
|
||||||
}
|
}
|
||||||
|
@ -9,13 +9,21 @@ class SequentialTask : public Task
|
|||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
explicit SequentialTask(QObject *parent = 0);
|
explicit SequentialTask(QObject *parent = nullptr, const QString& task_name = "");
|
||||||
virtual ~SequentialTask() {};
|
virtual ~SequentialTask();
|
||||||
|
|
||||||
|
inline auto isMultiStep() const -> bool override { return m_queue.size() > 1; };
|
||||||
|
auto getStepProgress() const -> qint64 override;
|
||||||
|
auto getStepTotalProgress() const -> qint64 override;
|
||||||
|
|
||||||
|
inline auto getStepStatus() const -> QString override { return m_step_status; }
|
||||||
|
|
||||||
void addTask(Task::Ptr task);
|
void addTask(Task::Ptr task);
|
||||||
|
|
||||||
protected:
|
protected slots:
|
||||||
void executeTask();
|
void executeTask() override;
|
||||||
|
public slots:
|
||||||
|
bool abort() override;
|
||||||
|
|
||||||
private
|
private
|
||||||
slots:
|
slots:
|
||||||
@ -24,7 +32,19 @@ slots:
|
|||||||
void subTaskStatus(const QString &msg);
|
void subTaskStatus(const QString &msg);
|
||||||
void subTaskProgress(qint64 current, qint64 total);
|
void subTaskProgress(qint64 current, qint64 total);
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void stepStatus(QString status);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
void setStepStatus(QString status) { m_step_status = status; };
|
||||||
|
|
||||||
|
private:
|
||||||
|
QString m_name;
|
||||||
|
QString m_step_status;
|
||||||
|
|
||||||
QQueue<Task::Ptr > m_queue;
|
QQueue<Task::Ptr > m_queue;
|
||||||
int m_currentIndex;
|
int m_currentIndex;
|
||||||
|
|
||||||
|
qint64 m_stepProgress = 0;
|
||||||
|
qint64 m_stepTotalProgress = 100;
|
||||||
};
|
};
|
||||||
|
@ -21,29 +21,27 @@
|
|||||||
|
|
||||||
#include "QObjectPtr.h"
|
#include "QObjectPtr.h"
|
||||||
|
|
||||||
class Task : public QObject
|
class Task : public QObject {
|
||||||
{
|
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
using Ptr = shared_qobject_ptr<Task>;
|
using Ptr = shared_qobject_ptr<Task>;
|
||||||
|
|
||||||
enum class State
|
enum class State { Inactive, Running, Succeeded, Failed, AbortedByUser };
|
||||||
{
|
|
||||||
Inactive,
|
|
||||||
Running,
|
|
||||||
Succeeded,
|
|
||||||
Failed,
|
|
||||||
AbortedByUser
|
|
||||||
};
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit Task(QObject *parent = 0);
|
explicit Task(QObject* parent = 0);
|
||||||
virtual ~Task() {};
|
virtual ~Task() = default;
|
||||||
|
|
||||||
bool isRunning() const;
|
bool isRunning() const;
|
||||||
bool isFinished() const;
|
bool isFinished() const;
|
||||||
bool wasSuccessful() const;
|
bool wasSuccessful() const;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* MultiStep tasks are combinations of multiple tasks into a single logical task.
|
||||||
|
* The main usage of this is in SequencialTask.
|
||||||
|
*/
|
||||||
|
virtual auto isMultiStep() const -> bool { return false; }
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the string that was passed to emitFailed as the error message when the task failed.
|
* Returns the string that was passed to emitFailed as the error message when the task failed.
|
||||||
* If the task hasn't failed, returns an empty string.
|
* If the task hasn't failed, returns an empty string.
|
||||||
@ -54,52 +52,45 @@ public:
|
|||||||
|
|
||||||
virtual bool canAbort() const { return false; }
|
virtual bool canAbort() const { return false; }
|
||||||
|
|
||||||
QString getStatus()
|
QString getStatus() { return m_status; }
|
||||||
{
|
virtual auto getStepStatus() const -> QString { return {}; }
|
||||||
return m_status;
|
|
||||||
}
|
|
||||||
|
|
||||||
qint64 getProgress()
|
qint64 getProgress() { return m_progress; }
|
||||||
{
|
qint64 getTotalProgress() { return m_progressTotal; }
|
||||||
return m_progress;
|
virtual auto getStepProgress() const -> qint64 { return 0; }
|
||||||
}
|
virtual auto getStepTotalProgress() const -> qint64 { return 100; }
|
||||||
|
|
||||||
qint64 getTotalProgress()
|
protected:
|
||||||
{
|
void logWarning(const QString& line);
|
||||||
return m_progressTotal;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
private:
|
||||||
void logWarning(const QString & line);
|
|
||||||
|
|
||||||
private:
|
|
||||||
QString describe();
|
QString describe();
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void started();
|
void started();
|
||||||
void progress(qint64 current, qint64 total);
|
virtual void progress(qint64 current, qint64 total);
|
||||||
void finished();
|
void finished();
|
||||||
void succeeded();
|
void succeeded();
|
||||||
void failed(QString reason);
|
void failed(QString reason);
|
||||||
void status(QString status);
|
void status(QString status);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
virtual void start();
|
virtual void start();
|
||||||
virtual bool abort() { return false; };
|
virtual bool abort() { return false; };
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
virtual void executeTask() = 0;
|
virtual void executeTask() = 0;
|
||||||
|
|
||||||
protected slots:
|
protected slots:
|
||||||
virtual void emitSucceeded();
|
virtual void emitSucceeded();
|
||||||
virtual void emitAborted();
|
virtual void emitAborted();
|
||||||
virtual void emitFailed(QString reason);
|
virtual void emitFailed(QString reason);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void setStatus(const QString &status);
|
void setStatus(const QString& status);
|
||||||
void setProgress(qint64 current, qint64 total);
|
void setProgress(qint64 current, qint64 total);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
State m_state = State::Inactive;
|
State m_state = State::Inactive;
|
||||||
QStringList m_Warnings;
|
QStringList m_Warnings;
|
||||||
QString m_failReason = "";
|
QString m_failReason = "";
|
||||||
@ -107,4 +98,3 @@ private:
|
|||||||
int m_progress = 0;
|
int m_progress = 0;
|
||||||
int m_progressTotal = 100;
|
int m_progressTotal = 100;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -81,6 +81,12 @@ int ProgressDialog::execWithTask(Task *task)
|
|||||||
connect(task, SIGNAL(status(QString)), SLOT(changeStatus(const QString &)));
|
connect(task, SIGNAL(status(QString)), SLOT(changeStatus(const QString &)));
|
||||||
connect(task, SIGNAL(progress(qint64, qint64)), SLOT(changeProgress(qint64, qint64)));
|
connect(task, SIGNAL(progress(qint64, qint64)), SLOT(changeProgress(qint64, qint64)));
|
||||||
|
|
||||||
|
m_is_multi_step = task->isMultiStep();
|
||||||
|
if(!m_is_multi_step){
|
||||||
|
ui->globalStatusLabel->setHidden(true);
|
||||||
|
ui->globalProgressBar->setHidden(true);
|
||||||
|
}
|
||||||
|
|
||||||
// if this didn't connect to an already running task, invoke start
|
// if this didn't connect to an already running task, invoke start
|
||||||
if(!task->isRunning())
|
if(!task->isRunning())
|
||||||
{
|
{
|
||||||
@ -152,14 +158,24 @@ void ProgressDialog::onTaskSucceeded()
|
|||||||
|
|
||||||
void ProgressDialog::changeStatus(const QString &status)
|
void ProgressDialog::changeStatus(const QString &status)
|
||||||
{
|
{
|
||||||
ui->statusLabel->setText(status);
|
ui->statusLabel->setText(task->getStepStatus());
|
||||||
|
ui->globalStatusLabel->setText(status);
|
||||||
updateSize();
|
updateSize();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ProgressDialog::changeProgress(qint64 current, qint64 total)
|
void ProgressDialog::changeProgress(qint64 current, qint64 total)
|
||||||
{
|
{
|
||||||
ui->taskProgressBar->setMaximum(total);
|
ui->globalProgressBar->setMaximum(total);
|
||||||
ui->taskProgressBar->setValue(current);
|
ui->globalProgressBar->setValue(current);
|
||||||
|
|
||||||
|
if(!m_is_multi_step){
|
||||||
|
ui->taskProgressBar->setMaximum(total);
|
||||||
|
ui->taskProgressBar->setValue(current);
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
ui->taskProgressBar->setMaximum(task->getStepProgress());
|
||||||
|
ui->taskProgressBar->setValue(task->getStepTotalProgress());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ProgressDialog::keyPressEvent(QKeyEvent *e)
|
void ProgressDialog::keyPressEvent(QKeyEvent *e)
|
||||||
|
@ -19,6 +19,7 @@
|
|||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
class Task;
|
class Task;
|
||||||
|
class SequentialTask;
|
||||||
|
|
||||||
namespace Ui
|
namespace Ui
|
||||||
{
|
{
|
||||||
@ -35,7 +36,7 @@ public:
|
|||||||
|
|
||||||
void updateSize();
|
void updateSize();
|
||||||
|
|
||||||
int execWithTask(Task *task);
|
int execWithTask(Task* task);
|
||||||
int execWithTask(std::unique_ptr<Task> &&task);
|
int execWithTask(std::unique_ptr<Task> &&task);
|
||||||
int execWithTask(std::unique_ptr<Task> &task);
|
int execWithTask(std::unique_ptr<Task> &task);
|
||||||
|
|
||||||
@ -68,4 +69,6 @@ private:
|
|||||||
Ui::ProgressDialog *ui;
|
Ui::ProgressDialog *ui;
|
||||||
|
|
||||||
Task *task;
|
Task *task;
|
||||||
|
|
||||||
|
bool m_is_multi_step = false;
|
||||||
};
|
};
|
||||||
|
@ -2,14 +2,6 @@
|
|||||||
<ui version="4.0">
|
<ui version="4.0">
|
||||||
<class>ProgressDialog</class>
|
<class>ProgressDialog</class>
|
||||||
<widget class="QDialog" name="ProgressDialog">
|
<widget class="QDialog" name="ProgressDialog">
|
||||||
<property name="geometry">
|
|
||||||
<rect>
|
|
||||||
<x>0</x>
|
|
||||||
<y>0</y>
|
|
||||||
<width>400</width>
|
|
||||||
<height>100</height>
|
|
||||||
</rect>
|
|
||||||
</property>
|
|
||||||
<property name="minimumSize">
|
<property name="minimumSize">
|
||||||
<size>
|
<size>
|
||||||
<width>400</width>
|
<width>400</width>
|
||||||
@ -26,27 +18,7 @@
|
|||||||
<string>Please wait...</string>
|
<string>Please wait...</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QGridLayout" name="gridLayout">
|
<layout class="QGridLayout" name="gridLayout">
|
||||||
<item row="0" column="0">
|
<item row="4" column="0">
|
||||||
<widget class="QLabel" name="statusLabel">
|
|
||||||
<property name="text">
|
|
||||||
<string>Task Status...</string>
|
|
||||||
</property>
|
|
||||||
<property name="wordWrap">
|
|
||||||
<bool>true</bool>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item row="1" column="0">
|
|
||||||
<widget class="QProgressBar" name="taskProgressBar">
|
|
||||||
<property name="value">
|
|
||||||
<number>24</number>
|
|
||||||
</property>
|
|
||||||
<property name="textVisible">
|
|
||||||
<bool>false</bool>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item row="2" column="0">
|
|
||||||
<widget class="QPushButton" name="skipButton">
|
<widget class="QPushButton" name="skipButton">
|
||||||
<property name="sizePolicy">
|
<property name="sizePolicy">
|
||||||
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
|
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
|
||||||
@ -59,6 +31,43 @@
|
|||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item row="0" column="0">
|
||||||
|
<widget class="QLabel" name="globalStatusLabel">
|
||||||
|
<property name="text">
|
||||||
|
<string>Global Task Status...</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="2" column="0">
|
||||||
|
<widget class="QLabel" name="statusLabel">
|
||||||
|
<property name="text">
|
||||||
|
<string>Task Status...</string>
|
||||||
|
</property>
|
||||||
|
<property name="wordWrap">
|
||||||
|
<bool>true</bool>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="3" column="0">
|
||||||
|
<widget class="QProgressBar" name="taskProgressBar">
|
||||||
|
<property name="value">
|
||||||
|
<number>24</number>
|
||||||
|
</property>
|
||||||
|
<property name="textVisible">
|
||||||
|
<bool>false</bool>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="1" column="0">
|
||||||
|
<widget class="QProgressBar" name="globalProgressBar">
|
||||||
|
<property name="enabled">
|
||||||
|
<bool>true</bool>
|
||||||
|
</property>
|
||||||
|
<property name="value">
|
||||||
|
<number>24</number>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
<resources/>
|
<resources/>
|
||||||
|
@ -58,6 +58,7 @@
|
|||||||
|
|
||||||
#include "Version.h"
|
#include "Version.h"
|
||||||
#include "ui/dialogs/ProgressDialog.h"
|
#include "ui/dialogs/ProgressDialog.h"
|
||||||
|
#include "tasks/SequentialTask.h"
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
// FIXME: wasteful
|
// FIXME: wasteful
|
||||||
@ -394,25 +395,25 @@ void ModFolderPage::on_actionInstall_mods_triggered()
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
ModDownloadDialog mdownload(m_mods, this, m_inst);
|
ModDownloadDialog mdownload(m_mods, this, m_inst);
|
||||||
if(mdownload.exec()) {
|
if (mdownload.exec()) {
|
||||||
for(auto task : mdownload.getTasks()){
|
SequentialTask* tasks = new SequentialTask(this);
|
||||||
connect(task, &Task::failed, [this, task](QString reason) {
|
connect(tasks, &Task::failed, [this, tasks](QString reason) {
|
||||||
task->deleteLater();
|
CustomMessageBox::selectable(this, tr("Error"), reason, QMessageBox::Critical)->show();
|
||||||
CustomMessageBox::selectable(this, tr("Error"), reason, QMessageBox::Critical)->show();
|
tasks->deleteLater();
|
||||||
});
|
});
|
||||||
connect(task, &Task::succeeded, [this, task]() {
|
connect(tasks, &Task::succeeded, [this, tasks]() {
|
||||||
QStringList warnings = task->warnings();
|
QStringList warnings = tasks->warnings();
|
||||||
if (warnings.count()) {
|
if (warnings.count()) { CustomMessageBox::selectable(this, tr("Warnings"), warnings.join('\n'), QMessageBox::Warning)->show(); }
|
||||||
CustomMessageBox::selectable(this, tr("Warnings"), warnings.join('\n'),
|
tasks->deleteLater();
|
||||||
QMessageBox::Warning)->show();
|
});
|
||||||
}
|
|
||||||
task->deleteLater();
|
for (auto task : mdownload.getTasks()) {
|
||||||
});
|
tasks->addTask(task);
|
||||||
ProgressDialog loadDialog(this);
|
|
||||||
loadDialog.setSkipButton(true, tr("Abort"));
|
|
||||||
loadDialog.execWithTask(task);
|
|
||||||
m_mods->update();
|
|
||||||
}
|
}
|
||||||
|
ProgressDialog loadDialog(this);
|
||||||
|
loadDialog.setSkipButton(true, tr("Abort"));
|
||||||
|
loadDialog.execWithTask(tasks);
|
||||||
|
m_mods->update();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user