By default, Qt uses swap interval of 1 what causes the (egl|wgl|glx)SwapBuffers call to wait for vblank. Using swap interval of 0 makes the call to return immediately.
		
			
				
	
	
		
			250 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			250 lines
		
	
	
	
		
			6.3 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/rp_widget.h"
 | 
						|
 | 
						|
#include "base/platform/base_platform_info.h"
 | 
						|
#include "base/qt_signal_producer.h"
 | 
						|
#include "ui/gl/gl_detection.h"
 | 
						|
 | 
						|
#include <QtGui/QWindow>
 | 
						|
#include <QtGui/QtEvents>
 | 
						|
#include <private/qwidget_p.h>
 | 
						|
 | 
						|
class TWidgetPrivate : public QWidgetPrivate {
 | 
						|
public:
 | 
						|
#if QT_VERSION >= QT_VERSION_CHECK(6, 4, 0)
 | 
						|
	QPlatformBackingStoreRhiConfig rhiConfig() const override {
 | 
						|
		const auto q = static_cast<TWidget*>(q_ptr);
 | 
						|
		if (!q->testAttribute(Qt::WA_WState_Created)) {
 | 
						|
			return QWidgetPrivate::rhiConfig();
 | 
						|
		}
 | 
						|
		if (const auto config = q->rhiConfig()) {
 | 
						|
			return *config;
 | 
						|
		}
 | 
						|
		if (::Platform::IsMac10_14OrGreater()) {
 | 
						|
			return { QPlatformBackingStoreRhiConfig::Metal };
 | 
						|
		}
 | 
						|
		// We can't specify the widget here as q_evaluateRhiConfig is called
 | 
						|
		// in QWidgetWindow constructor, while windowHandle is set right after
 | 
						|
		// the constructor is completed
 | 
						|
		if (::Platform::IsWayland() // old versions of mutter produce flicker without OpenGL
 | 
						|
			&& Ui::GL::ChooseBackendDefault(
 | 
						|
					Ui::GL::CheckCapabilities(nullptr))
 | 
						|
				== Ui::GL::Backend::OpenGL) {
 | 
						|
			return { QPlatformBackingStoreRhiConfig::OpenGL };
 | 
						|
		}
 | 
						|
		return QWidgetPrivate::rhiConfig();
 | 
						|
	}
 | 
						|
#endif // Qt >= 6.4.0
 | 
						|
};
 | 
						|
 | 
						|
TWidget::TWidget(QWidget *parent)
 | 
						|
: TWidgetHelper<QWidget>(*(new TWidgetPrivate), parent, {}) {
 | 
						|
	[[maybe_unused]] static const auto Once = [] {
 | 
						|
		auto format = QSurfaceFormat::defaultFormat();
 | 
						|
		format.setSwapInterval(0);
 | 
						|
		QSurfaceFormat::setDefaultFormat(format);
 | 
						|
		return true;
 | 
						|
	}();
 | 
						|
}
 | 
						|
 | 
						|
namespace Ui {
 | 
						|
namespace {
 | 
						|
 | 
						|
[[nodiscard]] std::vector<QPointer<QWidget>> GetChildWidgets(
 | 
						|
		not_null<QWidget*> widget) {
 | 
						|
	const auto &children = widget->children();
 | 
						|
	auto result = std::vector<QPointer<QWidget>>();
 | 
						|
	result.reserve(children.size());
 | 
						|
	for (const auto child : children) {
 | 
						|
		if (child && child->isWidgetType()) {
 | 
						|
			result.push_back(static_cast<QWidget*>(child));
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return result;
 | 
						|
}
 | 
						|
 | 
						|
} // namespace
 | 
						|
 | 
						|
void ToggleChildrenVisibility(not_null<QWidget*> widget, bool visible) {
 | 
						|
	for (const auto &child : GetChildWidgets(widget)) {
 | 
						|
		if (child) {
 | 
						|
			child->setVisible(visible);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void ResizeFitChild(
 | 
						|
		not_null<RpWidget*> parent,
 | 
						|
		not_null<RpWidget*> child) {
 | 
						|
	parent->widthValue(
 | 
						|
	) | rpl::start_with_next([=](int width) {
 | 
						|
		child->resizeToWidth(width);
 | 
						|
	}, child->lifetime());
 | 
						|
 | 
						|
	child->heightValue(
 | 
						|
	) | rpl::start_with_next([=](int height) {
 | 
						|
		parent->resize(parent->width(), height);
 | 
						|
	}, child->lifetime());
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<not_null<QEvent*>> RpWidgetWrap::events() const {
 | 
						|
	auto &stream = eventStreams().events;
 | 
						|
	return stream.events();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<QRect> RpWidgetWrap::geometryValue() const {
 | 
						|
	auto &stream = eventStreams().geometry;
 | 
						|
	return stream.events_starting_with_copy(rpWidget()->geometry());
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<QSize> RpWidgetWrap::sizeValue() const {
 | 
						|
	return geometryValue()
 | 
						|
		| rpl::map([](QRect &&value) { return value.size(); })
 | 
						|
		| rpl::distinct_until_changed();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<int> RpWidgetWrap::heightValue() const {
 | 
						|
	return geometryValue()
 | 
						|
		| rpl::map([](QRect &&value) { return value.height(); })
 | 
						|
		| rpl::distinct_until_changed();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<int> RpWidgetWrap::widthValue() const {
 | 
						|
	return geometryValue()
 | 
						|
		| rpl::map([](QRect &&value) { return value.width(); })
 | 
						|
		| rpl::distinct_until_changed();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<QPoint> RpWidgetWrap::positionValue() const {
 | 
						|
	return geometryValue()
 | 
						|
		| rpl::map([](QRect &&value) { return value.topLeft(); })
 | 
						|
		| rpl::distinct_until_changed();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<int> RpWidgetWrap::leftValue() const {
 | 
						|
	return geometryValue()
 | 
						|
		| rpl::map([](QRect &&value) { return value.left(); })
 | 
						|
		| rpl::distinct_until_changed();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<int> RpWidgetWrap::topValue() const {
 | 
						|
	return geometryValue()
 | 
						|
		| rpl::map([](QRect &&value) { return value.top(); })
 | 
						|
		| rpl::distinct_until_changed();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<int> RpWidgetWrap::desiredHeightValue() const {
 | 
						|
	return heightValue();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<bool> RpWidgetWrap::shownValue() const {
 | 
						|
	auto &stream = eventStreams().shown;
 | 
						|
	return stream.events_starting_with(!rpWidget()->isHidden());
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<QRect> RpWidgetWrap::paintRequest() const {
 | 
						|
	return eventStreams().paint.events();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<> RpWidgetWrap::alive() const {
 | 
						|
	return eventStreams().alive.events();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<> RpWidgetWrap::windowDeactivateEvents() const {
 | 
						|
	const auto window = rpWidget()->window()->windowHandle();
 | 
						|
	Assert(window != nullptr);
 | 
						|
 | 
						|
	return base::qt_signal_producer(
 | 
						|
		window,
 | 
						|
		&QWindow::activeChanged
 | 
						|
	) | rpl::filter([=] {
 | 
						|
		return !window->isActive();
 | 
						|
	});
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<> RpWidgetWrap::macWindowDeactivateEvents() const {
 | 
						|
#ifdef Q_OS_MAC
 | 
						|
	return windowDeactivateEvents();
 | 
						|
#else // Q_OS_MAC
 | 
						|
	return rpl::never<rpl::empty_value>();
 | 
						|
#endif // Q_OS_MAC
 | 
						|
}
 | 
						|
 | 
						|
rpl::lifetime &RpWidgetWrap::lifetime() {
 | 
						|
	return _lifetime;
 | 
						|
}
 | 
						|
 | 
						|
bool RpWidgetWrap::handleEvent(QEvent *event) {
 | 
						|
	Expects(event != nullptr);
 | 
						|
 | 
						|
	auto streams = _eventStreams.get();
 | 
						|
	if (!streams) {
 | 
						|
		return eventHook(event);
 | 
						|
	}
 | 
						|
	auto that = QPointer<QWidget>();
 | 
						|
	const auto allAreObserved = streams->events.has_consumers();
 | 
						|
	if (allAreObserved) {
 | 
						|
		that = rpWidget();
 | 
						|
		streams->events.fire_copy(event);
 | 
						|
		if (!that) {
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	switch (event->type()) {
 | 
						|
	case QEvent::Move:
 | 
						|
	case QEvent::Resize:
 | 
						|
		if (streams->geometry.has_consumers()) {
 | 
						|
			if (!allAreObserved) {
 | 
						|
				that = rpWidget();
 | 
						|
			}
 | 
						|
			streams->geometry.fire_copy(rpWidget()->geometry());
 | 
						|
			if (!that) {
 | 
						|
				return true;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		break;
 | 
						|
 | 
						|
	case QEvent::Paint:
 | 
						|
		if (streams->paint.has_consumers()) {
 | 
						|
			if (!allAreObserved) {
 | 
						|
				that = rpWidget();
 | 
						|
			}
 | 
						|
			const auto rect = static_cast<QPaintEvent*>(event)->rect();
 | 
						|
			streams->paint.fire_copy(rect);
 | 
						|
			if (!that) {
 | 
						|
				return true;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
	return eventHook(event);
 | 
						|
}
 | 
						|
 | 
						|
RpWidgetWrap::Initer::Initer(QWidget *parent, bool setZeroGeometry) {
 | 
						|
	if (setZeroGeometry) {
 | 
						|
		parent->setGeometry(0, 0, 0, 0);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void RpWidgetWrap::visibilityChangedHook(bool wasVisible, bool nowVisible) {
 | 
						|
	if (nowVisible != wasVisible) {
 | 
						|
		if (auto streams = _eventStreams.get()) {
 | 
						|
			streams->shown.fire_copy(nowVisible);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
auto RpWidgetWrap::eventStreams() const -> EventStreams& {
 | 
						|
	if (!_eventStreams) {
 | 
						|
		_eventStreams = std::make_unique<EventStreams>();
 | 
						|
	}
 | 
						|
	return *_eventStreams;
 | 
						|
}
 | 
						|
 | 
						|
} // namespace Ui
 |