1 ///////////////////////////////////////////////////////////////////////////////
2 // Simple x264 Launcher
3 // Copyright (C) 2004-2016 LoRd_MuldeR <MuldeR2@GMX.de>
5 // This program is free software; you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation; either version 2 of the License, or
8 // (at your option) any later version.
10 // This program is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 // GNU General Public License for more details.
15 // You should have received a copy of the GNU General Public License along
16 // with this program; if not, write to the Free Software Foundation, Inc.,
17 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 // http://www.gnu.org/licenses/gpl-2.0.txt
20 ///////////////////////////////////////////////////////////////////////////////
24 #include "model_jobList.h"
25 #include "thread_encode.h"
26 #include "encoder_factory.h"
27 #include "model_options.h"
28 #include "model_preferences.h"
32 #include <MUtils/Sound.h>
39 static const char *KEY_ENTRY_COUNT = "entry_count";
40 static const char *KEY_SOURCE_FILE = "source_file";
41 static const char *KEY_OUTPUT_FILE = "output_file";
42 static const char *KEY_ENC_OPTIONS = "enc_options";
44 static const char *JOB_TEMPLATE = "job_%08x";
46 #define VALID_INDEX(INDEX) ((INDEX).isValid() && ((INDEX).row() >= 0) && ((INDEX).row() < m_jobs.count()))
48 JobListModel::JobListModel(PreferencesModel *preferences)
50 m_preferences = preferences;
53 JobListModel::~JobListModel(void)
55 while(!m_jobs.isEmpty())
57 QUuid id = m_jobs.takeFirst();
58 EncodeThread *thread = m_threads.value(id, NULL);
59 LogFileModel *logFile = m_logFile.value(id, NULL);
60 MUTILS_DELETE(thread);
61 MUTILS_DELETE(logFile);
65 ///////////////////////////////////////////////////////////////////////////////
67 ///////////////////////////////////////////////////////////////////////////////
69 int JobListModel::columnCount(const QModelIndex &parent) const
74 int JobListModel::rowCount(const QModelIndex &parent) const
76 return m_jobs.count();
79 QVariant JobListModel::headerData(int section, Qt::Orientation orientation, int role) const
81 if((orientation == Qt::Horizontal) && (role == Qt::DisplayRole))
86 return QVariant::fromValue<QString>(tr("Job"));
89 return QVariant::fromValue<QString>(tr("Status"));
92 return QVariant::fromValue<QString>(tr("Progress"));
95 return QVariant::fromValue<QString>(tr("Details"));
106 QModelIndex JobListModel::index(int row, int column, const QModelIndex &parent) const
108 return createIndex(row, column, NULL);
111 QModelIndex JobListModel::parent(const QModelIndex &index) const
113 return QModelIndex();
116 QVariant JobListModel::data(const QModelIndex &index, int role) const
118 if(role == Qt::DisplayRole)
120 if(index.row() >= 0 && index.row() < m_jobs.count())
122 switch(index.column())
125 return m_name.value(m_jobs.at(index.row()));
128 switch(m_status.value(m_jobs.at(index.row())))
130 case JobStatus_Enqueued:
131 return QVariant::fromValue<QString>(tr("Enqueued."));
133 case JobStatus_Starting:
134 return QVariant::fromValue<QString>(tr("Starting..."));
136 case JobStatus_Indexing:
137 return QVariant::fromValue<QString>(tr("Indexing..."));
139 case JobStatus_Running:
140 return QVariant::fromValue<QString>(tr("Running..."));
142 case JobStatus_Running_Pass1:
143 return QVariant::fromValue<QString>(tr("Running... (Pass 1)"));
145 case JobStatus_Running_Pass2:
146 return QVariant::fromValue<QString>(tr("Running... (Pass 2)"));
148 case JobStatus_Completed:
149 return QVariant::fromValue<QString>(tr("Completed."));
151 case JobStatus_Failed:
152 return QVariant::fromValue<QString>(tr("Failed!"));
154 case JobStatus_Pausing:
155 return QVariant::fromValue<QString>(tr("Pausing..."));
157 case JobStatus_Paused:
158 return QVariant::fromValue<QString>(tr("Paused."));
160 case JobStatus_Resuming:
161 return QVariant::fromValue<QString>(tr("Resuming..."));
163 case JobStatus_Aborting:
164 return QVariant::fromValue<QString>(tr("Aborting..."));
166 case JobStatus_Aborted:
167 return QVariant::fromValue<QString>(tr("Aborted!"));
170 return QVariant::fromValue<QString>(tr("(Unknown)"));
175 return QString().sprintf("%d%%", m_progress.value(m_jobs.at(index.row())));
178 return m_details.value(m_jobs.at(index.row()));
186 else if(role == Qt::DecorationRole)
188 if(index.row() >= 0 && index.row() < m_jobs.count() && index.column() == 0)
190 switch(m_status.value(m_jobs.at(index.row())))
192 case JobStatus_Enqueued:
193 return QIcon(":/buttons/hourglass.png");
195 case JobStatus_Starting:
196 return QIcon(":/buttons/lightning.png");
198 case JobStatus_Indexing:
199 return QIcon(":/buttons/find.png");
201 case JobStatus_Running:
202 case JobStatus_Running_Pass1:
203 case JobStatus_Running_Pass2:
204 return QIcon(":/buttons/play.png");
206 case JobStatus_Completed:
207 return QIcon(":/buttons/accept.png");
209 case JobStatus_Failed:
210 return QIcon(":/buttons/exclamation.png");
212 case JobStatus_Pausing:
213 return QIcon(":/buttons/clock_pause.png");
215 case JobStatus_Paused:
216 return QIcon(":/buttons/suspended.png");
218 case JobStatus_Resuming:
219 return QIcon(":/buttons/clock_play.png");
221 case JobStatus_Aborting:
222 return QIcon(":/buttons/clock_stop.png");
224 case JobStatus_Aborted:
225 return QIcon(":/buttons/error.png");
237 ///////////////////////////////////////////////////////////////////////////////
239 ///////////////////////////////////////////////////////////////////////////////
241 QModelIndex JobListModel::insertJob(EncodeThread *thread)
243 const QUuid id = thread->getId();
244 if(m_jobs.contains(id))
246 return QModelIndex();
249 const AbstractEncoderInfo &encoderInfo = EncoderFactory::getEncoderInfo(thread->options()->encType());
250 QString config = encoderInfo.getName();
251 switch(encoderInfo.rcModeToType(thread->options()->rcMode()))
253 case AbstractEncoderInfo::RC_TYPE_QUANTIZER:
254 config.append(QString(", %1@%2").arg(encoderInfo.rcModeToString(thread->options()->rcMode()), QString::number(qRound(thread->options()->quantizer()))));
256 case AbstractEncoderInfo::RC_TYPE_RATE_KBPS:
257 case AbstractEncoderInfo::RC_TYPE_MULTIPASS:
258 config.append(QString(", %1@%2").arg(encoderInfo.rcModeToString(thread->options()->rcMode()), QString::number(thread->options()->bitrate())));
263 QString jobName = QString("%1 (%2)").arg(QFileInfo(thread->sourceFileName()).completeBaseName().simplified(), config);
267 for(int i = 0; i < m_jobs.count(); i++)
269 if(m_name.value(m_jobs.at(i)).compare(jobName, Qt::CaseInsensitive) == 0)
277 jobName = QString("%1 %2 (%3)").arg(QFileInfo(thread->sourceFileName()).completeBaseName().simplified(), QString::number(n++), config);
283 LogFileModel *logFile = new LogFileModel(thread->sourceFileName(), thread->outputFileName(), config);
285 beginInsertRows(QModelIndex(), m_jobs.count(), m_jobs.count());
287 m_name.insert(id, jobName);
288 m_status.insert(id, JobStatus_Enqueued);
289 m_progress.insert(id, 0);
290 m_threads.insert(id, thread);
291 m_logFile.insert(id, logFile);
292 m_details.insert(id, tr("Not started yet."));
295 connect(thread, SIGNAL(statusChanged(QUuid, JobStatus)), this, SLOT(updateStatus(QUuid, JobStatus)), Qt::QueuedConnection);
296 connect(thread, SIGNAL(progressChanged(QUuid, unsigned int)), this, SLOT(updateProgress(QUuid, unsigned int)), Qt::QueuedConnection);
297 connect(thread, SIGNAL(messageLogged(QUuid, QString)), logFile, SLOT(addLogMessage(QUuid, QString)), Qt::QueuedConnection);
298 connect(thread, SIGNAL(detailsChanged(QUuid, QString)), this, SLOT(updateDetails(QUuid, QString)), Qt::QueuedConnection);
300 return createIndex(m_jobs.count() - 1, 0, NULL);
303 bool JobListModel::startJob(const QModelIndex &index)
305 if(VALID_INDEX(index))
307 QUuid id = m_jobs.at(index.row());
308 if(m_status.value(id) == JobStatus_Enqueued)
310 updateStatus(id, JobStatus_Starting);
311 updateDetails(id, tr("Starting up, please wait..."));
312 m_threads.value(id)->start();
320 bool JobListModel::pauseJob(const QModelIndex &index)
322 if(VALID_INDEX(index))
324 QUuid id = m_jobs.at(index.row());
325 JobStatus status = m_status.value(id);
326 if((status == JobStatus_Indexing) || (status == JobStatus_Running) ||
327 (status == JobStatus_Running_Pass1) || (status == JobStatus_Running_Pass2))
329 updateStatus(id, JobStatus_Pausing);
330 m_threads.value(id)->pauseJob();
338 bool JobListModel::resumeJob(const QModelIndex &index)
340 if(VALID_INDEX(index))
342 QUuid id = m_jobs.at(index.row());
343 JobStatus status = m_status.value(id);
344 if(status == JobStatus_Paused)
346 updateStatus(id, JobStatus_Resuming);
347 m_threads.value(id)->resumeJob();
355 bool JobListModel::abortJob(const QModelIndex &index)
357 if(VALID_INDEX(index))
359 QUuid id = m_jobs.at(index.row());
360 if(m_status.value(id) == JobStatus_Indexing || m_status.value(id) == JobStatus_Running ||
361 m_status.value(id) == JobStatus_Running_Pass1 || JobStatus_Running_Pass2)
363 updateStatus(id, JobStatus_Aborting);
364 m_threads.value(id)->abortJob();
372 bool JobListModel::deleteJob(const QModelIndex &index)
374 if(VALID_INDEX(index))
376 QUuid id = m_jobs.at(index.row());
377 if(m_status.value(id) == JobStatus_Completed || m_status.value(id) == JobStatus_Failed ||
378 m_status.value(id) == JobStatus_Aborted || m_status.value(id) == JobStatus_Enqueued)
380 int idx = index.row();
381 QUuid id = m_jobs.at(idx);
382 EncodeThread *thread = m_threads.value(id, NULL);
383 LogFileModel *logFile = m_logFile.value(id, NULL);
384 if((thread == NULL) || (!thread->isRunning()))
387 beginRemoveRows(QModelIndex(), idx, idx);
388 m_jobs.removeAt(index.row());
390 m_threads.remove(id);
392 m_progress.remove(id);
393 m_logFile.remove(id);
394 m_details.remove(id);
396 MUTILS_DELETE(thread);
397 MUTILS_DELETE(logFile);
406 bool JobListModel::moveJob(const QModelIndex &index, const int &direction)
408 if(VALID_INDEX(index))
410 if((direction == MOVE_UP) && (index.row() > 0))
412 beginMoveRows(QModelIndex(), index.row(), index.row(), QModelIndex(), index.row() - 1);
413 m_jobs.swap(index.row(), index.row() - 1);
417 if((direction == MOVE_DOWN) && (index.row() < m_jobs.size() - 1))
419 beginMoveRows(QModelIndex(), index.row(), index.row(), QModelIndex(), index.row() + 2);
420 m_jobs.swap(index.row(), index.row() + 1);
429 LogFileModel *JobListModel::getLogFile(const QModelIndex &index)
431 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
433 return m_logFile.value(m_jobs.at(index.row()));
439 const QString &JobListModel::getJobSourceFile(const QModelIndex &index)
441 static QString nullStr;
443 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
445 EncodeThread *thread = m_threads.value(m_jobs.at(index.row()));
446 return (thread != NULL) ? thread->sourceFileName() : nullStr;
452 const QString &JobListModel::getJobOutputFile(const QModelIndex &index)
454 static QString nullStr;
456 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
458 EncodeThread *thread = m_threads.value(m_jobs.at(index.row()));
459 return (thread != NULL) ? thread->outputFileName() : nullStr;
465 JobStatus JobListModel::getJobStatus(const QModelIndex &index)
467 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
469 return m_status.value(m_jobs.at(index.row()));
472 return static_cast<JobStatus>(-1);
475 unsigned int JobListModel::getJobProgress(const QModelIndex &index)
477 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
479 return m_progress.value(m_jobs.at(index.row()));
485 const OptionsModel *JobListModel::getJobOptions(const QModelIndex &index)
487 static QString nullStr;
489 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
491 EncodeThread *thread = m_threads.value(m_jobs.at(index.row()));
492 return (thread != NULL) ? thread->options() : NULL;
498 QModelIndex JobListModel::getJobIndexById(const QUuid &id)
500 if(m_jobs.contains(id))
502 return createIndex(m_jobs.indexOf(id), 0);
505 return QModelIndex();
508 ///////////////////////////////////////////////////////////////////////////////
510 ///////////////////////////////////////////////////////////////////////////////
512 void JobListModel::updateStatus(const QUuid &jobId, JobStatus newStatus)
516 if((index = m_jobs.indexOf(jobId)) >= 0)
518 m_status.insert(jobId, newStatus);
519 emit dataChanged(createIndex(index, 0), createIndex(index, 1));
521 if(m_preferences->getEnableSounds())
525 case JobStatus_Completed:
526 MUtils::Sound::play_sound("tada", true);
528 case JobStatus_Aborted:
529 MUtils::Sound::play_sound("shattering", true);
531 case JobStatus_Failed:
532 MUtils::Sound::play_sound("failure", true);
539 void JobListModel::updateProgress(const QUuid &jobId, unsigned int newProgress)
543 if((index = m_jobs.indexOf(jobId)) >= 0)
545 m_progress.insert(jobId, qBound(0U, newProgress, 100U));
546 emit dataChanged(createIndex(index, 2), createIndex(index, 2));
550 void JobListModel::updateDetails(const QUuid &jobId, const QString &details)
554 if((index = m_jobs.indexOf(jobId)) >= 0)
556 m_details.insert(jobId, details);
557 emit dataChanged(createIndex(index, 3), createIndex(index, 3));
561 size_t JobListModel::saveQueuedJobs(void)
563 const QString appDir = x264_data_path();
564 QSettings settings(QString("%1/queue.ini").arg(appDir), QSettings::IniFormat);
567 settings.setValue(KEY_ENTRY_COUNT, 0);
568 size_t jobCounter = 0;
570 for(QList<QUuid>::ConstIterator iter = m_jobs.constBegin(); iter != m_jobs.constEnd(); iter++)
572 if(m_status.value(*iter) == JobStatus_Enqueued)
574 if(const EncodeThread *thread = m_threads.value(*iter))
576 settings.beginGroup(QString().sprintf(JOB_TEMPLATE, jobCounter++));
577 settings.setValue(KEY_SOURCE_FILE, thread->sourceFileName());
578 settings.setValue(KEY_OUTPUT_FILE, thread->outputFileName());
580 settings.beginGroup(KEY_ENC_OPTIONS);
581 OptionsModel::saveOptions(thread->options(), settings);
586 settings.setValue(KEY_ENTRY_COUNT, jobCounter);
595 size_t JobListModel::loadQueuedJobs(const SysinfoModel *sysinfo)
597 const QString appDir = x264_data_path();
598 QSettings settings(QString("%1/queue.ini").arg(appDir), QSettings::IniFormat);
601 const size_t jobCounter = settings.value(KEY_ENTRY_COUNT, 0).toUInt(&ok);
603 if((!ok) || (jobCounter < 1))
608 const QStringList groups = settings.childGroups();
609 for(size_t i = 0; i < jobCounter; i++)
611 if(!groups.contains(QString().sprintf(JOB_TEMPLATE, i)))
617 size_t jobsCreated = 0;
618 for(size_t i = 0; i < jobCounter; i++)
620 settings.beginGroup(QString().sprintf(JOB_TEMPLATE, i));
621 const QString sourceFileName = settings.value(KEY_SOURCE_FILE, QString()).toString().trimmed();
622 const QString outputFileName = settings.value(KEY_OUTPUT_FILE, QString()).toString().trimmed();
624 if(sourceFileName.isEmpty() || outputFileName.isEmpty())
630 settings.beginGroup(KEY_ENC_OPTIONS);
631 OptionsModel options(sysinfo);
632 const bool okay = OptionsModel::loadOptions(&options, settings);
639 EncodeThread *thread = new EncodeThread(sourceFileName, outputFileName, &options, sysinfo, m_preferences);
648 void JobListModel::clearQueuedJobs(void)
650 const QString appDir = x264_data_path();
651 QSettings settings(QString("%1/queue.ini").arg(appDir), QSettings::IniFormat);
653 settings.setValue(KEY_ENTRY_COUNT, 0);