280 lines
		
	
	
	
		
			7.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			280 lines
		
	
	
	
		
			7.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // This file is part of Desktop App Toolkit,
 | |
| // a set of libraries for developing nice desktop applications.
 | |
| //
 | |
| // For license and copyright information please follow this link:
 | |
| // https://github.com/desktop-app/legal/blob/master/LEGAL
 | |
| //
 | |
| #include "ui/platform/win/ui_window_title_win.h"
 | |
| 
 | |
| #include "ui/platform/win/ui_window_win.h"
 | |
| #include "ui/widgets/buttons.h"
 | |
| #include "ui/widgets/shadow.h"
 | |
| #include "ui/widgets/rp_window.h"
 | |
| #include "ui/ui_utility.h"
 | |
| #include "base/platform/base_platform_info.h"
 | |
| #include "base/platform/win/base_windows_safe_library.h"
 | |
| #include "base/debug_log.h"
 | |
| #include "styles/style_widgets.h"
 | |
| #include "styles/palette.h"
 | |
| 
 | |
| #include <QtGui/QPainter>
 | |
| #include <QtGui/QtEvents>
 | |
| #include <QtGui/QWindow>
 | |
| 
 | |
| #include <windows.h>
 | |
| #include <shellscalingapi.h>
 | |
| 
 | |
| namespace Ui {
 | |
| namespace Platform {
 | |
| namespace {
 | |
| 
 | |
| HRESULT(__stdcall *GetScaleFactorForMonitor)(
 | |
| 	_In_ HMONITOR hMon,
 | |
| 	_Out_ DEVICE_SCALE_FACTOR *pScale);
 | |
| 
 | |
| [[nodiscard]] bool ScaleQuerySupported() {
 | |
| 	static const auto Result = [&] {
 | |
| #define LOAD_SYMBOL(lib, name) base::Platform::LoadMethod(lib, #name, name)
 | |
| 		const auto shcore = base::Platform::SafeLoadLibrary(L"Shcore.dll");
 | |
| 		return LOAD_SYMBOL(shcore, GetScaleFactorForMonitor);
 | |
| #undef LOAD_SYMBOL
 | |
| 	}();
 | |
| 	return Result;
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| namespace internal {
 | |
| 
 | |
| TitleControls::Layout TitleControlsLayout() {
 | |
| 	return TitleControls::Layout{
 | |
| 		.right = {
 | |
| 			TitleControls::Control::Minimize,
 | |
| 			TitleControls::Control::Maximize,
 | |
| 			TitleControls::Control::Close,
 | |
| 		}
 | |
| 	};
 | |
| }
 | |
| 
 | |
| } // namespace internal
 | |
| 
 | |
| struct TitleWidget::PaddingHelper {
 | |
| 	explicit PaddingHelper(QWidget *parent) : controlsParent(parent) {
 | |
| 	}
 | |
| 
 | |
| 	RpWidget controlsParent;
 | |
| 	rpl::variable<int> padding = 0;
 | |
| };
 | |
| 
 | |
| TitleWidget::TitleWidget(not_null<RpWidget*> parent)
 | |
| : RpWidget(parent)
 | |
| , _paddingHelper(CheckTitlePaddingRequired()
 | |
| 	? std::make_unique<PaddingHelper>(this)
 | |
| 	: nullptr)
 | |
| , _controls(
 | |
| 	_paddingHelper ? &_paddingHelper->controlsParent : this,
 | |
| 	st::defaultWindowTitle)
 | |
| , _shadow(this, st::titleShadow) {
 | |
| 	setAttribute(Qt::WA_OpaquePaintEvent);
 | |
| 
 | |
| 	parent->widthValue(
 | |
| 	) | rpl::start_with_next([=](int width) {
 | |
| 		refreshGeometryWithWidth(width);
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void TitleWidget::initInWindow(not_null<RpWindow*> window) {
 | |
| 	window->hitTestRequests(
 | |
| 	) | rpl::filter([=](not_null<HitTestRequest*> request) {
 | |
| 		return !isHidden() && geometry().contains(request->point);
 | |
| 	}) | rpl::start_with_next([=](not_null<HitTestRequest*> request) {
 | |
| 		request->result = hitTest(request->point, request->result);
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	SetupSemiNativeSystemButtons(&_controls, window, lifetime(), [=] {
 | |
| 		return !isHidden() && (_controls.st()->height > 0);
 | |
| 	});
 | |
| }
 | |
| 
 | |
| TitleWidget::~TitleWidget() = default;
 | |
| 
 | |
| void TitleWidget::setText(const QString &text) {
 | |
| 	window()->setWindowTitle(text);
 | |
| }
 | |
| 
 | |
| void TitleWidget::setStyle(const style::WindowTitle &st) {
 | |
| 	_controls.setStyle(st);
 | |
| 	if (!st.shadow) {
 | |
| 		_shadow.destroy();
 | |
| 	} else if (!_shadow) {
 | |
| 		_shadow.create(this, st::titleShadow);
 | |
| 		updateShadowGeometry();
 | |
| 	}
 | |
| 	refreshGeometryWithWidth(window()->width());
 | |
| }
 | |
| 
 | |
| void TitleWidget::updateShadowGeometry() {
 | |
| 	const auto thickness = st::lineWidth;
 | |
| 	_shadow->setGeometry(0, height() - thickness, width(), thickness);
 | |
| }
 | |
| 
 | |
| void TitleWidget::refreshGeometryWithWidth(int width) {
 | |
| 	const auto add = additionalPadding();
 | |
| 	setGeometry(0, 0, width, _controls.st()->height + add);
 | |
| 	if (_paddingHelper) {
 | |
| 		_paddingHelper->controlsParent.setGeometry(
 | |
| 			0,
 | |
| 			add,
 | |
| 			width,
 | |
| 			_controls.st()->height);
 | |
| 	}
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| not_null<const style::WindowTitle*> TitleWidget::st() const {
 | |
| 	return _controls.st();
 | |
| }
 | |
| 
 | |
| void TitleWidget::setResizeEnabled(bool enabled) {
 | |
| 	_controls.setResizeEnabled(enabled);
 | |
| }
 | |
| 
 | |
| void TitleWidget::paintEvent(QPaintEvent *e) {
 | |
| 	const auto active = window()->isActiveWindow();
 | |
| 	QPainter(this).fillRect(
 | |
| 		e->rect(),
 | |
| 		active ? _controls.st()->bgActive : _controls.st()->bg);
 | |
| }
 | |
| 
 | |
| void TitleWidget::resizeEvent(QResizeEvent *e) {
 | |
| 	if (_shadow) {
 | |
| 		updateShadowGeometry();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| HitTestResult TitleWidget::hitTest(
 | |
| 		QPoint point,
 | |
| 		HitTestResult oldResult) const {
 | |
| 	const auto origin = _paddingHelper
 | |
| 		? _paddingHelper->controlsParent.pos()
 | |
| 		: QPoint();
 | |
| 	const auto padding = _paddingHelper
 | |
| 		? _paddingHelper->padding.current()
 | |
| 		: 0;
 | |
| 	const auto controlsResult = _controls.hitTest(point - origin, padding);
 | |
| 	return (controlsResult != HitTestResult::None)
 | |
| 		? controlsResult
 | |
| 		: (oldResult != HitTestResult::Client)
 | |
| 		? oldResult
 | |
| 		: HitTestResult::Caption;
 | |
| }
 | |
| 
 | |
| bool TitleWidget::additionalPaddingRequired() const {
 | |
| 	return _paddingHelper && !isHidden();
 | |
| }
 | |
| 
 | |
| void TitleWidget::refreshAdditionalPaddings() {
 | |
| 	if (!additionalPaddingRequired()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto handle = GetWindowHandle(this);
 | |
| 	if (!handle) {
 | |
| 		LOG(("System Error: GetWindowHandle failed."));
 | |
| 		return;
 | |
| 	}
 | |
| 	refreshAdditionalPaddings(handle);
 | |
| }
 | |
| 
 | |
| void TitleWidget::refreshAdditionalPaddings(HWND handle) {
 | |
| 	if (!additionalPaddingRequired()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	auto placement = WINDOWPLACEMENT{
 | |
| 		.length = sizeof(WINDOWPLACEMENT),
 | |
| 	};
 | |
| 	if (!GetWindowPlacement(handle, &placement)) {
 | |
| 		LOG(("System Error: GetWindowPlacement failed."));
 | |
| 		return;
 | |
| 	}
 | |
| 	refreshAdditionalPaddings(handle, placement);
 | |
| }
 | |
| 
 | |
| void TitleWidget::refreshAdditionalPaddings(
 | |
| 		HWND handle,
 | |
| 		const WINDOWPLACEMENT &placement) {
 | |
| 	if (!additionalPaddingRequired()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	auto geometry = RECT();
 | |
| 	if (!GetWindowRect(handle, &geometry)) {
 | |
| 		LOG(("System Error: GetWindowRect failed."));
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto normal = placement.rcNormalPosition;
 | |
| 	const auto rounded = (normal.left == geometry.left)
 | |
| 		&& (normal.right == geometry.right)
 | |
| 		&& (normal.top == geometry.top)
 | |
| 		&& (normal.bottom == geometry.bottom);
 | |
| 	const auto padding = [&] {
 | |
| 		if (!rounded) {
 | |
| 			return 0;
 | |
| 		}
 | |
| 		const auto monitor = MonitorFromWindow(
 | |
| 			handle,
 | |
| 			MONITOR_DEFAULTTONEAREST);
 | |
| 		if (!monitor) {
 | |
| 			LOG(("System Error: MonitorFromWindow failed."));
 | |
| 			return -1;
 | |
| 		}
 | |
| 		auto factor = DEVICE_SCALE_FACTOR();
 | |
| 		if (!SUCCEEDED(GetScaleFactorForMonitor(monitor, &factor))) {
 | |
| 			LOG(("System Error: GetScaleFactorForMonitor failed."));
 | |
| 			return -1;
 | |
| 		} else if (factor < 100 || factor > 500) {
 | |
| 			LOG(("System Error: Bad scale factor %1.").arg(int(factor)));
 | |
| 			return -1;
 | |
| 		}
 | |
| 		const auto pixels = (factor + 50) / 100;
 | |
| 		return int(base::SafeRound(
 | |
| 			pixels / window()->windowHandle()->devicePixelRatio()));
 | |
| 	}();
 | |
| 	if (padding < 0) {
 | |
| 		return;
 | |
| 	}
 | |
| 	setAdditionalPadding(padding);
 | |
| }
 | |
| 
 | |
| int TitleWidget::additionalPadding() const {
 | |
| 	return _paddingHelper ? _paddingHelper->padding.current() : 0;
 | |
| }
 | |
| 
 | |
| rpl::producer<int> TitleWidget::additionalPaddingValue() const {
 | |
| 	return _paddingHelper ? _paddingHelper->padding.value() : rpl::single(0);
 | |
| }
 | |
| 
 | |
| void TitleWidget::setAdditionalPadding(int padding) {
 | |
| 	Expects(_paddingHelper != nullptr);
 | |
| 
 | |
| 	padding /= window()->devicePixelRatio();
 | |
| 	if (_paddingHelper->padding.current() == padding) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_paddingHelper->padding = padding;
 | |
| 	refreshGeometryWithWidth(window()->width());
 | |
| }
 | |
| 
 | |
| void TitleWidget::setVisibleHook(bool visible) {
 | |
| 	RpWidget::setVisibleHook(visible);
 | |
| 	if (additionalPaddingRequired()) {
 | |
| 		PostponeCall(this, [=] {
 | |
| 			refreshAdditionalPaddings();
 | |
| 		});
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool CheckTitlePaddingRequired() {
 | |
| 	return ::Platform::IsWindows11OrGreater() && ScaleQuerySupported();
 | |
| }
 | |
| 
 | |
| } // namespace Platform
 | |
| } // namespace Ui
 | 
