2 #include "playlist/PrettyItemDelegate.h"
3 #include "networkaccess.h"
4 #include "videowidget.h"
5 #include "minisplitter.h"
7 #include "downloadmanager.h"
8 #include "downloaditem.h"
9 #include "MainWindow.h"
12 NetworkAccess* http();
16 QMap<QString, QAction*>* globalActions();
17 QMap<QString, QMenu*>* globalMenus();
18 QNetworkAccessManager* networkAccessManager();
21 MediaView::MediaView(QWidget *parent) : QWidget(parent) {
23 reallyStopped = false;
26 QBoxLayout *layout = new QHBoxLayout();
29 splitter = new MiniSplitter(this);
30 splitter->setChildrenCollapsible(false);
32 sortBar = new THBlackBar(this);
33 mostRelevantAction = new QAction(tr("Most relevant"), this);
34 QKeySequence keySequence(Qt::CTRL + Qt::Key_1);
35 mostRelevantAction->setShortcut(keySequence);
36 mostRelevantAction->setStatusTip(mostRelevantAction->text() + " (" + keySequence.toString(QKeySequence::NativeText) + ")");
37 addAction(mostRelevantAction);
38 connect(mostRelevantAction, SIGNAL(triggered()), this, SLOT(searchMostRelevant()), Qt::QueuedConnection);
39 sortBar->addAction(mostRelevantAction);
40 mostRecentAction = new QAction(tr("Most recent"), this);
41 keySequence = QKeySequence(Qt::CTRL + Qt::Key_2);
42 mostRecentAction->setShortcut(keySequence);
43 mostRecentAction->setStatusTip(mostRecentAction->text() + " (" + keySequence.toString(QKeySequence::NativeText) + ")");
44 addAction(mostRecentAction);
45 connect(mostRecentAction, SIGNAL(triggered()), this, SLOT(searchMostRecent()), Qt::QueuedConnection);
46 sortBar->addAction(mostRecentAction);
47 mostViewedAction = new QAction(tr("Most viewed"), this);
48 keySequence = QKeySequence(Qt::CTRL + Qt::Key_3);
49 mostViewedAction->setShortcut(keySequence);
50 mostViewedAction->setStatusTip(mostViewedAction->text() + " (" + keySequence.toString(QKeySequence::NativeText) + ")");
51 addAction(mostViewedAction);
52 connect(mostViewedAction, SIGNAL(triggered()), this, SLOT(searchMostViewed()), Qt::QueuedConnection);
53 sortBar->addAction(mostViewedAction);
55 listView = new QListView(this);
56 listView->setItemDelegate(new PrettyItemDelegate(this));
57 listView->setSelectionMode(QAbstractItemView::ExtendedSelection);
60 listView->setDragEnabled(true);
61 listView->setAcceptDrops(true);
62 listView->setDropIndicatorShown(true);
63 listView->setDragDropMode(QAbstractItemView::DragDrop);
66 listView->setVerticalScrollMode(QAbstractItemView::ScrollPerPixel);
67 listView->setFrameShape( QFrame::NoFrame );
68 listView->setAttribute(Qt::WA_MacShowFocusRect, false);
69 listView->setMinimumSize(320,240);
70 listView->setUniformItemSizes(true);
72 // respond to the user doubleclicking a playlist item
73 connect(listView, SIGNAL(activated(const QModelIndex &)), this, SLOT(itemActivated(const QModelIndex &)));
75 listModel = new ListModel(this);
76 connect(listModel, SIGNAL(activeRowChanged(int)), this, SLOT(activeRowChanged(int)));
77 // needed to restore the selection after dragndrop
78 connect(listModel, SIGNAL(needSelectionFor(QList<Video*>)), this, SLOT(selectVideos(QList<Video*>)));
79 listView->setModel(listModel);
81 connect(listView->selectionModel(),
82 SIGNAL(selectionChanged ( const QItemSelection & , const QItemSelection & )),
83 this, SLOT(selectionChanged ( const QItemSelection & , const QItemSelection & )));
85 playlistWidget = new PlaylistWidget(this, sortBar, listView);
87 splitter->addWidget(playlistWidget);
89 videoAreaWidget = new VideoAreaWidget(this);
90 videoAreaWidget->setMinimumSize(320,240);
93 // mouse autohide does not work on the Mac (no mouseMoveEvent)
94 videoWidget = new Phonon::VideoWidget(this);
96 videoWidget = new VideoWidget(this);
99 videoAreaWidget->setVideoWidget(videoWidget);
100 videoAreaWidget->setListModel(listModel);
102 loadingWidget = new LoadingWidget(this);
103 videoAreaWidget->setLoadingWidget(loadingWidget);
105 splitter->addWidget(videoAreaWidget);
107 layout->addWidget(splitter);
110 // restore splitter state
112 splitter->restoreState(settings.value("splitter").toByteArray());
114 errorTimer = new QTimer(this);
115 errorTimer->setSingleShot(true);
116 errorTimer->setInterval(3000);
117 connect(errorTimer, SIGNAL(timeout()), SLOT(skipVideo()));
119 workaroundTimer = new QTimer(this);
120 workaroundTimer->setSingleShot(true);
121 workaroundTimer->setInterval(3000);
122 connect(workaroundTimer, SIGNAL(timeout()), SLOT(timerPlay()));
125 demoTimer = new QTimer(this);
126 demoTimer->setSingleShot(true);
127 demoTimer->setInterval(60000);
128 connect(demoTimer, SIGNAL(timeout()), SLOT(demoMessage()));
133 void MediaView::initialize() {
134 connect(videoAreaWidget, SIGNAL(doubleClicked()), The::globalActions()->value("fullscreen"), SLOT(trigger()));
135 videoAreaWidget->setContextMenuPolicy(Qt::CustomContextMenu);
136 connect(videoAreaWidget, SIGNAL(customContextMenuRequested(QPoint)),
137 this, SLOT(showVideoContextMenu(QPoint)));
140 void MediaView::setMediaObject(Phonon::MediaObject *mediaObject) {
141 this->mediaObject = mediaObject;
142 Phonon::createPath(this->mediaObject, videoWidget);
143 connect(mediaObject, SIGNAL(finished()), this, SLOT(playbackFinished()));
144 connect(mediaObject, SIGNAL(stateChanged(Phonon::State, Phonon::State)),
145 this, SLOT(stateChanged(Phonon::State, Phonon::State)));
146 connect(mediaObject, SIGNAL(currentSourceChanged(Phonon::MediaSource)),
147 this, SLOT(currentSourceChanged(Phonon::MediaSource)));
148 // connect(mediaObject, SIGNAL(bufferStatus(int)), loadingWidget, SLOT(bufferStatus(int)));
151 void MediaView::search(SearchParams *searchParams) {
152 reallyStopped = false;
158 videoAreaWidget->clear();
159 workaroundTimer->stop();
162 mediaObject->pause();
168 this->searchParams = searchParams;
170 // start serching for videos
171 listModel->search(searchParams);
173 // this implies that the enum and the bar action order is the same
174 sortBar->setCheckedAction(searchParams->sortBy()-1);
176 listView->setFocus();
179 QString keyword = searchParams->keywords();
180 QString display = keyword;
181 if (keyword.startsWith("http://")) {
182 int separator = keyword.indexOf("|");
183 if (separator > 0 && separator + 1 < keyword.length()) {
184 display = keyword.mid(separator+1);
188 playlistWidget->hide();
189 } else playlistWidget->show();
190 // tr("You're watching \"%1\"").arg(searchParams->keywords())
194 void MediaView::disappear() {
195 timerPlayFlag = true;
198 void MediaView::handleError(QString message) {
199 videoAreaWidget->showError(message);
200 skippedVideo = listModel->activeVideo();
201 // recover from errors by skipping to the next video
202 errorTimer->start(2000);
205 void MediaView::stateChanged(Phonon::State newState, Phonon::State /*oldState*/)
208 // qDebug() << "Phonon state: " << newState << oldState;
209 // slider->setEnabled(newState == Phonon::PlayingState);
213 case Phonon::ErrorState:
214 qDebug() << "Phonon error:" << mediaObject->errorString() << mediaObject->errorType();
215 if (mediaObject->errorType() == Phonon::FatalError)
216 handleError(mediaObject->errorString());
219 case Phonon::PlayingState:
220 // qDebug("playing");
221 videoAreaWidget->showVideo();
224 case Phonon::StoppedState:
225 // qDebug("stopped");
226 // play() has already been called when setting the source
227 // but Phonon on Linux needs a little more help to start playback
228 // if (!reallyStopped) mediaObject->play();
231 // Workaround for Mac playback start problem
232 if (!timerPlayFlag) {
233 // workaroundTimer->start();
239 case Phonon::PausedState:
243 case Phonon::BufferingState:
247 case Phonon::LoadingState:
256 void MediaView::pause() {
257 // qDebug() << "pause() called" << mediaObject->state();
258 switch( mediaObject->state() ) {
259 case Phonon::PlayingState:
260 mediaObject->pause();
268 void MediaView::stop() {
269 listModel->abortSearch();
270 reallyStopped = true;
272 videoAreaWidget->clear();
273 workaroundTimer->stop();
275 listView->selectionModel()->clearSelection();
277 downloadItem->stop();
283 void MediaView::activeRowChanged(int row) {
284 if (reallyStopped) return;
286 Video *video = listModel->videoAt(row);
289 // now that we have a new video to play
290 // stop all the timers
291 workaroundTimer->stop();
294 mediaObject->pause();
296 downloadItem->stop();
300 // slider->setMinimum(0);
302 // immediately show the loading widget
303 videoAreaWidget->showLoading(video);
305 connect(video, SIGNAL(gotStreamUrl(QUrl)), SLOT(gotStreamUrl(QUrl)));
306 // TODO handle signal in a proper slot and impl item error status
307 connect(video, SIGNAL(errorStreamUrl(QString)), SLOT(handleError(QString)));
309 video->loadStreamUrl();
311 // reset the timer flag
312 timerPlayFlag = false;
314 // video title in the statusbar
315 QMainWindow* mainWindow = dynamic_cast<QMainWindow*>(window());
316 if (mainWindow) mainWindow->statusBar()->showMessage(video->title());
318 The::globalActions()->value("download")->setEnabled(DownloadManager::instance()->itemForVideo(video) == 0);
320 // see you in gotStreamUrl...
324 void MediaView::gotStreamUrl(QUrl streamUrl) {
325 if (reallyStopped) return;
327 Video *video = static_cast<Video *>(sender());
329 qDebug() << "Cannot get sender";
332 video->disconnect(this);
335 QString tempDir = QDesktopServices::storageLocation(QDesktopServices::TempLocation);
337 QString tempFile = tempDir + "/minitube-" + getenv("USERNAME") + ".mp4";
339 QString tempFile = tempDir + "/minitube.mp4";
341 if (QFile::exists(tempFile) && !QFile::remove(tempFile)) {
342 qDebug() << "Cannot remove temp file";
345 Video *videoCopy = video->clone();
347 downloadItem->stop();
350 downloadItem = new DownloadItem(videoCopy, streamUrl, tempFile, this);
351 connect(downloadItem, SIGNAL(statusChanged()), SLOT(downloadStatusChanged()));
352 // connect(downloadItem, SIGNAL(progress(int)), SLOT(downloadProgress(int)));
353 connect(downloadItem, SIGNAL(bufferProgress(int)), loadingWidget, SLOT(bufferStatus(int)));
354 // connect(downloadItem, SIGNAL(finished()), SLOT(itemFinished()));
355 connect(video, SIGNAL(errorStreamUrl(QString)), SLOT(handleError(QString)));
356 connect(downloadItem, SIGNAL(error(QString)), SLOT(handleError(QString)));
357 downloadItem->start();
362 void MediaView::downloadProgress(int percent) {
363 MainWindow* mainWindow = dynamic_cast<MainWindow*>(window());
365 mainWindow->getSeekSlider()->setStyleSheet(" QSlider::groove:horizontal {"
366 "border: 1px solid #999999;"
367 // "border-left: 50px solid rgba(255, 0, 0, 128);"
369 "background: qlineargradient(x1:0, y1:0, x2:.5, y2:0, stop:0 rgba(255, 0, 0, 92), stop:"
370 + QString::number(percent/100.0) +
372 " rgba(255, 0, 0, 92), stop:" + QString::number((percent+1)/100.0) + " transparent, stop:1 transparent);"
375 "QSlider::handle:horizontal {"
376 "background: qlineargradient(x1:0, y1:0, x2:1, y2:1, stop:0 #b4b4b4, stop:1 #8f8f8f);"
377 "border: 1px solid #5c5c5c;"
381 "border-radius: 8px;"
389 void MediaView::downloadStatusChanged() {
390 switch(downloadItem->status()) {
395 // qDebug() << "Starting";
398 // qDebug() << "Finished" << mediaObject->state();
399 // if (mediaObject->state() == Phonon::StoppedState) startPlaying();
402 // qDebug() << "Failed";
404 // qDebug() << "Idle";
409 void MediaView::startPlaying() {
410 if (reallyStopped) return;
413 qDebug() << "Playing" << downloadItem->currentFilename();
414 mediaObject->setCurrentSource(downloadItem->currentFilename());
417 // ensure we always have 10 videos ahead
418 listModel->searchNeeded();
420 // ensure active item is visible
421 int row = listModel->activeRow();
423 QModelIndex index = listModel->index(row, 0, QModelIndex());
424 listView->scrollTo(index, QAbstractItemView::EnsureVisible);
428 demoTimer->start(30000);
433 void MediaView::itemActivated(const QModelIndex &index) {
434 if (listModel->rowExists(index.row()))
435 listModel->setActiveRow(index.row());
436 // the user doubleclicked on the "Search More" item
437 else listModel->searchMore();
440 void MediaView::currentSourceChanged(const Phonon::MediaSource /* source */ ) {
444 void MediaView::skipVideo() {
445 // skippedVideo is useful for DELAYED skip operations
446 // in order to be sure that we're skipping the video we wanted
447 // and not another one
449 if (listModel->activeVideo() != skippedVideo) {
450 qDebug() << "Skip of video canceled";
453 int nextRow = listModel->rowForVideo(skippedVideo);
455 if (nextRow == -1) return;
456 listModel->setActiveRow(nextRow);
460 void MediaView::skip() {
461 int nextRow = listModel->nextRow();
462 if (nextRow == -1) return;
463 listModel->setActiveRow(nextRow);
466 void MediaView::playbackFinished() {
467 // qDebug() << "finished" << mediaObject->currentTime() << mediaObject->totalTime();
468 // add 10 secs for imprecise Phonon backends (VLC, Xine)
469 if (mediaObject->currentTime() + 10000 < mediaObject->totalTime()) {
470 // mediaObject->seek(mediaObject->currentTime());
471 QTimer::singleShot(3000, this, SLOT(playbackResume()));
475 void MediaView::playbackResume() {
476 mediaObject->seek(mediaObject->currentTime());
480 void MediaView::openWebPage() {
481 Video* video = listModel->activeVideo();
483 mediaObject->pause();
484 QDesktopServices::openUrl(video->webpage());
487 void MediaView::copyWebPage() {
488 Video* video = listModel->activeVideo();
490 QString address = video->webpage().toString();
491 address.remove("&feature=youtube_gdata");
492 QApplication::clipboard()->setText(address);
493 QMainWindow* mainWindow = dynamic_cast<QMainWindow*>(window());
494 QString message = tr("You can now paste the YouTube link into another application");
495 if (mainWindow) mainWindow->statusBar()->showMessage(message);
498 void MediaView::copyVideoLink() {
499 Video* video = listModel->activeVideo();
501 QApplication::clipboard()->setText(video->getStreamUrl().toEncoded());
502 QString message = tr("You can now paste the video stream URL into another application")
503 + ". " + tr("The link will be valid only for a limited time.");
504 QMainWindow* mainWindow = dynamic_cast<QMainWindow*>(window());
505 if (mainWindow) mainWindow->statusBar()->showMessage(message);
508 void MediaView::removeSelected() {
509 if (!listView->selectionModel()->hasSelection()) return;
510 QModelIndexList indexes = listView->selectionModel()->selectedIndexes();
511 listModel->removeIndexes(indexes);
514 void MediaView::selectVideos(QList<Video*> videos) {
515 foreach (Video *video, videos) {
516 QModelIndex index = listModel->indexForVideo(video);
517 listView->selectionModel()->select(index, QItemSelectionModel::Select);
518 listView->scrollTo(index, QAbstractItemView::EnsureVisible);
522 void MediaView::selectionChanged(const QItemSelection & /*selected*/, const QItemSelection & /*deselected*/) {
523 const bool gotSelection = listView->selectionModel()->hasSelection();
524 The::globalActions()->value("remove")->setEnabled(gotSelection);
525 The::globalActions()->value("moveUp")->setEnabled(gotSelection);
526 The::globalActions()->value("moveDown")->setEnabled(gotSelection);
529 void MediaView::moveUpSelected() {
530 if (!listView->selectionModel()->hasSelection()) return;
532 QModelIndexList indexes = listView->selectionModel()->selectedIndexes();
533 qStableSort(indexes.begin(), indexes.end());
534 listModel->move(indexes, true);
536 // set current index after row moves to something more intuitive
537 int row = indexes.first().row();
538 listView->selectionModel()->setCurrentIndex(listModel->index(row>1?row:1), QItemSelectionModel::NoUpdate);
541 void MediaView::moveDownSelected() {
542 if (!listView->selectionModel()->hasSelection()) return;
544 QModelIndexList indexes = listView->selectionModel()->selectedIndexes();
545 qStableSort(indexes.begin(), indexes.end(), qGreater<QModelIndex>());
546 listModel->move(indexes, false);
548 // set current index after row moves to something more intuitive (respect 1 static item on bottom)
549 int row = indexes.first().row()+1, max = listModel->rowCount() - 2;
550 listView->selectionModel()->setCurrentIndex(listModel->index(row>max?max:row), QItemSelectionModel::NoUpdate);
553 void MediaView::showVideoContextMenu(QPoint point) {
554 The::globalMenus()->value("video")->popup(videoWidget->mapToGlobal(point));
557 void MediaView::searchMostRelevant() {
558 searchParams->setSortBy(SearchParams::SortByRelevance);
559 search(searchParams);
562 void MediaView::searchMostRecent() {
563 searchParams->setSortBy(SearchParams::SortByNewest);
564 search(searchParams);
567 void MediaView::searchMostViewed() {
568 searchParams->setSortBy(SearchParams::SortByViewCount);
569 search(searchParams);
572 void MediaView::setPlaylistVisible(bool visible) {
573 playlistWidget->setVisible(visible);
576 void MediaView::timerPlay() {
577 // Workaround Phonon bug on Mac OSX
578 // qDebug() << mediaObject->currentTime();
579 if (mediaObject->currentTime() <= 0 && mediaObject->state() == Phonon::PlayingState) {
580 // qDebug() << "Mac playback workaround";
581 mediaObject->pause();
582 // QTimer::singleShot(1000, mediaObject, SLOT(play()));
587 void MediaView::saveSplitterState() {
589 settings.setValue("splitter", splitter->saveState());
593 void MediaView::demoMessage() {
594 if (mediaObject->state() != Phonon::PlayingState) return;
595 mediaObject->pause();
598 msgBox.setIconPixmap(QPixmap(":/images/app.png").scaled(64, 64, Qt::KeepAspectRatio, Qt::SmoothTransformation));
599 msgBox.setText(tr("This is just the demo version of %1.").arg(Constants::APP_NAME));
600 msgBox.setInformativeText(tr("It allows you to test the application and see if it works for you."));
601 msgBox.setModal(true);
603 QPushButton *quitButton = msgBox.addButton(tr("Continue"), QMessageBox::RejectRole);
604 QPushButton *buyButton = msgBox.addButton(tr("Get the full version"), QMessageBox::ActionRole);
608 if (msgBox.clickedButton() == buyButton) {
609 QDesktopServices::openUrl(QString(Constants::WEBSITE) + "#download");
612 demoTimer->start(300000);
617 void MediaView::downloadVideo() {
618 Video* video = listModel->activeVideo();
621 DownloadManager::instance()->addItem(video);
625 The::globalActions()->value("downloads")->setVisible(true);
627 // The::globalActions()->value("download")->setEnabled(DownloadManager::instance()->itemForVideo(video) == 0);
629 QMainWindow* mainWindow = dynamic_cast<QMainWindow*>(window());
630 QString message = tr("Downloading %1").arg(video->title());
631 if (mainWindow) mainWindow->statusBar()->showMessage(message);
634 void MediaView::fullscreen() {
635 videoAreaWidget->setParent(0);
636 videoAreaWidget->showFullScreen();
640 void MediaView::setSlider(QSlider *slider) {
641 this->slider = slider;
642 // slider->setEnabled(false);
643 slider->setTracking(false);
644 // connect(slider, SIGNAL(valueChanged(int)), SLOT(sliderMoved(int)));
647 void MediaView::sliderMoved(int value) {
648 qDebug() << __func__;
649 int sliderPercent = (value * 100) / (slider->maximum() - slider->minimum());
650 qDebug() << slider->minimum() << value << slider->maximum();
651 if (sliderPercent <= downloadItem->currentPercent()) {
652 qDebug() << sliderPercent << downloadItem->currentPercent();
653 mediaObject->seek(value);
659 void MediaView::seekTo(int value) {
660 qDebug() << __func__;
661 mediaObject->pause();
662 workaroundTimer->stop();
664 // mediaObject->clear();
666 QString tempDir = QDesktopServices::storageLocation(QDesktopServices::TempLocation);
667 QString tempFile = tempDir + "/minitube" + QString::number(value) + ".mp4";
668 if (!QFile::remove(tempFile)) {
669 qDebug() << "Cannot remove temp file";
671 Video *videoCopy = downloadItem->getVideo()->clone();
672 QUrl streamUrl = videoCopy->getStreamUrl();
673 streamUrl.addQueryItem("begin", QString::number(value));
674 if (downloadItem) delete downloadItem;
675 downloadItem = new DownloadItem(videoCopy, streamUrl, tempFile, this);
676 connect(downloadItem, SIGNAL(statusChanged()), SLOT(downloadStatusChanged()));
677 // connect(downloadItem, SIGNAL(finished()), SLOT(itemFinished()));
678 downloadItem->start();
680 // slider->setMinimum(value);