307 lines
7.5 KiB
C++

/*
Gwenview: an image viewer
Copyright 2007 Aurélien Gâteau <agateau@kde.org>
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include "slideshow.h"
// STL
#include <algorithm>
#include <ctime>
#include <random>
// Qt
#include <QAction>
#include <QTimer>
// KF
#include <KLocalizedString>
// Local
#include "gwenview_lib_debug.h"
#include <gwenviewconfig.h>
#include <lib/gvdebug.h>
namespace Gwenview
{
#undef ENABLE_LOG
#undef LOG
// #define ENABLE_LOG
#ifdef ENABLE_LOG
#define LOG(x) qCDebug(GWENVIEW_LIB_LOG) << x
#else
#define LOG(x) ;
#endif
enum State {
Paused,
Started,
WaitForEndOfUrl,
};
struct SlideShowPrivate {
QTimer *mTimer = nullptr;
State mState;
QVector<QUrl> mUrls;
QVector<QUrl> mShuffledUrls;
QVector<QUrl>::ConstIterator mStartIt;
QUrl mCurrentUrl;
QUrl mLastShuffledUrl;
QAction *mLoopAction = nullptr;
QAction *mRandomAction = nullptr;
QUrl findNextUrl()
{
if (GwenviewConfig::random()) {
return findNextRandomUrl();
} else {
return findNextOrderedUrl();
}
}
QUrl findNextOrderedUrl()
{
QVector<QUrl>::ConstIterator it = std::find(mUrls.constBegin(), mUrls.constEnd(), mCurrentUrl);
GV_RETURN_VALUE_IF_FAIL2(it != mUrls.constEnd(), QUrl(), "Current url not found in list.");
++it;
if (GwenviewConfig::loop()) {
// Looping, if we reach the end, start again
if (it == mUrls.constEnd()) {
it = mUrls.constBegin();
}
} else {
// Not looping, have we reached the end?
// FIXME: stopAtEnd
if (/*(it==mUrls.end() && GwenviewConfig::stopAtEnd()) ||*/ it == mStartIt) {
it = mUrls.constEnd();
}
}
if (it != mUrls.constEnd()) {
return *it;
} else {
return {};
}
}
void initShuffledUrls()
{
mShuffledUrls = mUrls;
std::random_device rd;
std::mt19937 generator(rd());
std::shuffle(mShuffledUrls.begin(), mShuffledUrls.end(), generator);
// Ensure the first url is different from the previous last one, so that
// last url does not stay visible twice longer than usual
if (mLastShuffledUrl == mShuffledUrls.first() && mShuffledUrls.count() > 1) {
std::swap(mShuffledUrls[0], mShuffledUrls[1]);
}
mLastShuffledUrl = mShuffledUrls.last();
}
QUrl findNextRandomUrl()
{
if (mShuffledUrls.empty()) {
if (GwenviewConfig::loop()) {
initShuffledUrls();
} else {
return {};
}
}
const QUrl url = mShuffledUrls.last();
mShuffledUrls.pop_back();
return url;
}
void updateTimerInterval()
{
mTimer->setInterval(int(GwenviewConfig::interval() * 1000));
}
void doStart()
{
if (MimeTypeUtils::urlKind(mCurrentUrl) == MimeTypeUtils::KIND_VIDEO) {
LOG("mState = WaitForEndOfUrl");
// Just in case
mTimer->stop();
mState = WaitForEndOfUrl;
} else {
LOG("mState = Started");
mTimer->start();
mState = Started;
}
}
};
SlideShow::SlideShow(QObject *parent)
: QObject(parent)
, d(new SlideShowPrivate)
{
d->mState = Paused;
d->mTimer = new QTimer(this);
connect(d->mTimer, &QTimer::timeout, this, &SlideShow::goToNextUrl);
d->mLoopAction = new QAction(this);
d->mLoopAction->setText(i18nc("@item:inmenu toggle loop in slideshow", "Loop"));
d->mLoopAction->setCheckable(true);
connect(d->mLoopAction, &QAction::triggered, this, &SlideShow::updateConfig);
d->mRandomAction = new QAction(this);
d->mRandomAction->setText(i18nc("@item:inmenu toggle random order in slideshow", "Random"));
d->mRandomAction->setCheckable(true);
connect(d->mRandomAction, &QAction::toggled, this, &SlideShow::slotRandomActionToggled);
connect(d->mRandomAction, &QAction::triggered, this, &SlideShow::updateConfig);
d->mLoopAction->setChecked(GwenviewConfig::loop());
d->mRandomAction->setChecked(GwenviewConfig::random());
}
SlideShow::~SlideShow()
{
GwenviewConfig::self()->save();
delete d;
}
QAction *SlideShow::loopAction() const
{
return d->mLoopAction;
}
QAction *SlideShow::randomAction() const
{
return d->mRandomAction;
}
void SlideShow::start(const QList<QUrl> &urls)
{
d->mUrls.resize(urls.size());
std::copy(urls.begin(), urls.end(), d->mUrls.begin());
d->mStartIt = std::find(d->mUrls.constBegin(), d->mUrls.constEnd(), d->mCurrentUrl);
if (d->mStartIt == d->mUrls.constEnd()) {
qCWarning(GWENVIEW_LIB_LOG) << "Current url not found in list, aborting.\n";
return;
}
if (GwenviewConfig::random()) {
d->initShuffledUrls();
}
d->updateTimerInterval();
d->mTimer->setSingleShot(false);
d->doStart();
Q_EMIT stateChanged(true);
}
void SlideShow::setInterval(int intervalInSeconds)
{
GwenviewConfig::setInterval(double(intervalInSeconds));
d->updateTimerInterval();
Q_EMIT intervalChanged(intervalInSeconds);
}
int SlideShow::interval() const
{
return GwenviewConfig::interval();
}
int SlideShow::position() const
{
// TODO: also support videos
// QTimer::remainingTime() returns -1 if inactive
// and there are moments where mState == Started but timer already done but not yet next url reached
// so handle that
if (d->mState == Started) {
if (d->mTimer->isActive()) {
return interval() * 1000 - d->mTimer->remainingTime();
}
// already timeout reached, but not yet progressed to next url
return interval();
}
return 0;
}
void SlideShow::pause()
{
LOG("Stopping timer");
d->mTimer->stop();
d->mState = Paused;
Q_EMIT stateChanged(false);
}
void SlideShow::resumeAndGoToNextUrl()
{
LOG("");
if (d->mState == WaitForEndOfUrl) {
goToNextUrl();
}
}
void SlideShow::goToNextUrl()
{
LOG("");
const QUrl url = d->findNextUrl();
LOG("url:" << url);
if (!url.isValid()) {
pause();
return;
}
Q_EMIT goToUrl(url);
}
void SlideShow::setCurrentUrl(const QUrl &url)
{
LOG(url);
if (d->mCurrentUrl == url) {
return;
}
d->mCurrentUrl = url;
// Restart timer to avoid showing new url for the remaining time of the old
// url
if (d->mState != Paused) {
d->doStart();
}
}
bool SlideShow::isRunning() const
{
return d->mState != Paused;
}
void SlideShow::updateConfig()
{
GwenviewConfig::setLoop(d->mLoopAction->isChecked());
GwenviewConfig::setRandom(d->mRandomAction->isChecked());
}
void SlideShow::slotRandomActionToggled(bool on)
{
if (on && d->mState != Paused) {
d->initShuffledUrls();
}
}
} // namespace
#include "moc_slideshow.cpp"