mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-01-03 09:29:48 +00:00
36eacd8698
* fix: scanning issues * fix: sync, storing silent unspents * chore: deps * fix: label issues, clear spent utxo * chore: deps * fix: build * fix: missing types * feat: new electrs API & changes, fixes for last block scanning * feat: Scan Silent Payments homepage toggle * chore: build configure * feat: generic fixes, testnet UI improvements, useSSL on bitcoin nodes * fix: invalid Object in sendData * feat: improve addresses page & address book displays * feat: silent payments labeled addresses disclaimer * fix: missing i18n * chore: print * feat: single block scan, rescan by date working for btc mainnet * feat: new cake features page replace market page, move sp scan toggle, auto switch node pop up alert * feat: delete silent addresses * fix: red dot in non ssl nodes * fix: inconsistent connection states, fix tx history * fix: tx & balance displays, cpfp sending * feat: new rust lib * chore: node path * fix: check node based on network * fix: missing txcount from addresses * style: padding in feature page cards * fix: restore not getting all wallet addresses by type * fix: auto switch node broken * fix: silent payment txs not being restored * feat: change scanning to subscription model, sync improvements * fix: scan re-subscription * fix: default nodes * fix: improve scanning by date, fix single block scan * refactor: common function for input tx selection * various fixes for build issues * initial monero.dart implementation * ... * multiple wallets new lib minor fixes * other fixes from monero.dart and monero_c * fix: nodes & build * update build scripts fix polyseed * remove unnecessary code * Add windows app, build scripts and build guide for it. * Minor fix in generated monero configs * fix: send all with multiple outs * add missing monero_c command * add android build script * Merge and fix main * undo android ndk removal * Fix modified exception_handler.dart * Temporarily remove haven * fix build issues * fix pr script * Fixes for build monero.dart (monero_c) for windows. * monero build script * wip: ios build script * refactor: unchanged file * Added build guides for iOS and macOS. Replaced nproc call on macOS. Added macOS configuration for configure_cake_wallet.sh script. * Update monero.dart and monero_c versions. * Add missed windows build scripts * Update the application configuration for windows build script. * Update cw_monero pubspec lock file for monero.dart * Update pr_test_build.yml * chore: upgrade * chore: merge changes * refactor: unchanged files [skip ci] * Fix conflicts with main * fix for multiple wallets * Add tron to windows application configuration. * Add macOS option for description message in configure_cake_wallet.sh * Include missed monero dll for windows. * fix conflicts with main * Disable haven configuration for iOS as default. Add ability to configure cakewallet for iOS with for configuration script. Remove cw_shared configuration for cw_monero. * fix: scan fixes, add date, allow sending while scanning * add missing nano secrets file [skip ci] * ios library * don't pull prebuilds android * Add auto generation of manifest file for android project even for iOS, macOS, Windows. * feat: sync fixes, sp settings * feat: fix resyncing * store crash fix * make init async so it won't lag disable print starts * fix monero_c build issues * libstdc++ * Fix MacOS saving wallet file issue Fix Secure Storage issue (somehow) * update pubspec.lock * fix build script * Use dylib as iOS framework. Use custom path for loading of iOS framework for monero.dart. Add script for generate iOS framework for monero wallet. * fix: date from height logic, status disconnected & chain tip get * fix: params * feat: electrum migration if using cake electrum * fix nodes update versions * re-enable tron * update sp_scanner to work on iOS [skip ci] * bump monero_c hash * bump monero_c commit * bump moneroc version * bump monero_c commit * Add ability to build monero wallet lib as universal lib. Update macOS build guide. Change default arch for macOS project to . * fix: wrong socket for old electrum nodes * Fix unchecked wallet type call * get App Dir correctly in default_settings_migration.dart * handle previous issue with fetching linux documents directory [skip ci] * backup fix * fix NTFS issues * Close the wallet when the wallet gets changed * fix: double balance * feat: node domain * fix: menu name * bump monero_c commit * fix: update tip on set scanning * fix: connection switching back and forth * feat: check if node is electrs, and supports sp * chore: fix build * minor enhancements * fixes and enhancements * solve conflicts with main * Only stop wallet on rename and delete * fix: status toggle * minor enhancement * Monero.com fixes * bump monero_c commit * update sp_scanner to include windows and linux * Update macOS build guide. Change brew dependencies for build unbound locally. * fix conflicts and update macos build guide * remove build cache when on gh actions * update secure storage * free up even more storage * free up more storage * Add initial wownero * fix conflicts * fix workflow issue * build wownero * ios and windows changes * macos * complete wownero flow (app side) * add keychain group entitlement and update script for RunnerBase on macos * update secure_storage version to 8.1.0 in configure.dart * add wownero framework * update ios builds * proper path for wownero and monero * finalizing wownero * finalizing wownero * free up even more storage * revert commenting of build gradle configs * revert commenting of secrets [skip ci] * free more storage * minor fixes * link android wownero libraries * bump monero_c commit * wownero fixes * rename target * build_single.sh using clean env * bump monero_c commit * minor fix * Add wownero polyseed * fix conflicts with main * fix: wallet seed display fix: wownero not refreshing * fix: wallet seed display fix: wownero not refreshing * bump monero_c commit * minor fixes * fix: incorrectly displaying XMR instead of WOW * fix: incorrect restore height in wownero * bump monero_c commit * Add Inno Setup Script for windows exe installer * drop libc++_shared.so * fixes from comments * Fix CMake for windows * Merge latest monero dart changes [skip ci] * bump monero_c commit * add wownero to build scripts for macos [skip ci] * add 14 word seed support to wownero * UI fixes for wownero seed restore * minor fixes * reformat code to pass lints * wownero: fixes haven: removal popup * minor iOS fix [skip ci] * fix: wownero confirmation count (it is spendable after 3 confirms) fix: transaction history not displaying in WOW and XMR when tx has 0 confirms, This is more of a workaround, because I have no idea why would the cpp code not return pending transaction. * Update preferences_key.dart [skip ci] * minor fixes --------- Co-authored-by: Rafael Saes <git@rafael.saes.dev> Co-authored-by: Czarek Nakamoto <cyjan@mrcyjanek.net> Co-authored-by: M <m@cakewallet.com> Co-authored-by: Konstantin Ullrich <konstantinullrich12@gmail.com> Co-authored-by: Matthew Fosse <matt@fosse.co>
288 lines
8.3 KiB
C++
288 lines
8.3 KiB
C++
#include "win32_window.h"
|
|
|
|
#include <dwmapi.h>
|
|
#include <flutter_windows.h>
|
|
|
|
#include "resource.h"
|
|
|
|
namespace {
|
|
|
|
/// Window attribute that enables dark mode window decorations.
|
|
///
|
|
/// Redefined in case the developer's machine has a Windows SDK older than
|
|
/// version 10.0.22000.0.
|
|
/// See: https://docs.microsoft.com/windows/win32/api/dwmapi/ne-dwmapi-dwmwindowattribute
|
|
#ifndef DWMWA_USE_IMMERSIVE_DARK_MODE
|
|
#define DWMWA_USE_IMMERSIVE_DARK_MODE 20
|
|
#endif
|
|
|
|
constexpr const wchar_t kWindowClassName[] = L"FLUTTER_RUNNER_WIN32_WINDOW";
|
|
|
|
/// Registry key for app theme preference.
|
|
///
|
|
/// A value of 0 indicates apps should use dark mode. A non-zero or missing
|
|
/// value indicates apps should use light mode.
|
|
constexpr const wchar_t kGetPreferredBrightnessRegKey[] =
|
|
L"Software\\Microsoft\\Windows\\CurrentVersion\\Themes\\Personalize";
|
|
constexpr const wchar_t kGetPreferredBrightnessRegValue[] = L"AppsUseLightTheme";
|
|
|
|
// The number of Win32Window objects that currently exist.
|
|
static int g_active_window_count = 0;
|
|
|
|
using EnableNonClientDpiScaling = BOOL __stdcall(HWND hwnd);
|
|
|
|
// Scale helper to convert logical scaler values to physical using passed in
|
|
// scale factor
|
|
int Scale(int source, double scale_factor) {
|
|
return static_cast<int>(source * scale_factor);
|
|
}
|
|
|
|
// Dynamically loads the |EnableNonClientDpiScaling| from the User32 module.
|
|
// This API is only needed for PerMonitor V1 awareness mode.
|
|
void EnableFullDpiSupportIfAvailable(HWND hwnd) {
|
|
HMODULE user32_module = LoadLibraryA("User32.dll");
|
|
if (!user32_module) {
|
|
return;
|
|
}
|
|
auto enable_non_client_dpi_scaling =
|
|
reinterpret_cast<EnableNonClientDpiScaling*>(
|
|
GetProcAddress(user32_module, "EnableNonClientDpiScaling"));
|
|
if (enable_non_client_dpi_scaling != nullptr) {
|
|
enable_non_client_dpi_scaling(hwnd);
|
|
}
|
|
FreeLibrary(user32_module);
|
|
}
|
|
|
|
} // namespace
|
|
|
|
// Manages the Win32Window's window class registration.
|
|
class WindowClassRegistrar {
|
|
public:
|
|
~WindowClassRegistrar() = default;
|
|
|
|
// Returns the singleton registrar instance.
|
|
static WindowClassRegistrar* GetInstance() {
|
|
if (!instance_) {
|
|
instance_ = new WindowClassRegistrar();
|
|
}
|
|
return instance_;
|
|
}
|
|
|
|
// Returns the name of the window class, registering the class if it hasn't
|
|
// previously been registered.
|
|
const wchar_t* GetWindowClass();
|
|
|
|
// Unregisters the window class. Should only be called if there are no
|
|
// instances of the window.
|
|
void UnregisterWindowClass();
|
|
|
|
private:
|
|
WindowClassRegistrar() = default;
|
|
|
|
static WindowClassRegistrar* instance_;
|
|
|
|
bool class_registered_ = false;
|
|
};
|
|
|
|
WindowClassRegistrar* WindowClassRegistrar::instance_ = nullptr;
|
|
|
|
const wchar_t* WindowClassRegistrar::GetWindowClass() {
|
|
if (!class_registered_) {
|
|
WNDCLASS window_class{};
|
|
window_class.hCursor = LoadCursor(nullptr, IDC_ARROW);
|
|
window_class.lpszClassName = kWindowClassName;
|
|
window_class.style = CS_HREDRAW | CS_VREDRAW;
|
|
window_class.cbClsExtra = 0;
|
|
window_class.cbWndExtra = 0;
|
|
window_class.hInstance = GetModuleHandle(nullptr);
|
|
window_class.hIcon =
|
|
LoadIcon(window_class.hInstance, MAKEINTRESOURCE(IDI_APP_ICON));
|
|
window_class.hbrBackground = 0;
|
|
window_class.lpszMenuName = nullptr;
|
|
window_class.lpfnWndProc = Win32Window::WndProc;
|
|
RegisterClass(&window_class);
|
|
class_registered_ = true;
|
|
}
|
|
return kWindowClassName;
|
|
}
|
|
|
|
void WindowClassRegistrar::UnregisterWindowClass() {
|
|
UnregisterClass(kWindowClassName, nullptr);
|
|
class_registered_ = false;
|
|
}
|
|
|
|
Win32Window::Win32Window() {
|
|
++g_active_window_count;
|
|
}
|
|
|
|
Win32Window::~Win32Window() {
|
|
--g_active_window_count;
|
|
Destroy();
|
|
}
|
|
|
|
bool Win32Window::Create(const std::wstring& title,
|
|
const Point& origin,
|
|
const Size& size) {
|
|
Destroy();
|
|
|
|
const wchar_t* window_class =
|
|
WindowClassRegistrar::GetInstance()->GetWindowClass();
|
|
|
|
const POINT target_point = {static_cast<LONG>(origin.x),
|
|
static_cast<LONG>(origin.y)};
|
|
HMONITOR monitor = MonitorFromPoint(target_point, MONITOR_DEFAULTTONEAREST);
|
|
UINT dpi = FlutterDesktopGetDpiForMonitor(monitor);
|
|
double scale_factor = dpi / 96.0;
|
|
|
|
HWND window = CreateWindow(
|
|
window_class, title.c_str(), WS_OVERLAPPEDWINDOW,
|
|
Scale(origin.x, scale_factor), Scale(origin.y, scale_factor),
|
|
Scale(size.width, scale_factor), Scale(size.height, scale_factor),
|
|
nullptr, nullptr, GetModuleHandle(nullptr), this);
|
|
|
|
if (!window) {
|
|
return false;
|
|
}
|
|
|
|
UpdateTheme(window);
|
|
|
|
return OnCreate();
|
|
}
|
|
|
|
bool Win32Window::Show() {
|
|
return ShowWindow(window_handle_, SW_SHOWNORMAL);
|
|
}
|
|
|
|
// static
|
|
LRESULT CALLBACK Win32Window::WndProc(HWND const window,
|
|
UINT const message,
|
|
WPARAM const wparam,
|
|
LPARAM const lparam) noexcept {
|
|
if (message == WM_NCCREATE) {
|
|
auto window_struct = reinterpret_cast<CREATESTRUCT*>(lparam);
|
|
SetWindowLongPtr(window, GWLP_USERDATA,
|
|
reinterpret_cast<LONG_PTR>(window_struct->lpCreateParams));
|
|
|
|
auto that = static_cast<Win32Window*>(window_struct->lpCreateParams);
|
|
EnableFullDpiSupportIfAvailable(window);
|
|
that->window_handle_ = window;
|
|
} else if (Win32Window* that = GetThisFromHandle(window)) {
|
|
return that->MessageHandler(window, message, wparam, lparam);
|
|
}
|
|
|
|
return DefWindowProc(window, message, wparam, lparam);
|
|
}
|
|
|
|
LRESULT
|
|
Win32Window::MessageHandler(HWND hwnd,
|
|
UINT const message,
|
|
WPARAM const wparam,
|
|
LPARAM const lparam) noexcept {
|
|
switch (message) {
|
|
case WM_DESTROY:
|
|
window_handle_ = nullptr;
|
|
Destroy();
|
|
if (quit_on_close_) {
|
|
PostQuitMessage(0);
|
|
}
|
|
return 0;
|
|
|
|
case WM_DPICHANGED: {
|
|
auto newRectSize = reinterpret_cast<RECT*>(lparam);
|
|
LONG newWidth = newRectSize->right - newRectSize->left;
|
|
LONG newHeight = newRectSize->bottom - newRectSize->top;
|
|
|
|
SetWindowPos(hwnd, nullptr, newRectSize->left, newRectSize->top, newWidth,
|
|
newHeight, SWP_NOZORDER | SWP_NOACTIVATE);
|
|
|
|
return 0;
|
|
}
|
|
case WM_SIZE: {
|
|
RECT rect = GetClientArea();
|
|
if (child_content_ != nullptr) {
|
|
// Size and position the child window.
|
|
MoveWindow(child_content_, rect.left, rect.top, rect.right - rect.left,
|
|
rect.bottom - rect.top, TRUE);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
case WM_ACTIVATE:
|
|
if (child_content_ != nullptr) {
|
|
SetFocus(child_content_);
|
|
}
|
|
return 0;
|
|
|
|
case WM_DWMCOLORIZATIONCOLORCHANGED:
|
|
UpdateTheme(hwnd);
|
|
return 0;
|
|
}
|
|
|
|
return DefWindowProc(window_handle_, message, wparam, lparam);
|
|
}
|
|
|
|
void Win32Window::Destroy() {
|
|
OnDestroy();
|
|
|
|
if (window_handle_) {
|
|
DestroyWindow(window_handle_);
|
|
window_handle_ = nullptr;
|
|
}
|
|
if (g_active_window_count == 0) {
|
|
WindowClassRegistrar::GetInstance()->UnregisterWindowClass();
|
|
}
|
|
}
|
|
|
|
Win32Window* Win32Window::GetThisFromHandle(HWND const window) noexcept {
|
|
return reinterpret_cast<Win32Window*>(
|
|
GetWindowLongPtr(window, GWLP_USERDATA));
|
|
}
|
|
|
|
void Win32Window::SetChildContent(HWND content) {
|
|
child_content_ = content;
|
|
SetParent(content, window_handle_);
|
|
RECT frame = GetClientArea();
|
|
|
|
MoveWindow(content, frame.left, frame.top, frame.right - frame.left,
|
|
frame.bottom - frame.top, true);
|
|
|
|
SetFocus(child_content_);
|
|
}
|
|
|
|
RECT Win32Window::GetClientArea() {
|
|
RECT frame;
|
|
GetClientRect(window_handle_, &frame);
|
|
return frame;
|
|
}
|
|
|
|
HWND Win32Window::GetHandle() {
|
|
return window_handle_;
|
|
}
|
|
|
|
void Win32Window::SetQuitOnClose(bool quit_on_close) {
|
|
quit_on_close_ = quit_on_close;
|
|
}
|
|
|
|
bool Win32Window::OnCreate() {
|
|
// No-op; provided for subclasses.
|
|
return true;
|
|
}
|
|
|
|
void Win32Window::OnDestroy() {
|
|
// No-op; provided for subclasses.
|
|
}
|
|
|
|
void Win32Window::UpdateTheme(HWND const window) {
|
|
DWORD light_mode;
|
|
DWORD light_mode_size = sizeof(light_mode);
|
|
LSTATUS result = RegGetValue(HKEY_CURRENT_USER, kGetPreferredBrightnessRegKey,
|
|
kGetPreferredBrightnessRegValue,
|
|
RRF_RT_REG_DWORD, nullptr, &light_mode,
|
|
&light_mode_size);
|
|
|
|
if (result == ERROR_SUCCESS) {
|
|
BOOL enable_dark_mode = light_mode == 0;
|
|
DwmSetWindowAttribute(window, DWMWA_USE_IMMERSIVE_DARK_MODE,
|
|
&enable_dark_mode, sizeof(enable_dark_mode));
|
|
}
|
|
}
|