mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-26 09:59:15 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			281 lines
		
	
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			281 lines
		
	
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2018 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include "UpdaterCommon/UI.h"
 | |
| 
 | |
| #include <string>
 | |
| #include <thread>
 | |
| 
 | |
| #include <Windows.h>
 | |
| #include <CommCtrl.h>
 | |
| #include <ShObjIdl.h>
 | |
| #include <shellapi.h>
 | |
| #include <wrl/client.h>
 | |
| 
 | |
| #include "Common/Event.h"
 | |
| #include "Common/ScopeGuard.h"
 | |
| #include "Common/StringUtil.h"
 | |
| 
 | |
| namespace
 | |
| {
 | |
| HWND window_handle = nullptr;
 | |
| HWND label_handle = nullptr;
 | |
| HWND total_progressbar_handle = nullptr;
 | |
| HWND current_progressbar_handle = nullptr;
 | |
| Microsoft::WRL::ComPtr<ITaskbarList3> taskbar_list;
 | |
| 
 | |
| std::thread ui_thread;
 | |
| Common::Event window_created_event;
 | |
| 
 | |
| int GetWindowHeight(HWND hwnd)
 | |
| {
 | |
|   RECT rect;
 | |
|   GetWindowRect(hwnd, &rect);
 | |
| 
 | |
|   return rect.bottom - rect.top;
 | |
| }
 | |
| 
 | |
| LRESULT CALLBACK WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
 | |
| {
 | |
|   switch (uMsg)
 | |
|   {
 | |
|   case WM_DESTROY:
 | |
|     PostQuitMessage(0);
 | |
|     return 0;
 | |
|   }
 | |
|   return DefWindowProc(hwnd, uMsg, wParam, lParam);
 | |
| }
 | |
| };  // namespace
 | |
| 
 | |
| constexpr int PROGRESSBAR_FLAGS = WS_VISIBLE | WS_CHILD | PBS_SMOOTH | PBS_SMOOTHREVERSE;
 | |
| constexpr int WINDOW_FLAGS = WS_CLIPCHILDREN;
 | |
| constexpr int PADDING_HEIGHT = 5;
 | |
| 
 | |
| namespace UI
 | |
| {
 | |
| bool InitWindow()
 | |
| {
 | |
|   InitCommonControls();
 | |
| 
 | |
|   // Notify main thread we're done creating the window when we return
 | |
|   Common::ScopeGuard ui_guard{[] { window_created_event.Set(); }};
 | |
| 
 | |
|   WNDCLASS wndcl = {};
 | |
|   wndcl.lpfnWndProc = WindowProc;
 | |
|   wndcl.hbrBackground = GetSysColorBrush(COLOR_MENU);
 | |
|   wndcl.lpszClassName = L"UPDATER";
 | |
| 
 | |
|   if (!RegisterClass(&wndcl))
 | |
|     return false;
 | |
| 
 | |
|   window_handle =
 | |
|       CreateWindow(L"UPDATER", L"Dolphin Updater", WINDOW_FLAGS, CW_USEDEFAULT, CW_USEDEFAULT, 500,
 | |
|                    100, nullptr, nullptr, GetModuleHandle(nullptr), 0);
 | |
| 
 | |
|   if (!window_handle)
 | |
|     return false;
 | |
| 
 | |
|   if (SUCCEEDED(CoCreateInstance(CLSID_TaskbarList, NULL, CLSCTX_INPROC_SERVER,
 | |
|                                  IID_PPV_ARGS(taskbar_list.GetAddressOf()))))
 | |
|   {
 | |
|     if (FAILED(taskbar_list->HrInit()))
 | |
|     {
 | |
|       taskbar_list.Reset();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   int y = PADDING_HEIGHT;
 | |
| 
 | |
|   label_handle = CreateWindow(L"STATIC", NULL, WS_VISIBLE | WS_CHILD, 5, y, 500, 25, window_handle,
 | |
|                               nullptr, nullptr, 0);
 | |
| 
 | |
|   if (!label_handle)
 | |
|     return false;
 | |
| 
 | |
|   // Get the default system font
 | |
|   NONCLIENTMETRICS metrics = {};
 | |
|   metrics.cbSize = sizeof(NONCLIENTMETRICS);
 | |
| 
 | |
|   if (!SystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(metrics), &metrics, 0))
 | |
|     return false;
 | |
| 
 | |
|   SendMessage(label_handle, WM_SETFONT,
 | |
|               reinterpret_cast<WPARAM>(CreateFontIndirect(&metrics.lfMessageFont)), 0);
 | |
| 
 | |
|   y += GetWindowHeight(label_handle) + PADDING_HEIGHT;
 | |
| 
 | |
|   total_progressbar_handle = CreateWindow(PROGRESS_CLASS, NULL, PROGRESSBAR_FLAGS, 5, y, 470, 25,
 | |
|                                           window_handle, nullptr, nullptr, 0);
 | |
| 
 | |
|   y += GetWindowHeight(total_progressbar_handle) + PADDING_HEIGHT;
 | |
| 
 | |
|   if (!total_progressbar_handle)
 | |
|     return false;
 | |
| 
 | |
|   current_progressbar_handle = CreateWindow(PROGRESS_CLASS, NULL, PROGRESSBAR_FLAGS, 5, y, 470, 25,
 | |
|                                             window_handle, nullptr, nullptr, 0);
 | |
| 
 | |
|   y += GetWindowHeight(current_progressbar_handle) + PADDING_HEIGHT;
 | |
| 
 | |
|   if (!current_progressbar_handle)
 | |
|     return false;
 | |
| 
 | |
|   RECT rect;
 | |
|   GetWindowRect(window_handle, &rect);
 | |
| 
 | |
|   // Account for the title bar
 | |
|   y += GetSystemMetrics(SM_CYFRAME) + GetSystemMetrics(SM_CYCAPTION) +
 | |
|        GetSystemMetrics(SM_CXPADDEDBORDER);
 | |
|   // ...and window border
 | |
|   y += GetSystemMetrics(SM_CYBORDER);
 | |
| 
 | |
|   // Add some padding for good measure
 | |
|   y += PADDING_HEIGHT * 3;
 | |
| 
 | |
|   SetWindowPos(window_handle, HWND_TOPMOST, 0, 0, rect.right - rect.left, y, SWP_NOMOVE);
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void SetTotalMarquee(bool marquee)
 | |
| {
 | |
|   SetWindowLong(total_progressbar_handle, GWL_STYLE,
 | |
|                 PROGRESSBAR_FLAGS | (marquee ? PBS_MARQUEE : 0));
 | |
|   SendMessage(total_progressbar_handle, PBM_SETMARQUEE, marquee, 0);
 | |
|   if (taskbar_list)
 | |
|   {
 | |
|     taskbar_list->SetProgressState(window_handle, marquee ? TBPF_INDETERMINATE : TBPF_NORMAL);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void ResetTotalProgress()
 | |
| {
 | |
|   SendMessage(total_progressbar_handle, PBM_SETPOS, 0, 0);
 | |
|   SetCurrentMarquee(true);
 | |
| }
 | |
| 
 | |
| void SetTotalProgress(int current, int total)
 | |
| {
 | |
|   SendMessage(total_progressbar_handle, PBM_SETRANGE32, 0, total);
 | |
|   SendMessage(total_progressbar_handle, PBM_SETPOS, current, 0);
 | |
|   if (taskbar_list)
 | |
|   {
 | |
|     taskbar_list->SetProgressValue(window_handle, current, total);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void SetCurrentMarquee(bool marquee)
 | |
| {
 | |
|   SetWindowLong(current_progressbar_handle, GWL_STYLE,
 | |
|                 PROGRESSBAR_FLAGS | (marquee ? PBS_MARQUEE : 0));
 | |
|   SendMessage(current_progressbar_handle, PBM_SETMARQUEE, marquee, 0);
 | |
| }
 | |
| 
 | |
| void ResetCurrentProgress()
 | |
| {
 | |
|   SendMessage(current_progressbar_handle, PBM_SETPOS, 0, 0);
 | |
|   SetCurrentMarquee(true);
 | |
| }
 | |
| 
 | |
| void Error(const std::string& text)
 | |
| {
 | |
|   auto wide_text = UTF8ToWString(text);
 | |
| 
 | |
|   MessageBox(nullptr,
 | |
|              (L"A fatal error occured and the updater cannot continue:\n " + wide_text).c_str(),
 | |
|              L"Error", MB_ICONERROR);
 | |
| 
 | |
|   if (taskbar_list)
 | |
|   {
 | |
|     taskbar_list->SetProgressState(window_handle, TBPF_ERROR);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void SetCurrentProgress(int current, int total)
 | |
| {
 | |
|   SendMessage(current_progressbar_handle, PBM_SETRANGE32, 0, total);
 | |
|   SendMessage(current_progressbar_handle, PBM_SETPOS, current, 0);
 | |
| }
 | |
| 
 | |
| void SetDescription(const std::string& text)
 | |
| {
 | |
|   SetWindowText(label_handle, UTF8ToWString(text).c_str());
 | |
| }
 | |
| 
 | |
| void MessageLoop()
 | |
| {
 | |
|   HRESULT result = CoInitializeEx(nullptr, COINIT_MULTITHREADED);
 | |
| 
 | |
|   Common::ScopeGuard ui_guard{[result] {
 | |
|     taskbar_list.Reset();
 | |
|     if (SUCCEEDED(result))
 | |
|       CoUninitialize();
 | |
|   }};
 | |
| 
 | |
|   if (!InitWindow())
 | |
|   {
 | |
|     MessageBox(nullptr, L"Window init failed!", L"", MB_ICONERROR);
 | |
|     // Destroying the parent (if exists) destroys all children windows
 | |
|     if (window_handle)
 | |
|     {
 | |
|       DestroyWindow(window_handle);
 | |
|       window_handle = nullptr;
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   SetTotalMarquee(true);
 | |
|   SetCurrentMarquee(true);
 | |
| 
 | |
|   MSG msg;
 | |
|   while (GetMessage(&msg, nullptr, 0, 0))
 | |
|   {
 | |
|     TranslateMessage(&msg);
 | |
|     DispatchMessage(&msg);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void Init()
 | |
| {
 | |
|   ui_thread = std::thread(MessageLoop);
 | |
| 
 | |
|   // Wait for UI thread to finish creating the window (or at least attempting to)
 | |
|   window_created_event.Wait();
 | |
| }
 | |
| 
 | |
| void Stop()
 | |
| {
 | |
|   if (window_handle)
 | |
|     PostMessage(window_handle, WM_CLOSE, 0, 0);
 | |
| 
 | |
|   ui_thread.join();
 | |
| }
 | |
| 
 | |
| void LaunchApplication(std::string path)
 | |
| {
 | |
|   // Indirectly start the application via explorer. This effectively drops admin priviliges because
 | |
|   // explorer is running as current user.
 | |
|   ShellExecuteW(nullptr, nullptr, L"explorer.exe", UTF8ToWString(path).c_str(), nullptr, SW_SHOW);
 | |
| }
 | |
| 
 | |
| void Sleep(int sleep)
 | |
| {
 | |
|   ::Sleep(sleep * 1000);
 | |
| }
 | |
| 
 | |
| void WaitForPID(u32 pid)
 | |
| {
 | |
|   HANDLE parent_handle = OpenProcess(SYNCHRONIZE, FALSE, static_cast<DWORD>(pid));
 | |
|   if (parent_handle)
 | |
|   {
 | |
|     WaitForSingleObject(parent_handle, INFINITE);
 | |
|     CloseHandle(parent_handle);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void SetVisible(bool visible)
 | |
| {
 | |
|   ShowWindow(window_handle, visible ? SW_SHOW : SW_HIDE);
 | |
| }
 | |
| 
 | |
| };  // namespace UI
 |