708 lines
		
	
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			708 lines
		
	
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
This file is part of Telegram Desktop,
 | 
						|
the official desktop application for the Telegram messaging service.
 | 
						|
 | 
						|
For license and copyright information please follow this link:
 | 
						|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | 
						|
*/
 | 
						|
#include "platform/win/main_window_win.h"
 | 
						|
 | 
						|
#include "styles/style_window.h"
 | 
						|
#include "platform/platform_specific.h"
 | 
						|
#include "platform/platform_notifications_manager.h"
 | 
						|
#include "platform/win/tray_win.h"
 | 
						|
#include "platform/win/windows_dlls.h"
 | 
						|
#include "platform/win/integration_win.h"
 | 
						|
#include "window/notifications_manager.h"
 | 
						|
#include "window/window_session_controller.h"
 | 
						|
#include "mainwindow.h"
 | 
						|
#include "main/main_session.h"
 | 
						|
#include "base/crc32hash.h"
 | 
						|
#include "base/platform/win/base_windows_wrl.h"
 | 
						|
#include "base/platform/base_platform_info.h"
 | 
						|
#include "core/application.h"
 | 
						|
#include "core/sandbox.h"
 | 
						|
#include "lang/lang_keys.h"
 | 
						|
#include "storage/localstorage.h"
 | 
						|
#include "ui/widgets/popup_menu.h"
 | 
						|
#include "ui/ui_utility.h"
 | 
						|
#include "window/themes/window_theme.h"
 | 
						|
#include "window/window_controller.h"
 | 
						|
#include "history/history.h"
 | 
						|
 | 
						|
#include <QtWidgets/QDesktopWidget>
 | 
						|
#include <QtWidgets/QStyleFactory>
 | 
						|
#include <QtWidgets/QApplication>
 | 
						|
#include <QtGui/QWindow>
 | 
						|
#include <QtGui/QScreen>
 | 
						|
 | 
						|
#include <Shobjidl.h>
 | 
						|
#include <shellapi.h>
 | 
						|
#include <WtsApi32.h>
 | 
						|
#include <dwmapi.h>
 | 
						|
 | 
						|
#include <windows.ui.viewmanagement.h>
 | 
						|
#include <UIViewSettingsInterop.h>
 | 
						|
 | 
						|
#include <Windowsx.h>
 | 
						|
#include <VersionHelpers.h>
 | 
						|
 | 
						|
// Taken from qtbase/src/gui/image/qpixmap_win.cpp
 | 
						|
HICON qt_pixmapToWinHICON(const QPixmap &);
 | 
						|
HBITMAP qt_imageToWinHBITMAP(const QImage &, int hbitmapFormat);
 | 
						|
 | 
						|
namespace ViewManagement = ABI::Windows::UI::ViewManagement;
 | 
						|
 | 
						|
namespace Platform {
 | 
						|
namespace {
 | 
						|
 | 
						|
// Mouse down on tray icon deactivates the application.
 | 
						|
// So there is no way to know for sure if the tray icon was clicked from
 | 
						|
// active application or from inactive application. So we assume that
 | 
						|
// if the application was deactivated less than 0.5s ago, then the tray
 | 
						|
// icon click (both left or right button) was made from the active app.
 | 
						|
constexpr auto kKeepActiveForTrayIcon = crl::time(500);
 | 
						|
 | 
						|
using namespace Microsoft::WRL;
 | 
						|
 | 
						|
// Taken from qtbase/src/gui/image/qpixmap_win.cpp
 | 
						|
enum HBitmapFormat {
 | 
						|
	HBitmapNoAlpha,
 | 
						|
	HBitmapPremultipliedAlpha,
 | 
						|
	HBitmapAlpha
 | 
						|
};
 | 
						|
 | 
						|
class EventFilter final : public QAbstractNativeEventFilter {
 | 
						|
public:
 | 
						|
	explicit EventFilter(not_null<MainWindow*> window);
 | 
						|
 | 
						|
private:
 | 
						|
	bool nativeEventFilter(
 | 
						|
		const QByteArray &eventType,
 | 
						|
		void *message,
 | 
						|
		long *result) override;
 | 
						|
 | 
						|
	bool mainWindowEvent(
 | 
						|
		HWND hWnd,
 | 
						|
		UINT msg,
 | 
						|
		WPARAM wParam,
 | 
						|
		LPARAM lParam,
 | 
						|
		LRESULT *result);
 | 
						|
 | 
						|
	const not_null<MainWindow*> _window;
 | 
						|
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
[[nodiscard]] HICON NativeIcon(const QIcon &icon, QSize size) {
 | 
						|
	if (!icon.isNull()) {
 | 
						|
		const auto pixmap = icon.pixmap(icon.actualSize(size));
 | 
						|
		if (!pixmap.isNull()) {
 | 
						|
			return qt_pixmapToWinHICON(pixmap);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nullptr;
 | 
						|
}
 | 
						|
 | 
						|
struct RealSize {
 | 
						|
	QSize value;
 | 
						|
	bool maximized = false;
 | 
						|
};
 | 
						|
[[nodiscard]] RealSize DetectRealSize(HWND hwnd) {
 | 
						|
	auto result = RECT();
 | 
						|
	auto placement = WINDOWPLACEMENT();
 | 
						|
	if (!GetWindowPlacement(hwnd, &placement)) {
 | 
						|
		return {};
 | 
						|
	} else if (placement.flags & WPF_RESTORETOMAXIMIZED) {
 | 
						|
		const auto monitor = MonitorFromRect(
 | 
						|
			&placement.rcNormalPosition,
 | 
						|
			MONITOR_DEFAULTTONULL);
 | 
						|
		if (!monitor) {
 | 
						|
			return {};
 | 
						|
		}
 | 
						|
		auto info = MONITORINFO{ .cbSize = sizeof(MONITORINFO) };
 | 
						|
		if (!GetMonitorInfo(monitor, &info)) {
 | 
						|
			return {};
 | 
						|
		}
 | 
						|
		result = info.rcWork;
 | 
						|
	} else {
 | 
						|
		CopyRect(&result, &placement.rcNormalPosition);
 | 
						|
	}
 | 
						|
	return {
 | 
						|
		{ int(result.right - result.left), int(result.bottom - result.top) },
 | 
						|
		((placement.flags & WPF_RESTORETOMAXIMIZED) != 0)
 | 
						|
	};
 | 
						|
}
 | 
						|
 | 
						|
[[nodiscard]] QImage PrepareLogoPreview(
 | 
						|
		QSize size,
 | 
						|
		QImage::Format format,
 | 
						|
		int radius = 0) {
 | 
						|
	auto result = QImage(size, QImage::Format_RGB32);
 | 
						|
	result.fill(st::windowBg->c);
 | 
						|
 | 
						|
	const auto logo = Window::Logo();
 | 
						|
	const auto width = size.width();
 | 
						|
	const auto height = size.height();
 | 
						|
	const auto side = logo.width();
 | 
						|
	const auto skip = width / 8;
 | 
						|
	const auto use = std::min({ width - skip, height - skip, side });
 | 
						|
	auto p = QPainter(&result);
 | 
						|
	if (use == side) {
 | 
						|
		p.drawImage((width - side) / 2, (height - side) / 2, logo);
 | 
						|
	} else {
 | 
						|
		const auto scaled = logo.scaled(
 | 
						|
			use,
 | 
						|
			use,
 | 
						|
			Qt::KeepAspectRatio,
 | 
						|
			Qt::SmoothTransformation);
 | 
						|
		p.drawImage((width - use) / 2, (height - use) / 2, scaled);
 | 
						|
	}
 | 
						|
	p.end();
 | 
						|
 | 
						|
	return radius
 | 
						|
		? Images::Round(std::move(result), Images::CornersMask(radius))
 | 
						|
		: result;
 | 
						|
}
 | 
						|
 | 
						|
EventFilter::EventFilter(not_null<MainWindow*> window) : _window(window) {
 | 
						|
}
 | 
						|
 | 
						|
bool EventFilter::nativeEventFilter(
 | 
						|
		const QByteArray &eventType,
 | 
						|
		void *message,
 | 
						|
		long *result) {
 | 
						|
	return Core::Sandbox::Instance().customEnterFromEventLoop([&] {
 | 
						|
		const auto msg = static_cast<MSG*>(message);
 | 
						|
		if (msg->hwnd == _window->psHwnd()
 | 
						|
			|| msg->hwnd && !_window->psHwnd()) {
 | 
						|
			return mainWindowEvent(
 | 
						|
				msg->hwnd,
 | 
						|
				msg->message,
 | 
						|
				msg->wParam,
 | 
						|
				msg->lParam,
 | 
						|
				(LRESULT*)result);
 | 
						|
		}
 | 
						|
		return false;
 | 
						|
	});
 | 
						|
}
 | 
						|
 | 
						|
bool EventFilter::mainWindowEvent(
 | 
						|
		HWND hWnd,
 | 
						|
		UINT msg,
 | 
						|
		WPARAM wParam,
 | 
						|
		LPARAM lParam,
 | 
						|
		LRESULT *result) {
 | 
						|
	switch (msg) {
 | 
						|
 | 
						|
	case WM_DESTROY: {
 | 
						|
		_window->destroyedFromSystem();
 | 
						|
	} return false;
 | 
						|
 | 
						|
	case WM_ACTIVATE: {
 | 
						|
		if (LOWORD(wParam) != WA_INACTIVE) {
 | 
						|
			_window->shadowsActivate();
 | 
						|
		} else {
 | 
						|
			_window->shadowsDeactivate();
 | 
						|
		}
 | 
						|
	} return false;
 | 
						|
 | 
						|
	case WM_SIZE: {
 | 
						|
		if (wParam == SIZE_MAXIMIZED || wParam == SIZE_RESTORED || wParam == SIZE_MINIMIZED) {
 | 
						|
			if (wParam == SIZE_RESTORED && _window->windowState() == Qt::WindowNoState) {
 | 
						|
				_window->positionUpdated();
 | 
						|
			}
 | 
						|
		}
 | 
						|
	} return false;
 | 
						|
 | 
						|
	case WM_MOVE: {
 | 
						|
		_window->positionUpdated();
 | 
						|
	} return false;
 | 
						|
 | 
						|
	case WM_DWMSENDICONICTHUMBNAIL: {
 | 
						|
		if (!Core::App().passcodeLocked()) {
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		const auto size = QSize(int(HIWORD(lParam)), int(LOWORD(lParam)));
 | 
						|
		return _window->setDwmThumbnail(size);
 | 
						|
	}
 | 
						|
 | 
						|
	case WM_DWMSENDICONICLIVEPREVIEWBITMAP: {
 | 
						|
		if (!Core::App().passcodeLocked()) {
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		const auto size = DetectRealSize(hWnd);
 | 
						|
		const auto radius = size.maximized ? 0 : style::ConvertScale(8);
 | 
						|
		return _window->setDwmPreview(size.value, radius);
 | 
						|
	}
 | 
						|
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
} // namespace
 | 
						|
 | 
						|
struct MainWindow::Private {
 | 
						|
	explicit Private(not_null<MainWindow*> window) : filter(window) {
 | 
						|
	}
 | 
						|
 | 
						|
	EventFilter filter;
 | 
						|
	ComPtr<ViewManagement::IUIViewSettings> viewSettings;
 | 
						|
};
 | 
						|
 | 
						|
MainWindow::BitmapPointer::BitmapPointer(HBITMAP value) : _value(value) {
 | 
						|
}
 | 
						|
 | 
						|
MainWindow::BitmapPointer::BitmapPointer(BitmapPointer &&other)
 | 
						|
: _value(base::take(other._value)) {
 | 
						|
}
 | 
						|
 | 
						|
MainWindow::BitmapPointer &MainWindow::BitmapPointer::operator=(
 | 
						|
		BitmapPointer &&other) {
 | 
						|
	if (_value != other._value) {
 | 
						|
		reset();
 | 
						|
		_value = base::take(other._value);
 | 
						|
	}
 | 
						|
	return *this;
 | 
						|
}
 | 
						|
 | 
						|
MainWindow::BitmapPointer::~BitmapPointer() {
 | 
						|
	reset();
 | 
						|
}
 | 
						|
 | 
						|
HBITMAP MainWindow::BitmapPointer::get() const {
 | 
						|
	return _value;
 | 
						|
}
 | 
						|
 | 
						|
MainWindow::BitmapPointer::operator bool() const {
 | 
						|
	return _value != nullptr;
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::BitmapPointer::release() {
 | 
						|
	_value = nullptr;
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::BitmapPointer::reset(HBITMAP value) {
 | 
						|
	if (_value != value) {
 | 
						|
		if (const auto old = std::exchange(_value, value)) {
 | 
						|
			DeleteObject(old);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
MainWindow::MainWindow(not_null<Window::Controller*> controller)
 | 
						|
: Window::MainWindow(controller)
 | 
						|
, _private(std::make_unique<Private>(this))
 | 
						|
, _taskbarHiderWindow(std::make_unique<QWindow>()) {
 | 
						|
	qApp->installNativeEventFilter(&_private->filter);
 | 
						|
 | 
						|
	setupNativeWindowFrame();
 | 
						|
 | 
						|
	SetWindowPriority(this, controller->isPrimary() ? 2 : 1);
 | 
						|
 | 
						|
	using namespace rpl::mappers;
 | 
						|
	Core::App().appDeactivatedValue(
 | 
						|
	) | rpl::distinct_until_changed(
 | 
						|
	) | rpl::filter(_1) | rpl::start_with_next([=] {
 | 
						|
		_lastDeactivateTime = crl::now();
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	setupPreviewPasscodeLock();
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::setupPreviewPasscodeLock() {
 | 
						|
	Core::App().passcodeLockValue(
 | 
						|
	) | rpl::start_with_next([=](bool locked) {
 | 
						|
		// Use iconic bitmap instead of the window content if passcoded.
 | 
						|
		BOOL fForceIconic = locked ? TRUE : FALSE;
 | 
						|
		BOOL fHasIconicBitmap = fForceIconic;
 | 
						|
		DwmSetWindowAttribute(
 | 
						|
			_hWnd,
 | 
						|
			DWMWA_FORCE_ICONIC_REPRESENTATION,
 | 
						|
			&fForceIconic,
 | 
						|
			sizeof(fForceIconic));
 | 
						|
		DwmSetWindowAttribute(
 | 
						|
			_hWnd,
 | 
						|
			DWMWA_HAS_ICONIC_BITMAP,
 | 
						|
			&fHasIconicBitmap,
 | 
						|
			sizeof(fHasIconicBitmap));
 | 
						|
	}, lifetime());
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::setupNativeWindowFrame() {
 | 
						|
	auto nativeFrame = rpl::single(
 | 
						|
		Core::App().settings().nativeWindowFrame()
 | 
						|
	) | rpl::then(
 | 
						|
		Core::App().settings().nativeWindowFrameChanges()
 | 
						|
	);
 | 
						|
 | 
						|
	rpl::combine(
 | 
						|
		std::move(nativeFrame),
 | 
						|
		Window::Theme::IsNightModeValue()
 | 
						|
	) | rpl::skip(1) | rpl::start_with_next([=](bool native, bool night) {
 | 
						|
		validateWindowTheme(native, night);
 | 
						|
	}, lifetime());
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::shadowsActivate() {
 | 
						|
	_hasActiveFrame = true;
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::shadowsDeactivate() {
 | 
						|
	_hasActiveFrame = false;
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::destroyedFromSystem() {
 | 
						|
	if (!Core::App().closeNonLastAsync(&controller())) {
 | 
						|
		Core::Quit();
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
bool MainWindow::setDwmThumbnail(QSize size) {
 | 
						|
	validateDwmPreviewColors();
 | 
						|
	if (size.isEmpty()) {
 | 
						|
		return false;
 | 
						|
	} else if (!_dwmThumbnail || _dwmThumbnailSize != size) {
 | 
						|
		const auto result = PrepareLogoPreview(size, QImage::Format_RGB32);
 | 
						|
		const auto bitmap = qt_imageToWinHBITMAP(result, HBitmapNoAlpha);
 | 
						|
		if (!bitmap) {
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		_dwmThumbnail.reset(bitmap);
 | 
						|
		_dwmThumbnailSize = size;
 | 
						|
	}
 | 
						|
	DwmSetIconicThumbnail(_hWnd, _dwmThumbnail.get(), NULL);
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
bool MainWindow::setDwmPreview(QSize size, int radius) {
 | 
						|
	Expects(radius >= 0);
 | 
						|
 | 
						|
	validateDwmPreviewColors();
 | 
						|
	if (size.isEmpty()) {
 | 
						|
		return false;
 | 
						|
	} else if (!_dwmPreview
 | 
						|
		|| _dwmPreviewSize != size
 | 
						|
		|| _dwmPreviewRadius != radius) {
 | 
						|
		const auto format = (radius > 0)
 | 
						|
			? QImage::Format_ARGB32_Premultiplied
 | 
						|
			: QImage::Format_RGB32;
 | 
						|
		const auto result = PrepareLogoPreview(size, format, radius);
 | 
						|
		const auto bitmap = qt_imageToWinHBITMAP(
 | 
						|
			result,
 | 
						|
			(radius > 0) ? HBitmapPremultipliedAlpha : HBitmapNoAlpha);
 | 
						|
		if (!bitmap) {
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		_dwmPreview.reset(bitmap);
 | 
						|
		_dwmPreviewRadius = radius;
 | 
						|
		_dwmPreviewSize = size;
 | 
						|
	}
 | 
						|
	const auto flags = 0;
 | 
						|
	DwmSetIconicLivePreviewBitmap(_hWnd, _dwmPreview.get(), NULL, flags);
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::validateDwmPreviewColors() {
 | 
						|
	if (_dwmBackground == st::windowBg->c) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	_dwmBackground = st::windowBg->c;
 | 
						|
	_dwmThumbnail.reset();
 | 
						|
	_dwmPreview.reset();
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::forceIconRefresh() {
 | 
						|
	const auto refresher = std::make_unique<QWidget>(this);
 | 
						|
	refresher->setWindowFlags(
 | 
						|
		static_cast<Qt::WindowFlags>(Qt::Tool) | Qt::FramelessWindowHint);
 | 
						|
	refresher->setGeometry(x() + 1, y() + 1, 1, 1);
 | 
						|
	auto palette = refresher->palette();
 | 
						|
	palette.setColor(
 | 
						|
		QPalette::Window,
 | 
						|
		(isActiveWindow() ? st::titleBgActive : st::titleBg)->c);
 | 
						|
	refresher->setPalette(palette);
 | 
						|
	refresher->show();
 | 
						|
	refresher->raise();
 | 
						|
	refresher->activateWindow();
 | 
						|
 | 
						|
	updateTaskbarAndIconCounters();
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::workmodeUpdated(Core::Settings::WorkMode mode) {
 | 
						|
	using WorkMode = Core::Settings::WorkMode;
 | 
						|
 | 
						|
	switch (mode) {
 | 
						|
	case WorkMode::WindowAndTray: {
 | 
						|
		HWND psOwner = (HWND)GetWindowLongPtr(_hWnd, GWLP_HWNDPARENT);
 | 
						|
		if (psOwner) {
 | 
						|
			SetWindowLongPtr(_hWnd, GWLP_HWNDPARENT, 0);
 | 
						|
			windowHandle()->setTransientParent(nullptr);
 | 
						|
			forceIconRefresh();
 | 
						|
		}
 | 
						|
	} break;
 | 
						|
 | 
						|
	case WorkMode::TrayOnly: {
 | 
						|
		HWND psOwner = (HWND)GetWindowLongPtr(_hWnd, GWLP_HWNDPARENT);
 | 
						|
		if (!psOwner) {
 | 
						|
			const auto hwnd = _taskbarHiderWindow->winId();
 | 
						|
			SetWindowLongPtr(_hWnd, GWLP_HWNDPARENT, (LONG_PTR)hwnd);
 | 
						|
			windowHandle()->setTransientParent(_taskbarHiderWindow.get());
 | 
						|
		}
 | 
						|
	} break;
 | 
						|
 | 
						|
	case WorkMode::WindowOnly: {
 | 
						|
		HWND psOwner = (HWND)GetWindowLongPtr(_hWnd, GWLP_HWNDPARENT);
 | 
						|
		if (psOwner) {
 | 
						|
			SetWindowLongPtr(_hWnd, GWLP_HWNDPARENT, 0);
 | 
						|
			windowHandle()->setTransientParent(nullptr);
 | 
						|
			forceIconRefresh();
 | 
						|
		}
 | 
						|
	} break;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
bool MainWindow::hasTabletView() const {
 | 
						|
	if (!_private->viewSettings) {
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	auto mode = ViewManagement::UserInteractionMode();
 | 
						|
	_private->viewSettings->get_UserInteractionMode(&mode);
 | 
						|
	return (mode == ViewManagement::UserInteractionMode_Touch);
 | 
						|
}
 | 
						|
 | 
						|
bool MainWindow::initGeometryFromSystem() {
 | 
						|
	if (!hasTabletView() || !screen()) {
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	Ui::RpWidget::setGeometry(screen()->availableGeometry());
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::updateWindowIcon() {
 | 
						|
	updateTaskbarAndIconCounters();
 | 
						|
}
 | 
						|
 | 
						|
bool MainWindow::isActiveForTrayMenu() {
 | 
						|
	return !_lastDeactivateTime
 | 
						|
		|| (_lastDeactivateTime + kKeepActiveForTrayIcon >= crl::now());
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::unreadCounterChangedHook() {
 | 
						|
	updateTaskbarAndIconCounters();
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::updateTaskbarAndIconCounters() {
 | 
						|
	const auto counter = Core::App().unreadBadge();
 | 
						|
	const auto muted = Core::App().unreadBadgeMuted();
 | 
						|
	const auto controller = sessionController();
 | 
						|
	const auto session = controller ? &controller->session() : nullptr;
 | 
						|
 | 
						|
	const auto iconSizeSmall = QSize(
 | 
						|
		GetSystemMetrics(SM_CXSMICON),
 | 
						|
		GetSystemMetrics(SM_CYSMICON));
 | 
						|
	const auto iconSizeBig = QSize(
 | 
						|
		GetSystemMetrics(SM_CXICON),
 | 
						|
		GetSystemMetrics(SM_CYICON));
 | 
						|
	const auto supportMode = session && session->supportMode();
 | 
						|
 | 
						|
	auto iconSmallPixmap16 = Tray::IconWithCounter(
 | 
						|
		Tray::CounterLayerArgs(16, counter, muted),
 | 
						|
		true,
 | 
						|
		supportMode);
 | 
						|
	auto iconSmallPixmap32 = Tray::IconWithCounter(
 | 
						|
		Tray::CounterLayerArgs(32, counter, muted),
 | 
						|
		true,
 | 
						|
		supportMode);
 | 
						|
	QIcon iconSmall, iconBig;
 | 
						|
	iconSmall.addPixmap(iconSmallPixmap16);
 | 
						|
	iconSmall.addPixmap(iconSmallPixmap32);
 | 
						|
	const auto integration = &Platform::WindowsIntegration::Instance();
 | 
						|
	const auto taskbarList = integration->taskbarList();
 | 
						|
	const auto bigCounter = taskbarList ? 0 : counter;
 | 
						|
	iconBig.addPixmap(Tray::IconWithCounter(
 | 
						|
		Tray::CounterLayerArgs(32, bigCounter, muted),
 | 
						|
		false,
 | 
						|
		supportMode));
 | 
						|
	iconBig.addPixmap(Tray::IconWithCounter(
 | 
						|
		Tray::CounterLayerArgs(64, bigCounter, muted),
 | 
						|
		false,
 | 
						|
		supportMode));
 | 
						|
 | 
						|
	destroyCachedIcons();
 | 
						|
	_iconSmall = NativeIcon(iconSmall, iconSizeSmall);
 | 
						|
	_iconBig = NativeIcon(iconBig, iconSizeBig);
 | 
						|
	SendMessage(_hWnd, WM_SETICON, ICON_SMALL, (LPARAM)_iconSmall);
 | 
						|
	SendMessage(_hWnd, WM_SETICON, ICON_BIG, (LPARAM)(_iconBig ? _iconBig : _iconSmall));
 | 
						|
	if (taskbarList) {
 | 
						|
		if (counter > 0) {
 | 
						|
			const auto pixmap = [&](int size) {
 | 
						|
				return Ui::PixmapFromImage(Window::GenerateCounterLayer(
 | 
						|
					Tray::CounterLayerArgs(size, counter, muted)));
 | 
						|
			};
 | 
						|
			QIcon iconOverlay;
 | 
						|
			iconOverlay.addPixmap(pixmap(16));
 | 
						|
			iconOverlay.addPixmap(pixmap(32));
 | 
						|
			_iconOverlay = NativeIcon(iconOverlay, iconSizeSmall);
 | 
						|
		}
 | 
						|
		const auto description = (counter > 0)
 | 
						|
			? tr::lng_unread_bar(tr::now, lt_count, counter).toStdWString()
 | 
						|
			: std::wstring();
 | 
						|
		taskbarList->SetOverlayIcon(_hWnd, _iconOverlay, description.c_str());
 | 
						|
	}
 | 
						|
	SetWindowPos(_hWnd, 0, 0, 0, 0, 0, SWP_FRAMECHANGED | SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_NOACTIVATE);
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::initHook() {
 | 
						|
	_hWnd = reinterpret_cast<HWND>(winId());
 | 
						|
	if (!_hWnd) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	WTSRegisterSessionNotification(_hWnd, NOTIFY_FOR_THIS_SESSION);
 | 
						|
 | 
						|
	using namespace base::Platform;
 | 
						|
	auto factory = ComPtr<IUIViewSettingsInterop>();
 | 
						|
	if (SupportsWRL()) {
 | 
						|
		ABI::Windows::Foundation::GetActivationFactory(
 | 
						|
			StringReferenceWrapper(
 | 
						|
				RuntimeClass_Windows_UI_ViewManagement_UIViewSettings).Get(),
 | 
						|
			&factory);
 | 
						|
		if (factory) {
 | 
						|
			// NB! No such method (or IUIViewSettingsInterop) in C++/WinRT :(
 | 
						|
			factory->GetForWindow(
 | 
						|
				_hWnd,
 | 
						|
				IID_PPV_ARGS(&_private->viewSettings));
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	validateWindowTheme(
 | 
						|
		Core::App().settings().nativeWindowFrame(),
 | 
						|
		Window::Theme::IsNightMode());
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::validateWindowTheme(bool native, bool night) {
 | 
						|
	if (!IsWindows8OrGreater()) {
 | 
						|
		const auto empty = native ? nullptr : L" ";
 | 
						|
		SetWindowTheme(_hWnd, empty, empty);
 | 
						|
		QApplication::setStyle(QStyleFactory::create(u"Windows"_q));
 | 
						|
#if 0
 | 
						|
	} else if (!Core::App().settings().systemDarkMode().has_value()/*
 | 
						|
		|| (!Dlls::AllowDarkModeForApp && !Dlls::SetPreferredAppMode)
 | 
						|
		|| !Dlls::AllowDarkModeForWindow
 | 
						|
		|| !Dlls::RefreshImmersiveColorPolicyState
 | 
						|
		|| !Dlls::FlushMenuThemes*/) {
 | 
						|
		return;
 | 
						|
#endif
 | 
						|
	} else if (!native) {
 | 
						|
		SetWindowTheme(_hWnd, nullptr, nullptr);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	// See "https://github.com/microsoft/terminal/blob/"
 | 
						|
	// "eb480b6bbbd83a2aafbe62992d360838e0ab9da5/"
 | 
						|
	// "src/interactivity/win32/windowtheme.cpp#L43-L63"
 | 
						|
 | 
						|
	auto darkValue = BOOL(night ? TRUE : FALSE);
 | 
						|
 | 
						|
	const auto updateStyle = [&] {
 | 
						|
		static const auto kSystemVersion = QOperatingSystemVersion::current();
 | 
						|
		if (kSystemVersion.microVersion() >= 18875 && Dlls::SetWindowCompositionAttribute) {
 | 
						|
			Dlls::WINDOWCOMPOSITIONATTRIBDATA data = {
 | 
						|
				Dlls::WINDOWCOMPOSITIONATTRIB::WCA_USEDARKMODECOLORS,
 | 
						|
				&darkValue,
 | 
						|
				sizeof(darkValue)
 | 
						|
			};
 | 
						|
			Dlls::SetWindowCompositionAttribute(_hWnd, &data);
 | 
						|
		} else if (kSystemVersion.microVersion() >= 17763) {
 | 
						|
			static const auto kDWMWA_USE_IMMERSIVE_DARK_MODE = (kSystemVersion.microVersion() >= 18985)
 | 
						|
				? DWORD(20)
 | 
						|
				: DWORD(19);
 | 
						|
			DwmSetWindowAttribute(
 | 
						|
				_hWnd,
 | 
						|
				kDWMWA_USE_IMMERSIVE_DARK_MODE,
 | 
						|
				&darkValue,
 | 
						|
				sizeof(darkValue));
 | 
						|
		}
 | 
						|
	};
 | 
						|
 | 
						|
	updateStyle();
 | 
						|
 | 
						|
	// See "https://osdn.net/projects/tortoisesvn/scm/svn/blobs/28812/"
 | 
						|
	// "trunk/src/TortoiseIDiff/MainWindow.cpp"
 | 
						|
	//
 | 
						|
	// But for now it works event with a small part of that.
 | 
						|
	//
 | 
						|
 | 
						|
	//const auto updateWindowTheme = [&] {
 | 
						|
	//	const auto set = [&](LPCWSTR name) {
 | 
						|
	//		return SetWindowTheme(_hWnd, name, nullptr);
 | 
						|
	//	};
 | 
						|
	//	if (!night || FAILED(set(L"DarkMode_Explorer"))) {
 | 
						|
	//		set(L"Explorer");
 | 
						|
	//	}
 | 
						|
	//};
 | 
						|
	//
 | 
						|
	//if (night) {
 | 
						|
	//	if (Dlls::SetPreferredAppMode) {
 | 
						|
	//		Dlls::SetPreferredAppMode(Dlls::PreferredAppMode::AllowDark);
 | 
						|
	//	} else {
 | 
						|
	//		Dlls::AllowDarkModeForApp(TRUE);
 | 
						|
	//	}
 | 
						|
	//	Dlls::AllowDarkModeForWindow(_hWnd, TRUE);
 | 
						|
	//	updateWindowTheme();
 | 
						|
	//	updateStyle();
 | 
						|
	//	Dlls::FlushMenuThemes();
 | 
						|
	//	Dlls::RefreshImmersiveColorPolicyState();
 | 
						|
	//} else {
 | 
						|
	//	updateWindowTheme();
 | 
						|
	//	Dlls::AllowDarkModeForWindow(_hWnd, FALSE);
 | 
						|
	//	updateStyle();
 | 
						|
	//	Dlls::FlushMenuThemes();
 | 
						|
	//	Dlls::RefreshImmersiveColorPolicyState();
 | 
						|
	//	if (Dlls::SetPreferredAppMode) {
 | 
						|
	//		Dlls::SetPreferredAppMode(Dlls::PreferredAppMode::Default);
 | 
						|
	//	} else {
 | 
						|
	//		Dlls::AllowDarkModeForApp(FALSE);
 | 
						|
	//	}
 | 
						|
	//}
 | 
						|
 | 
						|
	// Didn't find any other way to definitely repaint with the new style.
 | 
						|
	SendMessage(_hWnd, WM_NCACTIVATE, _hasActiveFrame ? 0 : 1, 0);
 | 
						|
	SendMessage(_hWnd, WM_NCACTIVATE, _hasActiveFrame ? 1 : 0, 0);
 | 
						|
}
 | 
						|
 | 
						|
HWND MainWindow::psHwnd() const {
 | 
						|
	return _hWnd;
 | 
						|
}
 | 
						|
 | 
						|
void MainWindow::destroyCachedIcons() {
 | 
						|
	const auto destroy = [](HICON &icon) {
 | 
						|
		if (icon) {
 | 
						|
			DestroyIcon(icon);
 | 
						|
			icon = nullptr;
 | 
						|
		}
 | 
						|
	};
 | 
						|
	destroy(_iconBig);
 | 
						|
	destroy(_iconSmall);
 | 
						|
	destroy(_iconOverlay);
 | 
						|
}
 | 
						|
 | 
						|
MainWindow::~MainWindow() {
 | 
						|
	WTSUnRegisterSessionNotification(_hWnd);
 | 
						|
	_private->viewSettings.Reset();
 | 
						|
	destroyCachedIcons();
 | 
						|
}
 | 
						|
 | 
						|
int32 ScreenNameChecksum(const QString &name) {
 | 
						|
	constexpr int DeviceNameSize = base::array_size(MONITORINFOEX().szDevice);
 | 
						|
	wchar_t buffer[DeviceNameSize] = { 0 };
 | 
						|
	if (name.size() < DeviceNameSize) {
 | 
						|
		name.toWCharArray(buffer);
 | 
						|
	} else {
 | 
						|
		memcpy(buffer, name.toStdWString().data(), sizeof(buffer));
 | 
						|
	}
 | 
						|
	return base::crc32(buffer, sizeof(buffer));
 | 
						|
}
 | 
						|
 | 
						|
} // namespace Platform
 |