/* This file is part of Clementine. Copyright 2010, David Sansome Clementine 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 3 of the License, or (at your option) any later version. Clementine 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 Clementine. If not, see . */ #include "playlist.h" #include "playlistmanager.h" #include "playlisttabbar.h" #include "playlistview.h" #include "songmimedata.h" #include "core/logging.h" #include "internet/internetmimedata.h" #include "ui/iconloader.h" #include "widgets/renametablineedit.h" #include #include #include #include #include PlaylistTabBar::PlaylistTabBar(QWidget *parent) : QTabBar(parent), manager_(NULL), menu_(new QMenu(this)), menu_index_(-1), suppress_current_changed_(false), rename_editor_(new RenameTabLineEdit(this)), removing_with_confirm_(true) { setAcceptDrops(true); setElideMode(Qt::ElideRight); setUsesScrollButtons(true); setTabsClosable(true); remove_ = menu_->addAction(IconLoader::Load("list-remove"), tr("Remove playlist"), this, SLOT(Remove())); rename_ = menu_->addAction(IconLoader::Load("edit-rename"), tr("Rename playlist..."), this, SLOT(Rename())); save_ = menu_->addAction(IconLoader::Load("document-save"), tr("Save playlist..."), this, SLOT(Save())); menu_->addSeparator(); rename_editor_->setVisible(false); connect(rename_editor_, SIGNAL(editingFinished()), SLOT(RenameInline())); connect(rename_editor_, SIGNAL(EditingCanceled()), SLOT(HideEditor())); connect(this, SIGNAL(currentChanged(int)), SLOT(CurrentIndexChanged(int))); connect(this, SIGNAL(tabMoved(int,int)), SLOT(TabMoved())); // We can't just emit Remove signal, we need to extract the playlist id first connect(this, SIGNAL(tabCloseRequested(int)), SLOT(RemoveFromTabIndex(int))); } void PlaylistTabBar::SetActions( QAction* new_playlist, QAction* load_playlist) { menu_->insertAction(0, new_playlist); menu_->insertAction(0, load_playlist); new_ = new_playlist; } void PlaylistTabBar::SetManager(PlaylistManager *manager) { manager_ = manager; } void PlaylistTabBar::contextMenuEvent(QContextMenuEvent* e) { //we need to finish the renaming action before showing context menu if (rename_editor_->isVisible()) { //discard any change HideEditor(); } menu_index_ = tabAt(e->pos()); rename_->setEnabled(menu_index_ != -1); remove_->setEnabled(menu_index_ != -1 && count() > 1); save_->setEnabled(menu_index_ != -1); menu_->popup(e->globalPos()); } void PlaylistTabBar::mouseReleaseEvent(QMouseEvent* e) { if (e->button() == Qt::MidButton) { // Update menu index menu_index_ = tabAt(e->pos()); // So we don't ask for confirmation removing_with_confirm_ = false; Remove(); } QTabBar::mouseReleaseEvent(e); } void PlaylistTabBar::mouseDoubleClickEvent(QMouseEvent *e) { int index = tabAt(e->pos()); //discard a double click with the middle button if (e->button() != Qt::MidButton) { if (index == -1) { new_->activate(QAction::Trigger); } else { //update current tab menu_index_ = index; //set position rename_editor_->setGeometry(tabRect(index)); rename_editor_->setText(tabText(index)); rename_editor_->setVisible(true); rename_editor_->setFocus(); } } QTabBar::mouseDoubleClickEvent(e); } void PlaylistTabBar::Rename() { if (menu_index_ == -1) return; QString name = tabText(menu_index_); name = QInputDialog::getText( this, tr("Rename playlist"), tr("Enter a new name for this playlist"), QLineEdit::Normal, name); if (name.isNull()) return; emit Rename(tabData(menu_index_).toInt(), name); } void PlaylistTabBar::RenameInline() { emit Rename(tabData(menu_index_).toInt(), rename_editor_->text()); HideEditor(); } void PlaylistTabBar::HideEditor() { //editingFinished() will be called twice due to Qt bug #40, so we reuse the same instance, don't delete it rename_editor_->setVisible(false); //hack to give back focus to playlist view manager_->SetCurrentPlaylist(manager_->current()->id()); } void PlaylistTabBar::Remove() { if (menu_index_ == -1) return; // Ask for confirmation if not middle clicking the tab if (removing_with_confirm_ && QMessageBox::question(this, tr("Remove playlist"), tr("This playlist will be removed; the action can't be undone. " "Are you sure you want to continue?"), QMessageBox::Yes, QMessageBox::Cancel) != QMessageBox::Yes) return; removing_with_confirm_ = true; emit Remove(tabData(menu_index_).toInt()); } void PlaylistTabBar::RemoveFromTabIndex(int index) { // Update the global index menu_index_ = index; // Ask for confirmation removing_with_confirm_ = true; Remove(); } void PlaylistTabBar::Save() { if (menu_index_ == -1) return; emit Save(tabData(menu_index_).toInt()); } int PlaylistTabBar::current_id() const { if (currentIndex() == -1) return -1; return tabData(currentIndex()).toInt(); } int PlaylistTabBar::index_of(int id) const { for (int i=0 ; i= count()) { qLog(Warning) << "Playlist tab index requested is out of bounds!"; return 0; } return tabData(index).toInt(); } void PlaylistTabBar::set_icon_by_id(int id, const QIcon &icon) { setTabIcon(index_of(id), icon); } void PlaylistTabBar::RemoveTab(int id) { removeTab(index_of(id)); } void PlaylistTabBar::set_text_by_id(int id, const QString &text) { setTabText(index_of(id), text); setTabToolTip(index_of(id), text); } void PlaylistTabBar::CurrentIndexChanged(int index) { if (!suppress_current_changed_) emit CurrentIdChanged(tabData(index).toInt()); } void PlaylistTabBar::InsertTab(int id, int index, const QString& text) { suppress_current_changed_ = true; insertTab(index, text); setTabData(index, id); setTabToolTip(index, text); suppress_current_changed_ = false; if (currentIndex() == index) emit CurrentIdChanged(id); } void PlaylistTabBar::TabMoved() { QList ids; for (int i=0 ; imimeData()->hasUrls() || e->mimeData()->hasFormat(Playlist::kRowsMimetype) || qobject_cast(e->mimeData())) { e->acceptProposedAction(); } } void PlaylistTabBar::dragMoveEvent(QDragMoveEvent* e) { drag_hover_tab_ = tabAt(e->pos()); if (drag_hover_tab_ != -1) { e->setDropAction(Qt::CopyAction); e->accept(tabRect(drag_hover_tab_)); if (!drag_hover_timer_.isActive()) drag_hover_timer_.start(kDragHoverTimeout, this); } else { drag_hover_timer_.stop(); } } void PlaylistTabBar::dragLeaveEvent(QDragLeaveEvent*) { drag_hover_timer_.stop(); } void PlaylistTabBar::timerEvent(QTimerEvent* e) { QTabBar::timerEvent(e); if (e->timerId() == drag_hover_timer_.timerId()) { drag_hover_timer_.stop(); if (drag_hover_tab_ != -1) setCurrentIndex(drag_hover_tab_); } } void PlaylistTabBar::dropEvent(QDropEvent* e) { if (drag_hover_tab_ == -1) { const MimeData *mime_data = qobject_cast(e->mimeData()); if(mime_data && !mime_data->name_for_new_playlist_.isEmpty()) { manager_->New(mime_data->name_for_new_playlist_); } else { manager_->New(tr("Playlist")); } setCurrentIndex(count() - 1); } else { setCurrentIndex(drag_hover_tab_); } manager_->current()->dropMimeData(e->mimeData(), e->proposedAction(), -1, 0, QModelIndex()); } bool PlaylistTabBar::event(QEvent* e) { switch (e->type()) { case QEvent::ToolTip: { QHelpEvent *he = static_cast(e); QRect displayed_tab; QSize real_tab; bool is_elided = false; real_tab = tabSizeHint(tabAt(he->pos())); displayed_tab = tabRect(tabAt(he->pos())); // Check whether the tab is elided or not is_elided = displayed_tab.width() < real_tab.width(); if(!is_elided) { // If it's not elided, don't show the tooltip QToolTip::hideText(); } else { QToolTip::showText(he->globalPos(), tabToolTip(tabAt(he->pos()))); } return true; } default: return QTabBar::event(e); } }