Make the Windows 7 thumbbar actually work. Move it to a different class that deals with QActions. Fixes issue #851

This commit is contained in:
David Sansome 2011-01-16 00:39:51 +00:00
parent da2ab82712
commit 2c8e038b44
5 changed files with 218 additions and 57 deletions

View File

@ -205,6 +205,7 @@ set(SOURCES
ui/screensaver.cpp
ui/settingsdialog.cpp
ui/systemtrayicon.cpp
ui/windows7thumbbar.cpp
widgets/autoexpandingtreeview.cpp
widgets/busyindicator.cpp
@ -377,6 +378,7 @@ set(HEADERS
ui/qtsystemtrayicon.h
ui/settingsdialog.h
ui/systemtrayicon.h
ui/windows7thumbbar.h
widgets/autoexpandingtreeview.h
widgets/busyindicator.h

View File

@ -15,10 +15,6 @@
along with Clementine. If not, see <http://www.gnu.org/licenses/>.
*/
#ifdef Q_OS_WIN32
# define _WIN32_WINNT 0x0600
#endif
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include "core/backgroundstreams.h"
@ -81,6 +77,7 @@
#include "ui/qtsystemtrayicon.h"
#include "ui/settingsdialog.h"
#include "ui/systemtrayicon.h"
#include "ui/windows7thumbbar.h"
#include "widgets/errordialog.h"
#include "widgets/fileview.h"
#include "widgets/multiloadingindicator.h"
@ -123,9 +120,6 @@
#ifdef Q_OS_WIN32
# include <qtsparkle/Updater>
# include <windows.h>
# include <commctrl.h>
# include <shobjidl.h>
#endif
@ -156,6 +150,7 @@ MainWindow::MainWindow(
QWidget* parent)
: QMainWindow(parent),
ui_(new Ui_MainWindow),
thumbbar_(new Windows7ThumbBar(this)),
tray_icon_(tray_icon),
osd_(osd),
task_manager_(task_manager),
@ -191,8 +186,7 @@ MainWindow::MainWindow(
playlist_menu_(new QMenu(this)),
library_sort_model_(new QSortFilterProxyModel(this)),
track_position_timer_(new QTimer(this)),
was_maximized_(false),
taskbar_list_(NULL)
was_maximized_(false)
{
// Create some objects in the database thread
playlist_backend_ = new PlaylistBackend;
@ -499,6 +493,16 @@ MainWindow::MainWindow(
connect(tray_icon_, SIGNAL(ShowHide()), SLOT(ToggleShowHide()));
connect(tray_icon_, SIGNAL(ChangeVolume(int)), SLOT(VolumeWheelEvent(int)));
// Windows 7 thumbbar buttons
thumbbar_->SetActions(QList<QAction*>()
<< ui_->action_previous_track
<< ui_->action_play_pause
<< ui_->action_stop
<< ui_->action_next_track
<< NULL // spacer
<< ui_->action_love
<< ui_->action_ban);
#if (defined(Q_OS_DARWIN) && defined(HAVE_SPARKLE)) || defined(Q_OS_WIN32)
// Add check for updates item to application menu.
QAction* check_updates = ui_->menu_tools->addAction(tr("Check for updates..."));
@ -1651,51 +1655,8 @@ void MainWindow::ShowScriptDialog() {
}
#ifdef Q_OS_WIN32
static const int kTaskbarIconSize = 16;
static void SetButton(const QIcon& icon, const QString& tooltip, uint id,
THUMBBUTTON* button) {
button->dwMask = THUMBBUTTONMASK(THB_ICON | THB_TOOLTIP | THB_FLAGS);
button->iId = id;
button->dwFlags = THBF_ENABLED;
button->hIcon = icon.pixmap(kTaskbarIconSize).toWinHICON();
tooltip.toWCharArray(button->szTip);
}
bool MainWindow::winEvent(MSG* m, long* result) {
static UINT sTaskbarButtonCreated = WM_NULL;
if (sTaskbarButtonCreated == WM_NULL) {
// Compute the value for the TaskbarButtonCreated message
sTaskbarButtonCreated = RegisterWindowMessage("TaskbarButtonCreated");
}
if (m->message == sTaskbarButtonCreated) {
if (!taskbar_list_) {
// Get the interface
if (CoCreateInstance(CLSID_ITaskbarList, NULL, CLSCTX_ALL,
IID_ITaskbarList3, (void**) &taskbar_list_)) {
qWarning() << "Error creating the ITaskbarList3 interface";
return false;
}
ITaskbarList3* taskbar_list = reinterpret_cast<ITaskbarList3*>(taskbar_list_);
if (taskbar_list->HrInit()) {
taskbar_list->Release();
taskbar_list_ = NULL;
return false;
}
// Add the playback control buttons
THUMBBUTTON buttons[4] = {};
SetButton(IconLoader::Load("media-skip-backward"), tr("Previous track"), 0, &buttons[0]);
SetButton(IconLoader::Load("media-playback-start"), tr("Pause"), 1, &buttons[1]);
SetButton(IconLoader::Load("media-playback-stop"), tr("Stop"), 2, &buttons[2]);
SetButton(IconLoader::Load("media-skip-forward"), tr("Next track"), 3, &buttons[3]);
taskbar_list->ThumbBarAddButtons(winId(), 4, buttons);
}
}
bool MainWindow::winEvent(MSG* msg, long*) {
thumbbar_->HandleWinEvent(msg);
return false;
}
#endif // Q_OS_WIN32

View File

@ -67,6 +67,7 @@ class TaskManager;
class TranscodeDialog;
class VisualisationContainer;
class WiimotedevShortcuts;
class Windows7ThumbBar;
class Ui_MainWindow;
class QSortFilterProxyModel;
@ -216,6 +217,7 @@ class MainWindow : public QMainWindow, public PlatformInterface {
private:
Ui_MainWindow* ui_;
Windows7ThumbBar* thumbbar_;
SystemTrayIcon* tray_icon_;
OSD* osd_;
@ -284,9 +286,6 @@ class MainWindow : public QMainWindow, public PlatformInterface {
bool autoclear_playlist_;
BackgroundStreams* background_streams_;
// Really an ITaskbarList3* but I don't want to have to include windows.h here
void* taskbar_list_;
};
#endif // MAINWINDOW_H

140
src/ui/windows7thumbbar.cpp Normal file
View File

@ -0,0 +1,140 @@
/* This file is part of Clementine.
Copyright 2010, David Sansome <me@davidsansome.com>
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 <http://www.gnu.org/licenses/>.
*/
#include "windows7thumbbar.h"
#include <QAction>
#include <QtDebug>
#ifdef Q_OS_WIN32
# define _WIN32_WINNT 0x0600
# include <windows.h>
# include <commctrl.h>
# include <shobjidl.h>
#endif // Q_OS_WIN32
const int Windows7ThumbBar::kIconSize = 16;
const int Windows7ThumbBar::kMaxButtonCount = 7;
Windows7ThumbBar::Windows7ThumbBar(QWidget* widget)
: QObject(widget),
widget_(widget),
button_created_message_id_(0),
taskbar_list_(NULL)
{
}
void Windows7ThumbBar::SetActions(const QList<QAction*>& actions) {
#ifdef Q_OS_WIN32
Q_ASSERT(actions.count() <= kMaxButtonCount);
actions_ = actions;
foreach (QAction* action, actions) {
if (action) {
connect(action, SIGNAL(changed()), SLOT(ActionChanged()));
}
}
#endif // Q_OS_WIN32
}
#ifdef Q_OS_WIN32
static void SetupButton(const QAction* action, THUMBBUTTON* button) {
if (action) {
button->hIcon = action->icon().pixmap(Windows7ThumbBar::kIconSize).toWinHICON();
button->dwFlags = action->isEnabled() ? THBF_ENABLED : THBF_DISABLED;
action->text().toWCharArray(button->szTip);
button->szTip[action->text().count()] = '\0';
if (!action->isVisible()) {
button->dwFlags = THUMBBUTTONFLAGS(button->dwFlags | THBF_HIDDEN);
}
} else {
button->hIcon = 0;
button->szTip[0] = '\0';
button->dwFlags = THBF_NOBACKGROUND;
}
}
#endif // Q_OS_WIN32
void Windows7ThumbBar::HandleWinEvent(MSG* msg) {
#ifdef Q_OS_WIN32
if (button_created_message_id_ == 0) {
// Compute the value for the TaskbarButtonCreated message
button_created_message_id_ = RegisterWindowMessage("TaskbarButtonCreated");
}
if (msg->message == button_created_message_id_) {
if (!taskbar_list_) {
// Create the taskbar list for the first time
if (CoCreateInstance(CLSID_ITaskbarList, NULL, CLSCTX_ALL,
IID_ITaskbarList3, (void**) &taskbar_list_)) {
qWarning() << "Error creating the ITaskbarList3 interface";
return;
}
ITaskbarList3* taskbar_list = reinterpret_cast<ITaskbarList3*>(taskbar_list_);
if (taskbar_list->HrInit()) {
taskbar_list->Release();
taskbar_list_ = NULL;
return;
}
// Add the buttons
THUMBBUTTON buttons[kMaxButtonCount];
for (int i=0 ; i<actions_.count() ; ++i) {
const QAction* action = actions_[i];
THUMBBUTTON* button = &buttons[i];
button->dwMask = THUMBBUTTONMASK(THB_ICON | THB_TOOLTIP | THB_FLAGS);
button->iId = i;
SetupButton(action, button);
}
taskbar_list->ThumbBarAddButtons(widget_->winId(), actions_.count(), buttons);
}
} else if (msg->message == WM_COMMAND) {
const int button_id = LOWORD(msg->wParam);
if (button_id >= 0 && button_id < actions_.count()) {
actions_[button_id]->activate(QAction::Trigger);
}
}
#endif // Q_OS_WIN32
}
void Windows7ThumbBar::ActionChanged() {
#ifdef Q_OS_WIN32
if (!taskbar_list_)
return;
ITaskbarList3* taskbar_list = reinterpret_cast<ITaskbarList3*>(taskbar_list_);
THUMBBUTTON buttons[kMaxButtonCount];
for (int i=0 ; i<actions_.count() ; ++i) {
const QAction* action = actions_[i];
THUMBBUTTON* button = &buttons[i];
button->dwMask = THUMBBUTTONMASK(THB_ICON | THB_TOOLTIP | THB_FLAGS);
button->iId = i;
SetupButton(action, button);
}
taskbar_list->ThumbBarUpdateButtons(widget_->winId(), actions_.count(), buttons);
#endif // Q_OS_WIN32
}

59
src/ui/windows7thumbbar.h Normal file
View File

@ -0,0 +1,59 @@
/* This file is part of Clementine.
Copyright 2010, David Sansome <me@davidsansome.com>
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 <http://www.gnu.org/licenses/>.
*/
#ifndef WINDOWS7THUMBBAR_H
#define WINDOWS7THUMBBAR_H
#include <QMainWindow>
#include <QWidget>
#ifndef Q_OS_WIN32
typedef void MSG;
#endif // Q_OS_WIN32
class Windows7ThumbBar : public QObject {
Q_OBJECT
public:
// Creates a list of buttons in the taskbar icon for this window. Does
// nothing and is safe to use on other operating systems too.
Windows7ThumbBar(QWidget* widget = 0);
static const int kIconSize;
static const int kMaxButtonCount;
// You must call this in the parent widget's constructor before returning
// to the event loop. If an action is NULL it becomes a spacer.
void SetActions(const QList<QAction*>& actions);
// Call this from the parent's winEvent() function.
void HandleWinEvent(MSG* msg);
private slots:
void ActionChanged();
private:
QWidget* widget_;
QList<QAction*> actions_;
unsigned int button_created_message_id_;
// Really an ITaskbarList3* but I don't want to have to include windows.h here
void* taskbar_list_;
};
#endif // WINDOWS7THUMBBAR_H