2019-09-04 18:26:26 +02:00
|
|
|
#include "pch.h"
|
2020-09-10 16:57:16 +03:00
|
|
|
#include "Generated files/resource.h"
|
2019-09-04 18:26:26 +02:00
|
|
|
#include "settings_window.h"
|
|
|
|
|
#include "tray_icon.h"
|
2025-05-26 05:03:35 -04:00
|
|
|
#include "general_settings.h"
|
2021-09-23 14:23:22 +01:00
|
|
|
#include "centralized_hotkeys.h"
|
|
|
|
|
#include "centralized_kb_hook.h"
|
2019-09-04 18:26:26 +02:00
|
|
|
#include <Windows.h>
|
|
|
|
|
|
2020-12-15 15:16:09 +03:00
|
|
|
#include <common/utils/resources.h>
|
|
|
|
|
#include <common/version/version.h>
|
2021-05-20 15:07:34 +03:00
|
|
|
#include <common/logger/logger.h>
|
2021-08-19 15:32:03 +01:00
|
|
|
#include <common/utils/elevation.h>
|
2023-09-15 08:34:17 +02:00
|
|
|
#include "bug_report.h"
|
2019-09-04 18:26:26 +02:00
|
|
|
|
2019-12-26 17:26:11 +01:00
|
|
|
namespace
|
|
|
|
|
{
|
|
|
|
|
HWND tray_icon_hwnd = NULL;
|
2019-09-04 18:26:26 +02:00
|
|
|
|
2023-09-15 08:34:17 +02:00
|
|
|
enum
|
|
|
|
|
{
|
2023-08-03 19:43:55 +02:00
|
|
|
wm_icon_notify = WM_APP,
|
|
|
|
|
wm_run_on_main_ui_thread,
|
|
|
|
|
};
|
2019-09-04 18:26:26 +02:00
|
|
|
|
2019-12-26 17:26:11 +01:00
|
|
|
// Contains the Windows Message for taskbar creation.
|
|
|
|
|
UINT wm_taskbar_restart = 0;
|
2019-09-04 18:26:26 +02:00
|
|
|
|
2019-12-26 17:26:11 +01:00
|
|
|
NOTIFYICONDATAW tray_icon_data;
|
|
|
|
|
bool tray_icon_created = false;
|
2019-09-04 18:26:26 +02:00
|
|
|
|
2019-12-26 17:26:11 +01:00
|
|
|
bool about_box_shown = false;
|
2019-11-27 17:19:10 +03:00
|
|
|
|
2019-12-26 17:26:11 +01:00
|
|
|
HMENU h_menu = nullptr;
|
|
|
|
|
HMENU h_sub_menu = nullptr;
|
2023-01-31 00:00:11 +01:00
|
|
|
bool double_click_timer_running = false;
|
|
|
|
|
bool double_clicked = false;
|
2023-02-03 15:10:14 +00:00
|
|
|
POINT tray_icon_click_point;
|
|
|
|
|
|
2019-11-27 17:19:10 +03:00
|
|
|
}
|
2019-09-04 18:26:26 +02:00
|
|
|
|
|
|
|
|
// Struct to fill with callback and the data. The window_proc is responsible for cleaning it.
|
2019-12-26 17:26:11 +01:00
|
|
|
struct run_on_main_ui_thread_msg
|
|
|
|
|
{
|
|
|
|
|
main_loop_callback_function _callback;
|
|
|
|
|
PVOID data;
|
2019-09-04 18:26:26 +02:00
|
|
|
};
|
|
|
|
|
|
2019-12-26 17:26:11 +01:00
|
|
|
bool dispatch_run_on_main_ui_thread(main_loop_callback_function _callback, PVOID data)
|
|
|
|
|
{
|
|
|
|
|
if (tray_icon_hwnd == NULL)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
struct run_on_main_ui_thread_msg* wnd_msg = new struct run_on_main_ui_thread_msg();
|
|
|
|
|
wnd_msg->_callback = _callback;
|
|
|
|
|
wnd_msg->data = data;
|
2019-09-04 18:26:26 +02:00
|
|
|
|
2023-02-08 11:17:33 +00:00
|
|
|
PostMessage(tray_icon_hwnd, wm_run_on_main_ui_thread, 0, reinterpret_cast<LPARAM>(wnd_msg));
|
2019-09-04 18:26:26 +02:00
|
|
|
|
2019-12-26 17:26:11 +01:00
|
|
|
return true;
|
2019-09-04 18:26:26 +02:00
|
|
|
}
|
|
|
|
|
|
2020-11-10 15:56:05 +03:00
|
|
|
void change_menu_item_text(const UINT item_id, wchar_t* new_text)
|
|
|
|
|
{
|
|
|
|
|
MENUITEMINFOW menuitem = { .cbSize = sizeof(MENUITEMINFOW), .fMask = MIIM_TYPE | MIIM_DATA };
|
|
|
|
|
GetMenuItemInfoW(h_menu, item_id, false, &menuitem);
|
|
|
|
|
menuitem.dwTypeData = new_text;
|
|
|
|
|
SetMenuItemInfoW(h_menu, item_id, false, &menuitem);
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-15 23:31:43 +01:00
|
|
|
void open_quick_access_flyout_window(const POINT flyout_position)
|
|
|
|
|
{
|
|
|
|
|
open_settings_window(std::nullopt, true, flyout_position);
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-29 12:54:05 +02:00
|
|
|
void handle_tray_command(HWND window, const WPARAM command_id, LPARAM lparam)
|
2021-01-21 11:55:58 +03:00
|
|
|
{
|
|
|
|
|
switch (command_id)
|
|
|
|
|
{
|
|
|
|
|
case ID_SETTINGS_MENU_COMMAND:
|
2023-09-15 08:34:17 +02:00
|
|
|
{
|
|
|
|
|
std::wstring settings_window{ winrt::to_hstring(ESettingsWindowNames_to_string(static_cast<ESettingsWindowNames>(lparam))) };
|
|
|
|
|
open_settings_window(settings_window, false);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2025-08-21 10:40:37 +02:00
|
|
|
case ID_CLOSE_MENU_COMMAND:
|
2021-01-21 11:55:58 +03:00
|
|
|
if (h_menu)
|
|
|
|
|
{
|
|
|
|
|
DestroyMenu(h_menu);
|
|
|
|
|
}
|
|
|
|
|
DestroyWindow(window);
|
|
|
|
|
break;
|
|
|
|
|
case ID_ABOUT_MENU_COMMAND:
|
|
|
|
|
if (!about_box_shown)
|
|
|
|
|
{
|
|
|
|
|
about_box_shown = true;
|
|
|
|
|
std::wstring about_msg = L"PowerToys\nVersion " + get_product_version() + L"\n\xa9 2019 Microsoft Corporation";
|
|
|
|
|
MessageBoxW(nullptr, about_msg.c_str(), L"About PowerToys", MB_OK);
|
|
|
|
|
about_box_shown = false;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case ID_REPORT_BUG_COMMAND:
|
2023-02-08 11:17:33 +00:00
|
|
|
{
|
2023-09-15 08:34:17 +02:00
|
|
|
launch_bug_report();
|
2021-01-21 11:55:58 +03:00
|
|
|
break;
|
|
|
|
|
}
|
2021-08-19 15:32:03 +01:00
|
|
|
|
|
|
|
|
case ID_DOCUMENTATION_MENU_COMMAND:
|
|
|
|
|
{
|
2022-04-19 22:00:28 +02:00
|
|
|
RunNonElevatedEx(L"https://aka.ms/PowerToysOverview", L"", L"");
|
2021-08-19 15:32:03 +01:00
|
|
|
break;
|
|
|
|
|
}
|
2024-03-15 23:31:43 +01:00
|
|
|
case ID_QUICK_ACCESS_MENU_COMMAND:
|
|
|
|
|
{
|
|
|
|
|
POINT mouse_pointer;
|
|
|
|
|
GetCursorPos(&mouse_pointer);
|
|
|
|
|
open_quick_access_flyout_window(mouse_pointer);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2021-08-19 15:32:03 +01:00
|
|
|
}
|
2021-01-21 11:55:58 +03:00
|
|
|
}
|
|
|
|
|
|
2023-01-31 00:00:11 +01:00
|
|
|
void click_timer_elapsed()
|
|
|
|
|
{
|
|
|
|
|
double_click_timer_running = false;
|
|
|
|
|
if (!double_clicked)
|
|
|
|
|
{
|
2024-03-15 23:31:43 +01:00
|
|
|
open_quick_access_flyout_window(tray_icon_click_point);
|
2023-01-31 00:00:11 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-26 17:26:11 +01:00
|
|
|
LRESULT __stdcall tray_icon_window_proc(HWND window, UINT message, WPARAM wparam, LPARAM lparam)
|
|
|
|
|
{
|
|
|
|
|
switch (message)
|
|
|
|
|
{
|
2021-05-20 15:07:34 +03:00
|
|
|
case WM_HOTKEY:
|
|
|
|
|
{
|
|
|
|
|
// We use the tray icon WndProc to avoid creating a dedicated window just for this message.
|
|
|
|
|
const auto modifiersMask = LOWORD(lparam);
|
|
|
|
|
const auto vkCode = HIWORD(lparam);
|
|
|
|
|
Logger::trace(L"On {} hotkey", CentralizedHotkeys::ToWstring({ modifiersMask, vkCode }));
|
|
|
|
|
CentralizedHotkeys::PopulateHotkey({ modifiersMask, vkCode });
|
|
|
|
|
break;
|
|
|
|
|
}
|
2019-12-26 17:26:11 +01:00
|
|
|
case WM_CREATE:
|
|
|
|
|
if (wm_taskbar_restart == 0)
|
|
|
|
|
{
|
|
|
|
|
tray_icon_hwnd = window;
|
|
|
|
|
wm_taskbar_restart = RegisterWindowMessageW(L"TaskbarCreated");
|
|
|
|
|
}
|
2019-09-04 18:26:26 +02:00
|
|
|
break;
|
2019-12-26 17:26:11 +01:00
|
|
|
case WM_DESTROY:
|
|
|
|
|
if (tray_icon_created)
|
|
|
|
|
{
|
|
|
|
|
Shell_NotifyIcon(NIM_DELETE, &tray_icon_data);
|
|
|
|
|
tray_icon_created = false;
|
2019-09-04 18:26:26 +02:00
|
|
|
}
|
2020-05-05 14:13:52 -07:00
|
|
|
close_settings_window();
|
2019-12-26 17:26:11 +01:00
|
|
|
PostQuitMessage(0);
|
|
|
|
|
break;
|
|
|
|
|
case WM_CLOSE:
|
2019-09-04 18:26:26 +02:00
|
|
|
DestroyWindow(window);
|
|
|
|
|
break;
|
2019-12-26 17:26:11 +01:00
|
|
|
case WM_COMMAND:
|
2021-09-29 12:54:05 +02:00
|
|
|
handle_tray_command(window, wparam, lparam);
|
2019-12-26 17:26:11 +01:00
|
|
|
break;
|
|
|
|
|
// Shell_NotifyIcon can fail when we invoke it during the time explorer.exe isn't present/ready to handle it.
|
|
|
|
|
// We'll also never receive wm_taskbar_restart message if the first call to Shell_NotifyIcon failed, so we use
|
|
|
|
|
// WM_WINDOWPOSCHANGING which is always received on explorer startup sequence.
|
2020-03-05 13:07:06 +03:00
|
|
|
case WM_WINDOWPOSCHANGING:
|
|
|
|
|
{
|
2019-12-26 17:26:11 +01:00
|
|
|
if (!tray_icon_created)
|
|
|
|
|
{
|
|
|
|
|
tray_icon_created = Shell_NotifyIcon(NIM_ADD, &tray_icon_data) == TRUE;
|
2019-09-04 18:26:26 +02:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2019-12-26 17:26:11 +01:00
|
|
|
default:
|
|
|
|
|
if (message == wm_icon_notify)
|
2019-09-04 18:26:26 +02:00
|
|
|
{
|
2019-12-26 17:26:11 +01:00
|
|
|
switch (lparam)
|
|
|
|
|
{
|
|
|
|
|
case WM_RBUTTONUP:
|
2020-03-05 13:07:06 +03:00
|
|
|
case WM_CONTEXTMENU:
|
|
|
|
|
{
|
2019-12-26 17:26:11 +01:00
|
|
|
if (!h_menu)
|
|
|
|
|
{
|
|
|
|
|
h_menu = LoadMenu(reinterpret_cast<HINSTANCE>(&__ImageBase), MAKEINTRESOURCE(ID_TRAY_MENU));
|
|
|
|
|
}
|
2020-11-10 15:56:05 +03:00
|
|
|
if (h_menu)
|
|
|
|
|
{
|
|
|
|
|
static std::wstring settings_menuitem_label = GET_RESOURCE_STRING(IDS_SETTINGS_MENU_TEXT);
|
2025-08-21 10:40:37 +02:00
|
|
|
static std::wstring close_menuitem_label = GET_RESOURCE_STRING(IDS_CLOSE_MENU_TEXT);
|
2021-01-21 11:55:58 +03:00
|
|
|
static std::wstring submit_bug_menuitem_label = GET_RESOURCE_STRING(IDS_SUBMIT_BUG_TEXT);
|
2021-08-19 15:32:03 +01:00
|
|
|
static std::wstring documentation_menuitem_label = GET_RESOURCE_STRING(IDS_DOCUMENTATION_MENU_TEXT);
|
2024-03-15 23:31:43 +01:00
|
|
|
static std::wstring quick_access_menuitem_label = GET_RESOURCE_STRING(IDS_QUICK_ACCESS_MENU_TEXT);
|
2020-11-10 15:56:05 +03:00
|
|
|
change_menu_item_text(ID_SETTINGS_MENU_COMMAND, settings_menuitem_label.data());
|
2025-08-21 10:40:37 +02:00
|
|
|
change_menu_item_text(ID_CLOSE_MENU_COMMAND, close_menuitem_label.data());
|
2021-01-21 11:55:58 +03:00
|
|
|
change_menu_item_text(ID_REPORT_BUG_COMMAND, submit_bug_menuitem_label.data());
|
2025-06-17 14:49:54 +08:00
|
|
|
bool bug_report_disabled = is_bug_report_running();
|
|
|
|
|
EnableMenuItem(h_sub_menu, ID_REPORT_BUG_COMMAND, MF_BYCOMMAND | (bug_report_disabled ? MF_GRAYED : MF_ENABLED));
|
2021-08-19 15:32:03 +01:00
|
|
|
change_menu_item_text(ID_DOCUMENTATION_MENU_COMMAND, documentation_menuitem_label.data());
|
2024-03-15 23:31:43 +01:00
|
|
|
change_menu_item_text(ID_QUICK_ACCESS_MENU_COMMAND, quick_access_menuitem_label.data());
|
2020-11-10 15:56:05 +03:00
|
|
|
}
|
2019-12-26 17:26:11 +01:00
|
|
|
if (!h_sub_menu)
|
|
|
|
|
{
|
|
|
|
|
h_sub_menu = GetSubMenu(h_menu, 0);
|
|
|
|
|
}
|
|
|
|
|
POINT mouse_pointer;
|
|
|
|
|
GetCursorPos(&mouse_pointer);
|
|
|
|
|
SetForegroundWindow(window); // Needed for the context menu to disappear.
|
|
|
|
|
TrackPopupMenu(h_sub_menu, TPM_CENTERALIGN | TPM_BOTTOMALIGN, mouse_pointer.x, mouse_pointer.y, 0, window, nullptr);
|
2023-01-31 00:00:11 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case WM_LBUTTONUP:
|
|
|
|
|
{
|
|
|
|
|
// ignore event if this is the second click of a double click
|
|
|
|
|
if (!double_click_timer_running)
|
|
|
|
|
{
|
2023-02-03 15:10:14 +00:00
|
|
|
// save the cursor position for sending where to show the popup.
|
|
|
|
|
GetCursorPos(&tray_icon_click_point);
|
|
|
|
|
|
2023-01-31 00:00:11 +01:00
|
|
|
// start timer for detecting single or double click
|
|
|
|
|
double_click_timer_running = true;
|
|
|
|
|
double_clicked = false;
|
2023-09-15 08:34:17 +02:00
|
|
|
|
2023-01-31 00:00:11 +01:00
|
|
|
UINT doubleClickTime = GetDoubleClickTime();
|
|
|
|
|
std::thread([doubleClickTime]() {
|
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(doubleClickTime));
|
|
|
|
|
click_timer_elapsed();
|
|
|
|
|
}).detach();
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case WM_LBUTTONDBLCLK:
|
|
|
|
|
{
|
|
|
|
|
double_clicked = true;
|
|
|
|
|
open_settings_window(std::nullopt, false);
|
|
|
|
|
break;
|
2019-12-26 17:26:11 +01:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2019-09-04 18:26:26 +02:00
|
|
|
}
|
2019-12-26 17:26:11 +01:00
|
|
|
else if (message == wm_run_on_main_ui_thread)
|
2019-09-04 18:26:26 +02:00
|
|
|
{
|
2019-12-26 17:26:11 +01:00
|
|
|
if (lparam != NULL)
|
|
|
|
|
{
|
2023-02-08 11:17:33 +00:00
|
|
|
struct run_on_main_ui_thread_msg* msg = reinterpret_cast<struct run_on_main_ui_thread_msg*>(lparam);
|
2019-12-26 17:26:11 +01:00
|
|
|
msg->_callback(msg->data);
|
|
|
|
|
delete msg;
|
|
|
|
|
lparam = NULL;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else if (message == wm_taskbar_restart)
|
|
|
|
|
{
|
|
|
|
|
tray_icon_created = Shell_NotifyIcon(NIM_ADD, &tray_icon_data) == TRUE;
|
|
|
|
|
break;
|
2019-09-04 18:26:26 +02:00
|
|
|
}
|
|
|
|
|
}
|
2019-12-26 17:26:11 +01:00
|
|
|
return DefWindowProc(window, message, wparam, lparam);
|
2019-09-04 18:26:26 +02:00
|
|
|
}
|
|
|
|
|
|
2025-06-17 14:49:54 +08:00
|
|
|
void update_bug_report_menu_status(bool isRunning)
|
|
|
|
|
{
|
|
|
|
|
if (h_sub_menu != nullptr)
|
|
|
|
|
{
|
|
|
|
|
EnableMenuItem(h_sub_menu, ID_REPORT_BUG_COMMAND, MF_BYCOMMAND | (isRunning ? MF_GRAYED : MF_ENABLED));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-12 12:11:26 +02:00
|
|
|
void start_tray_icon(bool isProcessElevated)
|
2019-12-26 17:26:11 +01:00
|
|
|
{
|
|
|
|
|
auto h_instance = reinterpret_cast<HINSTANCE>(&__ImageBase);
|
|
|
|
|
auto icon = LoadIcon(h_instance, MAKEINTRESOURCE(APPICON));
|
|
|
|
|
if (icon)
|
|
|
|
|
{
|
2023-08-03 19:43:55 +02:00
|
|
|
UINT id_tray_icon = 1;
|
2019-12-26 17:26:11 +01:00
|
|
|
|
|
|
|
|
WNDCLASS wc = {};
|
|
|
|
|
wc.hCursor = LoadCursor(nullptr, IDC_ARROW);
|
|
|
|
|
wc.hInstance = h_instance;
|
2020-04-21 10:30:12 +03:00
|
|
|
wc.lpszClassName = pt_tray_icon_window_class;
|
2019-12-26 17:26:11 +01:00
|
|
|
wc.style = CS_HREDRAW | CS_VREDRAW;
|
|
|
|
|
wc.lpfnWndProc = tray_icon_window_proc;
|
|
|
|
|
wc.hIcon = icon;
|
|
|
|
|
RegisterClass(&wc);
|
|
|
|
|
auto hwnd = CreateWindowW(wc.lpszClassName,
|
2020-04-21 10:30:12 +03:00
|
|
|
pt_tray_icon_window_class,
|
2019-12-26 17:26:11 +01:00
|
|
|
WS_OVERLAPPEDWINDOW | WS_POPUP,
|
|
|
|
|
CW_USEDEFAULT,
|
|
|
|
|
CW_USEDEFAULT,
|
|
|
|
|
CW_USEDEFAULT,
|
|
|
|
|
CW_USEDEFAULT,
|
|
|
|
|
nullptr,
|
|
|
|
|
nullptr,
|
|
|
|
|
wc.hInstance,
|
|
|
|
|
nullptr);
|
|
|
|
|
WINRT_VERIFY(hwnd);
|
2021-05-20 15:07:34 +03:00
|
|
|
CentralizedHotkeys::RegisterWindow(hwnd);
|
2021-09-23 14:23:22 +01:00
|
|
|
CentralizedKeyboardHook::RegisterWindow(hwnd);
|
2019-12-26 17:26:11 +01:00
|
|
|
memset(&tray_icon_data, 0, sizeof(tray_icon_data));
|
|
|
|
|
tray_icon_data.cbSize = sizeof(tray_icon_data);
|
|
|
|
|
tray_icon_data.hIcon = icon;
|
|
|
|
|
tray_icon_data.hWnd = hwnd;
|
2023-07-28 13:43:32 +02:00
|
|
|
tray_icon_data.uID = id_tray_icon;
|
2019-12-26 17:26:11 +01:00
|
|
|
tray_icon_data.uCallbackMessage = wm_icon_notify;
|
2023-09-12 12:11:26 +02:00
|
|
|
|
|
|
|
|
std::wstringstream pt_version_tooltip_stream;
|
|
|
|
|
if (isProcessElevated)
|
|
|
|
|
{
|
|
|
|
|
pt_version_tooltip_stream << GET_RESOURCE_STRING(IDS_TRAY_ICON_ADMIN_TOOLTIP) << L": ";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pt_version_tooltip_stream << L"PowerToys " << get_product_version() << '\0';
|
|
|
|
|
std::wstring pt_version_tooltip = pt_version_tooltip_stream.str();
|
|
|
|
|
wcscpy_s(tray_icon_data.szTip, sizeof(tray_icon_data.szTip) / sizeof(WCHAR), pt_version_tooltip.c_str());
|
2023-07-28 13:43:32 +02:00
|
|
|
tray_icon_data.uFlags = NIF_ICON | NIF_TIP | NIF_MESSAGE;
|
2021-08-04 19:54:09 +02:00
|
|
|
ChangeWindowMessageFilterEx(hwnd, WM_COMMAND, MSGFLT_ALLOW, nullptr);
|
2019-12-26 17:26:11 +01:00
|
|
|
|
|
|
|
|
tray_icon_created = Shell_NotifyIcon(NIM_ADD, &tray_icon_data) == TRUE;
|
2025-06-17 14:49:54 +08:00
|
|
|
|
|
|
|
|
// Register callback to update bug report menu item status
|
|
|
|
|
BugReportManager::instance().register_callback([](bool isRunning) {
|
|
|
|
|
dispatch_run_on_main_ui_thread([](PVOID data) {
|
|
|
|
|
bool* running = static_cast<bool*>(data);
|
|
|
|
|
update_bug_report_menu_status(*running);
|
|
|
|
|
delete running;
|
|
|
|
|
},
|
|
|
|
|
new bool(isRunning));
|
|
|
|
|
});
|
2019-12-26 17:26:11 +01:00
|
|
|
}
|
2019-09-04 18:26:26 +02:00
|
|
|
}
|
2019-12-16 18:36:52 +01:00
|
|
|
|
2025-05-26 05:03:35 -04:00
|
|
|
void set_tray_icon_visible(bool shouldIconBeVisible)
|
|
|
|
|
{
|
|
|
|
|
tray_icon_data.uFlags |= NIF_STATE;
|
|
|
|
|
tray_icon_data.dwStateMask = NIS_HIDDEN;
|
|
|
|
|
tray_icon_data.dwState = shouldIconBeVisible ? 0 : NIS_HIDDEN;
|
|
|
|
|
Shell_NotifyIcon(NIM_MODIFY, &tray_icon_data);
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-26 17:26:11 +01:00
|
|
|
void stop_tray_icon()
|
|
|
|
|
{
|
|
|
|
|
if (tray_icon_created)
|
|
|
|
|
{
|
2025-06-17 14:49:54 +08:00
|
|
|
// Clear bug report callbacks
|
|
|
|
|
BugReportManager::instance().clear_callbacks();
|
2019-12-26 17:26:11 +01:00
|
|
|
SendMessage(tray_icon_hwnd, WM_CLOSE, 0, 0);
|
|
|
|
|
}
|
2025-06-17 14:49:54 +08:00
|
|
|
}
|