Refactor code in taskbarHost

This commit is contained in:
Cheng Zhao 2015-08-06 12:44:07 +08:00
parent a28f70e85c
commit 958658513c
5 changed files with 92 additions and 88 deletions

View file

@ -438,16 +438,18 @@ void Window::SetOverlayIcon(const gfx::Image& overlay,
window_->SetOverlayIcon(overlay, description); window_->SetOverlayIcon(overlay, description);
} }
void Window::SetThumbarButtons(mate::Arguments* args) { bool Window::SetThumbarButtons(mate::Arguments* args) {
#if defined(OS_WIN) #if defined(OS_WIN)
std::vector<TaskbarHost::ThumbarButton> buttons; std::vector<TaskbarHost::ThumbarButton> buttons;
if (!args->GetNext(&buttons)) { if (!args->GetNext(&buttons)) {
args->ThrowError(); args->ThrowError();
return; return false;
} }
auto window = static_cast<NativeWindowViews*>(window_.get()); auto window = static_cast<NativeWindowViews*>(window_.get());
window->taskbar_host().SetThumbarButtons(window->GetAcceleratedWidget(), return window->taskbar_host().SetThumbarButtons(
buttons); window->GetAcceleratedWidget(), buttons);
#else
return false;
#endif #endif
} }

View file

@ -130,7 +130,7 @@ class Window : public mate::TrackableObject<Window>,
void SetProgressBar(double progress); void SetProgressBar(double progress);
void SetOverlayIcon(const gfx::Image& overlay, void SetOverlayIcon(const gfx::Image& overlay,
const std::string& description); const std::string& description);
void SetThumbarButtons(mate::Arguments* args); bool SetThumbarButtons(mate::Arguments* args);
void SetMenu(v8::Isolate* isolate, v8::Local<v8::Value> menu); void SetMenu(v8::Isolate* isolate, v8::Local<v8::Value> menu);
void SetAutoHideMenuBar(bool auto_hide); void SetAutoHideMenuBar(bool auto_hide);
bool IsMenuBarAutoHide(); bool IsMenuBarAutoHide();

View file

@ -8,57 +8,48 @@
#include <string> #include <string>
#include "base/stl_util.h"
#include "base/win/scoped_comptr.h" #include "base/win/scoped_comptr.h"
#include "base/win/win_util.h" #include "base/win/scoped_gdi_object.h"
#include "base/win/wrapped_window_proc.h"
#include "base/strings/utf_string_conversions.h" #include "base/strings/utf_string_conversions.h"
#include "third_party/skia/include/core/SkBitmap.h" #include "third_party/skia/include/core/SkBitmap.h"
#include "ui/gfx/icon_util.h" #include "ui/gfx/icon_util.h"
#include "ui/gfx/win/hwnd_util.h"
namespace atom { namespace atom {
namespace { namespace {
// From MSDN: https://msdn.microsoft.com/en-us/library/windows/desktop/ // From MSDN: https://msdn.microsoft.com/en-us/library/windows/desktop/dd378460(v=vs.85).aspx#thumbbars
// dd378460(v=vs.85).aspx#thumbbars
// The thumbnail toolbar has a maximum of seven buttons due to the limited room. // The thumbnail toolbar has a maximum of seven buttons due to the limited room.
const int kMaxButtonsCount = 7; const size_t kMaxButtonsCount = 7;
// The base id of Thumbar button. // The base id of Thumbar button.
const int kButtonIdBase = 40001; const int kButtonIdBase = 40001;
bool GetThumbarButtonFlags(const std::vector<std::string>& flags, bool GetThumbarButtonFlags(const std::vector<std::string>& flags,
THUMBBUTTONFLAGS* out) { THUMBBUTTONFLAGS* out) {
if (flags.empty()) { THUMBBUTTONFLAGS result = THBF_ENABLED; // THBF_ENABLED == 0
*out = THBF_ENABLED;
return true;
}
THUMBBUTTONFLAGS result = static_cast<THUMBBUTTONFLAGS>(0);
for (const auto& flag : flags) { for (const auto& flag : flags) {
if (flag == "enabled") { if (flag == "disabled")
result |= THBF_ENABLED;
} else if (flag == "disabled") {
result |= THBF_DISABLED; result |= THBF_DISABLED;
} else if (flag == "dismissonclick") { else if (flag == "dismissonclick")
result |= THBF_DISMISSONCLICK; result |= THBF_DISMISSONCLICK;
} else if (flag == "nobackground") { else if (flag == "nobackground")
result |= THBF_NOBACKGROUND; result |= THBF_NOBACKGROUND;
} else if (flag == "hidden") { else if (flag == "hidden")
result |= THBF_HIDDEN; result |= THBF_HIDDEN;
} else if (flag == "noninteractive") { else if (flag == "noninteractive")
result |= THBF_NONINTERACTIVE; result |= THBF_NONINTERACTIVE;
} else { else
return false; return false;
} }
}
*out = result; *out = result;
return true; return true;
} }
} // namespace } // namespace
TaskbarHost::TaskbarHost() : is_initialized_(false) { TaskbarHost::TaskbarHost() : thumbar_buttons_added_(false) {
} }
TaskbarHost::~TaskbarHost() { TaskbarHost::~TaskbarHost() {
@ -66,9 +57,6 @@ TaskbarHost::~TaskbarHost() {
bool TaskbarHost::SetThumbarButtons( bool TaskbarHost::SetThumbarButtons(
HWND window, const std::vector<ThumbarButton>& buttons) { HWND window, const std::vector<ThumbarButton>& buttons) {
if (buttons.size() > kMaxButtonsCount)
return false;
base::win::ScopedComPtr<ITaskbarList3> taskbar; base::win::ScopedComPtr<ITaskbarList3> taskbar;
if (FAILED(taskbar.CreateInstance(CLSID_TaskbarList, if (FAILED(taskbar.CreateInstance(CLSID_TaskbarList,
nullptr, nullptr,
@ -77,59 +65,67 @@ bool TaskbarHost::SetThumbarButtons(
return false; return false;
} }
callback_map_.clear();
// The number of buttons in thumbar can not be changed once it is created,
// so we have to claim kMaxButtonsCount buttons initialy in case users add
// more buttons later.
base::win::ScopedHICON icons[kMaxButtonsCount] = {};
THUMBBUTTON thumb_buttons[kMaxButtonsCount] = {}; THUMBBUTTON thumb_buttons[kMaxButtonsCount] = {};
thumbar_button_clicked_callback_map_.clear();
// Once a toolbar with a set of buttons is added to thumbnail, there is no way for (size_t i = 0; i < kMaxButtonsCount; ++i) {
// to remove it without re-creating the window. THUMBBUTTON& thumb_button = thumb_buttons[i];
// To achieve to re-set thumbar buttons, we initialize the buttons with
// HIDDEN state and only updated the caller's specified buttons. // Set ID.
// thumb_button.iId = kButtonIdBase + i;
// Initialize all thumb buttons with HIDDEN state. thumb_button.dwMask = THB_FLAGS;
for (int i = 0; i < kMaxButtonsCount; ++i) {
thumb_buttons[i].iId = kButtonIdBase + i; if (i >= buttons.size()) {
thumb_buttons[i].dwMask = THB_FLAGS; // dwFlags is valid. // This button is used to occupy the place in toolbar, and it does not
thumb_buttons[i].dwFlags = THBF_HIDDEN; // show.
thumb_button.dwFlags = THBF_HIDDEN;
continue;
} }
// Update the callers' specified buttons. // This button is user's button.
for (size_t i = 0; i < buttons.size(); ++i) { const ThumbarButton& button = buttons[i];
if (!GetThumbarButtonFlags(buttons[i].flags, &thumb_buttons[i].dwFlags))
// Generate flags.
thumb_button.dwFlags = THBF_ENABLED;
if (!GetThumbarButtonFlags(button.flags, &thumb_button.dwFlags))
return false; return false;
thumb_buttons[i].dwMask = THB_ICON | THB_FLAGS;
thumb_buttons[i].hIcon = IconUtil::CreateHICONFromSkBitmap( // Set icon.
buttons[i].icon.AsBitmap()); if (!button.icon.IsEmpty()) {
if (!buttons[i].tooltip.empty()) { thumb_button.dwMask |= THB_ICON;
thumb_buttons[i].dwMask |= THB_TOOLTIP; icons[i] = IconUtil::CreateHICONFromSkBitmap(button.icon.AsBitmap());
wcscpy_s(thumb_buttons[i].szTip, thumb_button.hIcon = icons[i].Get();
base::UTF8ToUTF16(buttons[i].tooltip).c_str());
} }
thumbar_button_clicked_callback_map_[thumb_buttons[i].iId] = // Set tooltip.
buttons[i].clicked_callback; if (!button.tooltip.empty()) {
thumb_button.dwMask |= THB_TOOLTIP;
wcscpy_s(thumb_button.szTip, base::UTF8ToUTF16(button.tooltip).c_str());
} }
bool is_success = false; // Save callback.
if (!is_initialized_) { callback_map_[thumb_button.iId] = button.clicked_callback;
is_initialized_ = true;
is_success = taskbar->ThumbBarAddButtons(
window, buttons.size(), thumb_buttons) == S_OK;
} else {
is_success = taskbar->ThumbBarUpdateButtons(
window, kMaxButtonsCount, thumb_buttons) == S_OK;
} }
// Release thumb_buttons' icons, the taskbar makes its own copy. // Finally add them to taskbar.
for (size_t i = 0; i < buttons.size(); ++i) { HRESULT r;
::DestroyIcon(thumb_buttons[i].hIcon); if (thumbar_buttons_added_)
} r = taskbar->ThumbBarUpdateButtons(window, kMaxButtonsCount, thumb_buttons);
return is_success; else
r = taskbar->ThumbBarAddButtons(window, kMaxButtonsCount, thumb_buttons);
thumbar_buttons_added_ = true;
return SUCCEEDED(r);
} }
bool TaskbarHost::HandleThumbarButtonEvent(int button_id) { bool TaskbarHost::HandleThumbarButtonEvent(int button_id) {
if (thumbar_button_clicked_callback_map_.find(button_id) != if (ContainsKey(callback_map_, button_id)) {
thumbar_button_clicked_callback_map_.end()) { auto callback = callback_map_[button_id];
auto callback = thumbar_button_clicked_callback_map_[button_id];
if (!callback.is_null()) if (!callback.is_null())
callback.Run(); callback.Run();
return true; return true;

View file

@ -8,6 +8,7 @@
#include <windows.h> #include <windows.h>
#include <map> #include <map>
#include <string>
#include <vector> #include <vector>
#include "base/callback.h" #include "base/callback.h"
@ -27,15 +28,19 @@ class TaskbarHost {
TaskbarHost(); TaskbarHost();
virtual ~TaskbarHost(); virtual ~TaskbarHost();
// Add or update the buttons in thumbar
bool SetThumbarButtons( bool SetThumbarButtons(
HWND window, const std::vector<ThumbarButton>& buttons); HWND window, const std::vector<ThumbarButton>& buttons);
// Called by the window that there is a button in thumbar clicked.
bool HandleThumbarButtonEvent(int button_id); bool HandleThumbarButtonEvent(int button_id);
private: private:
using ThumbarButtonClickedCallbackMap = std::map<int, base::Closure>; using CallbackMap = std::map<int, base::Closure>;
ThumbarButtonClickedCallbackMap thumbar_button_clicked_callback_map_; CallbackMap callback_map_;
bool is_initialized_; // Whether we have already added the buttons to thumbar.
bool thumbar_buttons_added_;
DISALLOW_COPY_AND_ASSIGN(TaskbarHost); DISALLOW_COPY_AND_ASSIGN(TaskbarHost);
}; };

View file

@ -646,13 +646,14 @@ __Note:__ This API is only available on Windows (Windows 7 and above)
### BrowserWindow.setThumbarButtons(buttons) ### BrowserWindow.setThumbarButtons(buttons)
* `buttons` Array of `button` objects * `buttons` Array
* `button` Object * `button` Object
* `icon` [NativeImage](native-image.md) - The icon showing in thumbnail * `icon` [NativeImage](native-image.md) - The icon showing in thumbnail
toolbar. toolbar.
* `tooltip` String - (Option) - The text of the button's tooltip. * `tooltip` String (optional) - The text of the button's tooltip.
* `flags` Array of Strings - (Option) Control specific states and behaviors * `flags` Array (optional) - Control specific states and behaviors
of the button. By default, it uses `enabled`. of the button. By default, it uses `enabled`. It can include following
Strings:
* `enabled` - The button is active and available to the user. * `enabled` - The button is active and available to the user.
* `disabled` - The button is disabled. It is present, but has a visual * `disabled` - The button is disabled. It is present, but has a visual
state that indicates that it will not respond to user action. state that indicates that it will not respond to user action.