1 ///////////////////////////////////////////////////////////////////////////////
2 // Simple x264 Launcher
3 // Copyright (C) 2004-2012 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 ///////////////////////////////////////////////////////////////////////////////
22 #include "model_jobList.h"
24 #include "thread_encode.h"
29 JobListModel::JobListModel(void)
33 JobListModel::~JobListModel(void)
35 while(!m_jobs.isEmpty())
37 QUuid id = m_jobs.takeFirst();
38 EncodeThread *thread = m_threads.value(id, NULL);
39 LogFileModel *logFile = m_logFile.value(id, NULL);
45 ///////////////////////////////////////////////////////////////////////////////
47 ///////////////////////////////////////////////////////////////////////////////
49 int JobListModel::columnCount(const QModelIndex &parent) const
54 int JobListModel::rowCount(const QModelIndex &parent) const
56 return m_jobs.count();
59 QVariant JobListModel::headerData(int section, Qt::Orientation orientation, int role) const
61 if((orientation == Qt::Horizontal) && (role == Qt::DisplayRole))
66 return QVariant::fromValue<QString>(tr("Job"));
69 return QVariant::fromValue<QString>(tr("Status"));
72 return QVariant::fromValue<QString>(tr("Progress"));
75 return QVariant::fromValue<QString>(tr("Details"));
86 QModelIndex JobListModel::index(int row, int column, const QModelIndex &parent) const
88 return createIndex(row, column, NULL);
91 QModelIndex JobListModel::parent(const QModelIndex &index) const
96 QVariant JobListModel::data(const QModelIndex &index, int role) const
98 if(role == Qt::DisplayRole)
100 if(index.row() >= 0 && index.row() < m_jobs.count())
102 switch(index.column())
105 return m_name.value(m_jobs.at(index.row()));
108 switch(m_status.value(m_jobs.at(index.row())))
110 case EncodeThread::JobStatus_Enqueued:
111 return QVariant::fromValue<QString>(tr("Enqueued."));
113 case EncodeThread::JobStatus_Starting:
114 return QVariant::fromValue<QString>(tr("Starting..."));
116 case EncodeThread::JobStatus_Indexing:
117 return QVariant::fromValue<QString>(tr("Indexing..."));
119 case EncodeThread::JobStatus_Running:
120 return QVariant::fromValue<QString>(tr("Running..."));
122 case EncodeThread::JobStatus_Running_Pass1:
123 return QVariant::fromValue<QString>(tr("Running... (Pass 1)"));
125 case EncodeThread::JobStatus_Running_Pass2:
126 return QVariant::fromValue<QString>(tr("Running... (Pass 2)"));
128 case EncodeThread::JobStatus_Completed:
129 return QVariant::fromValue<QString>(tr("Completed."));
131 case EncodeThread::JobStatus_Failed:
132 return QVariant::fromValue<QString>(tr("Failed!"));
134 case EncodeThread::JobStatus_Pausing:
135 return QVariant::fromValue<QString>(tr("Pausing..."));
137 case EncodeThread::JobStatus_Paused:
138 return QVariant::fromValue<QString>(tr("Paused."));
140 case EncodeThread::JobStatus_Resuming:
141 return QVariant::fromValue<QString>(tr("Resuming..."));
143 case EncodeThread::JobStatus_Aborting:
144 return QVariant::fromValue<QString>(tr("Aborting..."));
146 case EncodeThread::JobStatus_Aborted:
147 return QVariant::fromValue<QString>(tr("Aborted!"));
150 return QVariant::fromValue<QString>(tr("(Unknown)"));
155 return QString().sprintf("%d%%", m_progress.value(m_jobs.at(index.row())));
158 return m_details.value(m_jobs.at(index.row()));
166 else if(role == Qt::DecorationRole)
168 if(index.row() >= 0 && index.row() < m_jobs.count() && index.column() == 0)
170 switch(m_status.value(m_jobs.at(index.row())))
172 case EncodeThread::JobStatus_Enqueued:
173 return QIcon(":/buttons/hourglass.png");
175 case EncodeThread::JobStatus_Starting:
176 return QIcon(":/buttons/lightning.png");
178 case EncodeThread::JobStatus_Indexing:
179 return QIcon(":/buttons/find.png");
181 case EncodeThread::JobStatus_Running:
182 case EncodeThread::JobStatus_Running_Pass1:
183 case EncodeThread::JobStatus_Running_Pass2:
184 return QIcon(":/buttons/play.png");
186 case EncodeThread::JobStatus_Completed:
187 return QIcon(":/buttons/accept.png");
189 case EncodeThread::JobStatus_Failed:
190 return QIcon(":/buttons/exclamation.png");
192 case EncodeThread::JobStatus_Pausing:
193 return QIcon(":/buttons/clock_pause.png");
195 case EncodeThread::JobStatus_Paused:
196 return QIcon(":/buttons/suspended.png");
198 case EncodeThread::JobStatus_Resuming:
199 return QIcon(":/buttons/clock_play.png");
201 case EncodeThread::JobStatus_Aborting:
202 return QIcon(":/buttons/clock_stop.png");
204 case EncodeThread::JobStatus_Aborted:
205 return QIcon(":/buttons/error.png");
217 ///////////////////////////////////////////////////////////////////////////////
219 ///////////////////////////////////////////////////////////////////////////////
221 QModelIndex JobListModel::insertJob(EncodeThread *thread)
223 QUuid id = thread->getId();
224 LogFileModel *logFile = NULL;
226 if(m_jobs.contains(id))
228 return QModelIndex();
232 QString jobName = QFileInfo(thread->sourceFileName()).completeBaseName();
237 for(int i = 0; i < m_jobs.count(); i++)
239 if(m_name.value(m_jobs.at(i)).compare(jobName, Qt::CaseInsensitive) == 0)
247 jobName = QString("%1 (%2)").arg(QFileInfo(thread->sourceFileName()).completeBaseName(), QString::number(n++));
253 beginInsertRows(QModelIndex(), m_jobs.count(), m_jobs.count());
255 m_name.insert(id, jobName);
256 m_status.insert(id, EncodeThread::JobStatus_Enqueued);
257 m_progress.insert(id, 0);
258 m_threads.insert(id, thread);
259 m_logFile.insert(id, (logFile = new LogFileModel));
260 m_details.insert(id, tr("Not started yet."));
263 connect(thread, SIGNAL(statusChanged(QUuid, EncodeThread::JobStatus)), this, SLOT(updateStatus(QUuid, EncodeThread::JobStatus)), Qt::QueuedConnection);
264 connect(thread, SIGNAL(progressChanged(QUuid, unsigned int)), this, SLOT(updateProgress(QUuid, unsigned int)), Qt::QueuedConnection);
265 connect(thread, SIGNAL(messageLogged(QUuid, QString)), logFile, SLOT(addLogMessage(QUuid, QString)), Qt::QueuedConnection);
266 connect(thread, SIGNAL(detailsChanged(QUuid, QString)), this, SLOT(updateDetails(QUuid, QString)), Qt::QueuedConnection);
268 return createIndex(m_jobs.count() - 1, 0, NULL);
271 bool JobListModel::startJob(const QModelIndex &index)
273 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
275 QUuid id = m_jobs.at(index.row());
276 if(m_status.value(id) == EncodeThread::JobStatus_Enqueued)
278 updateStatus(id, EncodeThread::JobStatus_Starting);
279 updateDetails(id, tr("Starting up, please wait..."));
280 m_threads.value(id)->start();
288 bool JobListModel::pauseJob(const QModelIndex &index)
290 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
292 QUuid id = m_jobs.at(index.row());
293 EncodeThread::JobStatus status = m_status.value(id);
294 if((status == EncodeThread::JobStatus_Indexing) || (status == EncodeThread::JobStatus_Running) ||
295 (status == EncodeThread::JobStatus_Running_Pass1) || (status == EncodeThread::JobStatus_Running_Pass2))
297 updateStatus(id, EncodeThread::JobStatus_Pausing);
298 m_threads.value(id)->pauseJob();
306 bool JobListModel::resumeJob(const QModelIndex &index)
308 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
310 QUuid id = m_jobs.at(index.row());
311 EncodeThread::JobStatus status = m_status.value(id);
312 if(status == EncodeThread::JobStatus_Paused)
314 updateStatus(id, EncodeThread::JobStatus_Resuming);
315 m_threads.value(id)->resumeJob();
323 bool JobListModel::abortJob(const QModelIndex &index)
325 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
327 QUuid id = m_jobs.at(index.row());
328 if(m_status.value(id) == EncodeThread::JobStatus_Indexing || m_status.value(id) == EncodeThread::JobStatus_Running ||
329 m_status.value(id) == EncodeThread::JobStatus_Running_Pass1 || EncodeThread::JobStatus_Running_Pass2)
331 updateStatus(id, EncodeThread::JobStatus_Aborting);
332 m_threads.value(id)->abortJob();
340 bool JobListModel::deleteJob(const QModelIndex &index)
342 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
344 QUuid id = m_jobs.at(index.row());
345 if(m_status.value(id) == EncodeThread::JobStatus_Completed || m_status.value(id) == EncodeThread::JobStatus_Failed ||
346 m_status.value(id) == EncodeThread::JobStatus_Aborted || m_status.value(id) == EncodeThread::JobStatus_Enqueued)
348 int idx = index.row();
349 QUuid id = m_jobs.at(idx);
350 EncodeThread *thread = m_threads.value(id, NULL);
351 LogFileModel *logFile = m_logFile.value(id, NULL);
352 if((thread == NULL) || (!thread->isRunning()))
355 beginRemoveRows(QModelIndex(), idx, idx);
356 m_jobs.removeAt(index.row());
358 m_threads.remove(id);
360 m_progress.remove(id);
361 m_logFile.remove(id);
362 m_details.remove(id);
365 X264_DELETE(logFile);
374 LogFileModel *JobListModel::getLogFile(const QModelIndex &index)
376 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
378 return m_logFile.value(m_jobs.at(index.row()));
384 const QString &JobListModel::getJobOutputFile(const QModelIndex &index)
386 static QString nullStr;
388 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
390 EncodeThread *thread = m_threads.value(m_jobs.at(index.row()));
391 return (thread != NULL) ? thread->outputFileName() : nullStr;
397 EncodeThread::JobStatus JobListModel::getJobStatus(const QModelIndex &index)
399 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
401 return m_status.value(m_jobs.at(index.row()));
404 return static_cast<EncodeThread::JobStatus>(-1);
407 unsigned int JobListModel::getJobProgress(const QModelIndex &index)
409 if(index.isValid() && index.row() >= 0 && index.row() < m_jobs.count())
411 return m_progress.value(m_jobs.at(index.row()));
417 QModelIndex JobListModel::getJobIndexById(const QUuid &id)
419 if(m_jobs.contains(id))
421 return createIndex(m_jobs.indexOf(id), 0);
424 return QModelIndex();
427 ///////////////////////////////////////////////////////////////////////////////
429 ///////////////////////////////////////////////////////////////////////////////
431 void JobListModel::updateStatus(const QUuid &jobId, EncodeThread::JobStatus newStatus)
435 if((index = m_jobs.indexOf(jobId)) >= 0)
437 m_status.insert(jobId, newStatus);
438 emit dataChanged(createIndex(index, 0), createIndex(index, 1));
442 void JobListModel::updateProgress(const QUuid &jobId, unsigned int newProgress)
446 if((index = m_jobs.indexOf(jobId)) >= 0)
448 m_progress.insert(jobId, qBound(0U, newProgress, 100U));
449 emit dataChanged(createIndex(index, 2), createIndex(index, 2));
453 void JobListModel::updateDetails(const QUuid &jobId, const QString &details)
457 if((index = m_jobs.indexOf(jobId)) >= 0)
459 m_details.insert(jobId, details);
460 emit dataChanged(createIndex(index, 3), createIndex(index, 3));