mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-23 00:19:03 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			308 lines
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			308 lines
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2018 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include "UpdaterCommon/UI.h"
 | |
| 
 | |
| #include <cstdlib>
 | |
| #include <string>
 | |
| #include <thread>
 | |
| 
 | |
| #include <Windows.h>
 | |
| #include <CommCtrl.h>
 | |
| #include <ShObjIdl.h>
 | |
| #include <ShlObj.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, nullptr, 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", nullptr, 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, nullptr, 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, nullptr, 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 message = L"A fatal error occurred and the updater cannot continue:\n " +
 | |
|                  UTF8ToWString(text) + L"\n" +
 | |
|                  L"If the issue persists, please manually download the latest version from "
 | |
|                  L"dolphin-emu.org/download and extract it overtop your existing installation.\n" +
 | |
|                  L"Also consider filing a bug at bugs.dolphin-emu.org/projects/emulator";
 | |
| 
 | |
|   MessageBox(nullptr, message.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();
 | |
| }
 | |
| 
 | |
| bool IsTestMode()
 | |
| {
 | |
|   return std::getenv("DOLPHIN_UPDATE_SERVER_URL") != nullptr;
 | |
| }
 | |
| 
 | |
| void LaunchApplication(std::string path)
 | |
| {
 | |
|   const auto wpath = UTF8ToWString(path);
 | |
|   if (IsUserAnAdmin())
 | |
|   {
 | |
|     // Indirectly start the application via explorer. This effectively drops admin privileges
 | |
|     // because explorer is running as current user.
 | |
|     ShellExecuteW(nullptr, nullptr, L"explorer.exe", wpath.c_str(), nullptr, SW_SHOW);
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     std::wstring cmdline = L"\"" + wpath + L"\"";
 | |
|     STARTUPINFOW startup_info{.cb = sizeof(startup_info)};
 | |
|     PROCESS_INFORMATION process_info;
 | |
|     if (IsTestMode())
 | |
|       SetEnvironmentVariableA("DOLPHIN_UPDATE_TEST_DONE", "1");
 | |
|     if (CreateProcessW(wpath.c_str(), cmdline.data(), nullptr, nullptr, TRUE, 0, nullptr, nullptr,
 | |
|                        &startup_info, &process_info))
 | |
|     {
 | |
|       CloseHandle(process_info.hThread);
 | |
|       CloseHandle(process_info.hProcess);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| 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
 |