// vim: set tabstop=4 shiftwidth=4 expandtab: /* Gwenview: an image viewer Copyright 2009 Aurélien Gâteau 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, Cambridge, MA 02110-1301, USA. */ // Self #include "videoviewadapter.moc" // Qt #include #include #include #include #include #include #include #include #include #include // KDE #include #include #include // Local #include #include #include #include #include namespace Gwenview { struct VideoViewAdapterPrivate { VideoViewAdapter* q; Phonon::MediaObject* mMediaObject; Phonon::VideoWidget* mVideoWidget; Phonon::AudioOutput* mAudioOutput; HudWidget* mHud; GraphicsWidgetFloater* mFloater; HudSlider* mSeekSlider; QTime mLastSeekSliderActionTime; QAction* mPlayPauseAction; QAction* mMuteAction; HudSlider* mVolumeSlider; QTime mLastVolumeSliderChangeTime; Document::Ptr mDocument; void setupActions() { mPlayPauseAction = new QAction(q); mPlayPauseAction->setShortcut(Qt::Key_P); QObject::connect(mPlayPauseAction, SIGNAL(triggered()), q, SLOT(slotPlayPauseClicked())); QObject::connect(mMediaObject, SIGNAL(stateChanged(Phonon::State,Phonon::State)), q, SLOT(updatePlayUi())); mMuteAction = new QAction(q); mMuteAction->setShortcut(Qt::Key_M); QObject::connect(mMuteAction, SIGNAL(triggered()), q, SLOT(slotMuteClicked())); QObject::connect(mAudioOutput, SIGNAL(mutedChanged(bool)), q, SLOT(updateMuteAction())); } void setupHud(QGraphicsWidget* parent) { // Play/Pause HudButton* playPauseButton = new HudButton; playPauseButton->setDefaultAction(mPlayPauseAction); // Seek mSeekSlider = new HudSlider; mSeekSlider->setPageStep(5000); mSeekSlider->setSingleStep(200); QObject::connect(mSeekSlider, SIGNAL(actionTriggered(int)), q, SLOT(slotSeekSliderActionTriggered(int))); QObject::connect(mMediaObject, SIGNAL(tick(qint64)), q, SLOT(slotTicked(qint64))); QObject::connect(mMediaObject, SIGNAL(totalTimeChanged(qint64)), q, SLOT(updatePlayUi())); QObject::connect(mMediaObject, SIGNAL(seekableChanged(bool)), q, SLOT(updatePlayUi())); // Mute HudButton* muteButton = new HudButton; muteButton->setDefaultAction(mMuteAction); // Volume mVolumeSlider = new HudSlider; mVolumeSlider->setMinimumWidth(100); mVolumeSlider->setRange(0, 100); mVolumeSlider->setPageStep(5); mVolumeSlider->setSingleStep(1); QObject::connect(mVolumeSlider, SIGNAL(valueChanged(int)), q, SLOT(slotVolumeSliderChanged(int))); QObject::connect(mAudioOutput, SIGNAL(volumeChanged(qreal)), q, SLOT(slotOutputVolumeChanged(qreal))); // Layout QGraphicsWidget* hudContent = new QGraphicsWidget; QGraphicsLinearLayout* layout = new QGraphicsLinearLayout(hudContent); layout->addItem(playPauseButton); layout->addItem(mSeekSlider); layout->setStretchFactor(mSeekSlider, 5); layout->addItem(muteButton); layout->addItem(mVolumeSlider); layout->setStretchFactor(mVolumeSlider, 1); // Create hud mHud = new HudWidget(parent); mHud->init(hudContent, HudWidget::OptionNone); mHud->setZValue(1); // Init floater mFloater = new GraphicsWidgetFloater(parent); mFloater->setChildWidget(mHud); mFloater->setAlignment(Qt::AlignJustify | Qt::AlignBottom); } bool isPlaying() const { switch (mMediaObject->state()) { case Phonon::PlayingState: case Phonon::BufferingState: return true; default: return false; } } void updateHudVisibility(int yPos) { const int floaterY = mVideoWidget->height() - mFloater->verticalMargin() - mHud->effectiveSizeHint(Qt::MinimumSize).height() * 3 / 2; if (yPos < floaterY) { mHud->fadeOut(); } else { mHud->fadeIn(); } } void keyPressEvent(QKeyEvent* event) { if (event->modifiers() != Qt::NoModifier) { return; } switch (event->key()) { case Qt::Key_Left: mSeekSlider->triggerAction(QAbstractSlider::SliderSingleStepSub); break; case Qt::Key_Right: mSeekSlider->triggerAction(QAbstractSlider::SliderSingleStepAdd); break; case Qt::Key_Up: q->previousImageRequested(); break; case Qt::Key_Down: q->nextImageRequested(); break; default: break; } } }; /** * This is a workaround for a bug in QGraphicsProxyWidget: it does not forward * double-click events to the proxy-fied widget. * * QGraphicsProxyWidget::mouseDoubleClickEvent() correctly forwards the event * to its QWidget, but it is never called. This is because for it to be called, * the implementation of mousePressEvent() must call * QGraphicsItem::mousePressEvent() but it does not. */ class DoubleClickableProxyWidget : public QGraphicsProxyWidget { protected: void mousePressEvent(QGraphicsSceneMouseEvent* event) { QGraphicsWidget::mousePressEvent(event); } }; VideoViewAdapter::VideoViewAdapter() : d(new VideoViewAdapterPrivate) { d->q = this; d->mMediaObject = new Phonon::MediaObject(this); d->mMediaObject->setTickInterval(350); connect(d->mMediaObject, SIGNAL(finished()), SIGNAL(videoFinished())); d->mVideoWidget = new Phonon::VideoWidget; d->mVideoWidget->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); d->mVideoWidget->setAttribute(Qt::WA_Hover); d->mVideoWidget->installEventFilter(this); Phonon::createPath(d->mMediaObject, d->mVideoWidget); d->mAudioOutput = new Phonon::AudioOutput(Phonon::VideoCategory, this); Phonon::createPath(d->mMediaObject, d->mAudioOutput); QGraphicsProxyWidget* proxy = new DoubleClickableProxyWidget; proxy->setFlag(QGraphicsItem::ItemIsSelectable); // Needed for doubleclick to work proxy->setWidget(d->mVideoWidget); proxy->setAcceptHoverEvents(true); setWidget(proxy); d->setupActions(); d->setupHud(proxy); updatePlayUi(); updateMuteAction(); } VideoViewAdapter::~VideoViewAdapter() { // This prevents a memory leak that can occur after switching // to the next/previous video. For details see: // https://git.reviewboard.kde.org/r/108070/ d->mMediaObject->stop(); delete d; } void VideoViewAdapter::setDocument(Document::Ptr doc) { d->mHud->show(); d->mDocument = doc; d->mMediaObject->setCurrentSource(d->mDocument->url()); d->mMediaObject->play(); // If we do not use a queued connection, the signal arrives too early, // preventing the listing of the dir content when Gwenview is started with // a video as an argument. QMetaObject::invokeMethod(this, "completed", Qt::QueuedConnection); } Document::Ptr VideoViewAdapter::document() const { return d->mDocument; } void VideoViewAdapter::slotPlayPauseClicked() { if (d->isPlaying()) { d->mMediaObject->pause(); } else { d->mMediaObject->play(); } } void VideoViewAdapter::slotMuteClicked() { d->mAudioOutput->setMuted(!d->mAudioOutput->isMuted()); } bool VideoViewAdapter::eventFilter(QObject*, QEvent* event) { if (event->type() == QEvent::MouseMove) { d->updateHudVisibility(static_cast(event)->y()); } else if (event->type() == QEvent::KeyPress) { d->keyPressEvent(static_cast(event)); } else if (event->type() == QEvent::MouseButtonDblClick) { if (static_cast(event)->modifiers() == Qt::NoModifier) { toggleFullScreenRequested(); } } return false; } void VideoViewAdapter::updatePlayUi() { if (d->isPlaying()) { d->mPlayPauseAction->setIcon(KIcon("media-playback-pause")); } else { d->mPlayPauseAction->setIcon(KIcon("media-playback-start")); } d->mLastSeekSliderActionTime.restart(); d->mSeekSlider->setRange(0, d->mMediaObject->totalTime()); switch (d->mMediaObject->state()) { case Phonon::PlayingState: case Phonon::BufferingState: case Phonon::PausedState: d->mSeekSlider->setEnabled(true); break; case Phonon::StoppedState: case Phonon::LoadingState: case Phonon::ErrorState: d->mSeekSlider->setEnabled(false); d->mSeekSlider->setValue(0); break; } } void VideoViewAdapter::updateMuteAction() { d->mMuteAction->setIcon( KIcon(d->mAudioOutput->isMuted() ? "player-volume-muted" : "player-volume") ); } void VideoViewAdapter::slotVolumeSliderChanged(int value) { d->mLastVolumeSliderChangeTime.restart(); d->mAudioOutput->setVolume(value / 100.); } void VideoViewAdapter::slotOutputVolumeChanged(qreal value) { if (d->mLastVolumeSliderChangeTime.isValid() && d->mLastVolumeSliderChangeTime.elapsed() < 2000) { return; } d->mVolumeSlider->setValue(qRound(value * 100)); } void VideoViewAdapter::slotSeekSliderActionTriggered(int /*action*/) { d->mLastSeekSliderActionTime.restart(); d->mMediaObject->seek(d->mSeekSlider->sliderPosition()); } void VideoViewAdapter::slotTicked(qint64 value) { if (d->mLastSeekSliderActionTime.isValid() && d->mLastSeekSliderActionTime.elapsed() < 2000) { return; } if (!d->mSeekSlider->isSliderDown()) { d->mSeekSlider->setValue(value); } } } // namespace