init
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
Close stale issues and PRs / stale (push) Has been cancelled
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
Close stale issues and PRs / stale (push) Has been cancelled
This commit is contained in:
616
Telegram/lib_ui/ui/platform/linux/ui_utility_linux.cpp
Normal file
616
Telegram/lib_ui/ui/platform/linux/ui_utility_linux.cpp
Normal file
@@ -0,0 +1,616 @@
|
||||
// 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/linux/ui_utility_linux.h"
|
||||
|
||||
#include "base/platform/base_platform_info.h"
|
||||
#include "base/platform/linux/base_linux_library.h"
|
||||
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
#include "base/platform/linux/base_linux_xcb_utilities.h"
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
|
||||
#include <QtCore/QPoint>
|
||||
#include <QtGui/QWindow>
|
||||
#include <QtWidgets/QApplication>
|
||||
#include <qpa/qplatformwindow.h>
|
||||
#include <qpa/qplatformwindow_p.h>
|
||||
|
||||
extern "C" {
|
||||
|
||||
typedef int32_t wl_fixed_t;
|
||||
struct wl_object;
|
||||
struct wl_array;
|
||||
struct wl_proxy;
|
||||
struct xdg_toplevel;
|
||||
|
||||
union wl_argument {
|
||||
int32_t i; /**< `int` */
|
||||
uint32_t u; /**< `uint` */
|
||||
wl_fixed_t f; /**< `fixed` */
|
||||
const char *s; /**< `string` */
|
||||
struct wl_object *o; /**< `object` */
|
||||
uint32_t n; /**< `new_id` */
|
||||
struct wl_array *a; /**< `array` */
|
||||
int32_t h; /**< `fd` */
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
namespace Ui {
|
||||
namespace Platform {
|
||||
namespace {
|
||||
|
||||
static const auto kXCBFrameExtentsAtomName = u"_GTK_FRAME_EXTENTS"_q;
|
||||
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
std::optional<bool> XCBWindowMapped(xcb_window_t window) {
|
||||
const base::Platform::XCB::Connection connection;
|
||||
if (!connection || xcb_connection_has_error(connection)) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto cookie = xcb_get_window_attributes(connection, window);
|
||||
const auto reply = base::Platform::XCB::MakeReplyPointer(
|
||||
xcb_get_window_attributes_reply(
|
||||
connection,
|
||||
cookie,
|
||||
nullptr));
|
||||
|
||||
if (!reply) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
return reply->map_state == XCB_MAP_STATE_VIEWABLE;
|
||||
}
|
||||
|
||||
std::optional<bool> XCBWindowHidden(xcb_window_t window) {
|
||||
const base::Platform::XCB::Connection connection;
|
||||
if (!connection || xcb_connection_has_error(connection)) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto stateAtom = base::Platform::XCB::GetAtom(
|
||||
connection,
|
||||
"_NET_WM_STATE");
|
||||
|
||||
const auto stateHiddenAtom = base::Platform::XCB::GetAtom(
|
||||
connection,
|
||||
"_NET_WM_STATE_HIDDEN");
|
||||
|
||||
if (!stateAtom || !stateHiddenAtom) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto cookie = xcb_get_property(
|
||||
connection,
|
||||
false,
|
||||
window,
|
||||
stateAtom,
|
||||
XCB_ATOM_ATOM,
|
||||
0,
|
||||
1024);
|
||||
|
||||
const auto reply = base::Platform::XCB::MakeReplyPointer(
|
||||
xcb_get_property_reply(
|
||||
connection,
|
||||
cookie,
|
||||
nullptr));
|
||||
|
||||
if (!reply) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
if (reply->type != XCB_ATOM_ATOM || reply->format != 32) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto atomsStart = reinterpret_cast<xcb_atom_t*>(
|
||||
xcb_get_property_value(reply.get()));
|
||||
|
||||
const auto states = std::vector<xcb_atom_t>(
|
||||
atomsStart,
|
||||
atomsStart + reply->length);
|
||||
|
||||
return ranges::contains(states, stateHiddenAtom);
|
||||
}
|
||||
|
||||
QRect XCBWindowGeometry(xcb_window_t window) {
|
||||
const base::Platform::XCB::Connection connection;
|
||||
if (!connection || xcb_connection_has_error(connection)) {
|
||||
return {};
|
||||
}
|
||||
|
||||
const auto cookie = xcb_get_geometry(connection, window);
|
||||
const auto reply = base::Platform::XCB::MakeReplyPointer(
|
||||
xcb_get_geometry_reply(
|
||||
connection,
|
||||
cookie,
|
||||
nullptr));
|
||||
|
||||
if (!reply) {
|
||||
return {};
|
||||
}
|
||||
|
||||
return QRect(reply->x, reply->y, reply->width, reply->height);
|
||||
}
|
||||
|
||||
std::optional<uint> XCBCurrentWorkspace() {
|
||||
const base::Platform::XCB::Connection connection;
|
||||
if (!connection || xcb_connection_has_error(connection)) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto root = base::Platform::XCB::GetRootWindow(connection);
|
||||
if (!root) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto currentDesktopAtom = base::Platform::XCB::GetAtom(
|
||||
connection,
|
||||
"_NET_CURRENT_DESKTOP");
|
||||
|
||||
if (!currentDesktopAtom) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto cookie = xcb_get_property(
|
||||
connection,
|
||||
false,
|
||||
root,
|
||||
currentDesktopAtom,
|
||||
XCB_ATOM_CARDINAL,
|
||||
0,
|
||||
1024);
|
||||
|
||||
const auto reply = base::Platform::XCB::MakeReplyPointer(
|
||||
xcb_get_property_reply(
|
||||
connection,
|
||||
cookie,
|
||||
nullptr));
|
||||
|
||||
if (!reply) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
return (reply->type == XCB_ATOM_CARDINAL)
|
||||
? std::make_optional(
|
||||
*reinterpret_cast<ulong*>(xcb_get_property_value(reply.get())))
|
||||
: std::nullopt;
|
||||
}
|
||||
|
||||
std::optional<uint> XCBWindowWorkspace(xcb_window_t window) {
|
||||
const base::Platform::XCB::Connection connection;
|
||||
if (!connection || xcb_connection_has_error(connection)) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto desktopAtom = base::Platform::XCB::GetAtom(
|
||||
connection,
|
||||
"_NET_WM_DESKTOP");
|
||||
|
||||
if (!desktopAtom) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto cookie = xcb_get_property(
|
||||
connection,
|
||||
false,
|
||||
window,
|
||||
desktopAtom,
|
||||
XCB_ATOM_CARDINAL,
|
||||
0,
|
||||
1024);
|
||||
|
||||
const auto reply = base::Platform::XCB::MakeReplyPointer(
|
||||
xcb_get_property_reply(
|
||||
connection,
|
||||
cookie,
|
||||
nullptr));
|
||||
|
||||
if (!reply) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
return (reply->type == XCB_ATOM_CARDINAL)
|
||||
? std::make_optional(
|
||||
*reinterpret_cast<ulong*>(xcb_get_property_value(reply.get())))
|
||||
: std::nullopt;
|
||||
}
|
||||
|
||||
std::optional<bool> XCBIsOverlapped(
|
||||
not_null<QWidget*> widget,
|
||||
const QRect &rect) {
|
||||
const auto window = widget->winId();
|
||||
Expects(window != XCB_NONE);
|
||||
|
||||
const base::Platform::XCB::Connection connection;
|
||||
if (!connection || xcb_connection_has_error(connection)) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto root = base::Platform::XCB::GetRootWindow(connection);
|
||||
if (!root) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto windowWorkspace = XCBWindowWorkspace(window);
|
||||
const auto currentWorkspace = XCBCurrentWorkspace();
|
||||
if (windowWorkspace.has_value()
|
||||
&& currentWorkspace.has_value()
|
||||
&& *windowWorkspace != *currentWorkspace
|
||||
&& *windowWorkspace != 0xFFFFFFFF) {
|
||||
return true;
|
||||
}
|
||||
|
||||
const auto windowGeometry = XCBWindowGeometry(window);
|
||||
if (windowGeometry.isNull()) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto mappedRect = QRect(
|
||||
rect.topLeft()
|
||||
* widget->windowHandle()->devicePixelRatio()
|
||||
+ windowGeometry.topLeft(),
|
||||
rect.size() * widget->windowHandle()->devicePixelRatio());
|
||||
|
||||
const auto cookie = xcb_query_tree(connection, root);
|
||||
const auto reply = base::Platform::XCB::MakeReplyPointer(
|
||||
xcb_query_tree_reply(connection, cookie, nullptr));
|
||||
|
||||
if (!reply) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto tree = xcb_query_tree_children(reply.get());
|
||||
auto aboveTheWindow = false;
|
||||
|
||||
for (auto i = 0, l = xcb_query_tree_children_length(reply.get()); i < l; ++i) {
|
||||
if (window == tree[i]) {
|
||||
aboveTheWindow = true;
|
||||
continue;
|
||||
}
|
||||
|
||||
if (!aboveTheWindow) {
|
||||
continue;
|
||||
}
|
||||
|
||||
const auto geometry = XCBWindowGeometry(tree[i]);
|
||||
if (!mappedRect.intersects(geometry)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
const auto workspace = XCBWindowWorkspace(tree[i]);
|
||||
if (workspace.has_value()
|
||||
&& windowWorkspace.has_value()
|
||||
&& *workspace != *windowWorkspace
|
||||
&& *workspace != 0xFFFFFFFF) {
|
||||
continue;
|
||||
}
|
||||
|
||||
const auto mapped = XCBWindowMapped(tree[i]);
|
||||
if (mapped.has_value() && !*mapped) {
|
||||
continue;
|
||||
}
|
||||
|
||||
const auto hidden = XCBWindowHidden(tree[i]);
|
||||
if (hidden.has_value() && *hidden) {
|
||||
continue;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
void SetXCBFrameExtents(not_null<QWidget*> widget, const QMargins &extents) {
|
||||
const base::Platform::XCB::Connection connection;
|
||||
if (!connection || xcb_connection_has_error(connection)) {
|
||||
return;
|
||||
}
|
||||
|
||||
const auto frameExtentsAtom = base::Platform::XCB::GetAtom(
|
||||
connection,
|
||||
kXCBFrameExtentsAtomName);
|
||||
|
||||
if (!frameExtentsAtom) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (extents.isNull()) {
|
||||
free(
|
||||
xcb_request_check(
|
||||
connection,
|
||||
xcb_delete_property_checked(
|
||||
connection,
|
||||
widget->winId(),
|
||||
frameExtentsAtom)));
|
||||
return;
|
||||
}
|
||||
|
||||
const auto nativeExtents = extents
|
||||
* widget->windowHandle()->devicePixelRatio();
|
||||
|
||||
const auto extentsVector = std::vector<uint>{
|
||||
uint(nativeExtents.left()),
|
||||
uint(nativeExtents.right()),
|
||||
uint(nativeExtents.top()),
|
||||
uint(nativeExtents.bottom()),
|
||||
};
|
||||
|
||||
free(
|
||||
xcb_request_check(
|
||||
connection,
|
||||
xcb_change_property_checked(
|
||||
connection,
|
||||
XCB_PROP_MODE_REPLACE,
|
||||
widget->winId(),
|
||||
frameExtentsAtom,
|
||||
XCB_ATOM_CARDINAL,
|
||||
32,
|
||||
extentsVector.size(),
|
||||
extentsVector.data())));
|
||||
}
|
||||
|
||||
void ShowXCBWindowMenu(not_null<QWidget*> widget, const QPoint &point) {
|
||||
const base::Platform::XCB::Connection connection;
|
||||
if (!connection || xcb_connection_has_error(connection)) {
|
||||
return;
|
||||
}
|
||||
|
||||
const auto root = base::Platform::XCB::GetRootWindow(connection);
|
||||
if (!root) {
|
||||
return;
|
||||
}
|
||||
|
||||
const auto showWindowMenuAtom = base::Platform::XCB::GetAtom(
|
||||
connection,
|
||||
"_GTK_SHOW_WINDOW_MENU");
|
||||
|
||||
if (!showWindowMenuAtom) {
|
||||
return;
|
||||
}
|
||||
|
||||
const auto windowGeometry = XCBWindowGeometry(widget->winId());
|
||||
if (windowGeometry.isNull()) {
|
||||
return;
|
||||
}
|
||||
|
||||
const auto globalPos = point
|
||||
* widget->windowHandle()->devicePixelRatio()
|
||||
+ windowGeometry.topLeft();
|
||||
|
||||
xcb_client_message_event_t xev;
|
||||
xev.response_type = XCB_CLIENT_MESSAGE;
|
||||
xev.type = showWindowMenuAtom;
|
||||
xev.sequence = 0;
|
||||
xev.window = widget->winId();
|
||||
xev.format = 32;
|
||||
xev.data.data32[0] = 0;
|
||||
xev.data.data32[1] = globalPos.x();
|
||||
xev.data.data32[2] = globalPos.y();
|
||||
xev.data.data32[3] = 0;
|
||||
xev.data.data32[4] = 0;
|
||||
|
||||
free(
|
||||
xcb_request_check(
|
||||
connection,
|
||||
xcb_ungrab_pointer_checked(connection, XCB_CURRENT_TIME)));
|
||||
|
||||
free(
|
||||
xcb_request_check(
|
||||
connection,
|
||||
xcb_send_event_checked(
|
||||
connection,
|
||||
false,
|
||||
root,
|
||||
XCB_EVENT_MASK_SUBSTRUCTURE_REDIRECT
|
||||
| XCB_EVENT_MASK_SUBSTRUCTURE_NOTIFY,
|
||||
reinterpret_cast<const char*>(&xev))));
|
||||
}
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
|
||||
#if defined QT_FEATURE_wayland && QT_CONFIG(wayland)
|
||||
void ShowWaylandWindowMenu(not_null<QWidget*> widget, const QPoint &point) {
|
||||
static const auto wl_proxy_marshal_array = [] {
|
||||
void (*result)(
|
||||
struct wl_proxy *p,
|
||||
uint32_t opcode,
|
||||
union wl_argument *args) = nullptr;
|
||||
|
||||
if (const auto lib = base::Platform::LoadLibrary(
|
||||
"libwayland-client.so.0",
|
||||
RTLD_NODELETE)) {
|
||||
base::Platform::LoadSymbol(lib, "wl_proxy_marshal_array", result);
|
||||
}
|
||||
|
||||
return result;
|
||||
}();
|
||||
|
||||
if (!wl_proxy_marshal_array) {
|
||||
return;
|
||||
}
|
||||
|
||||
using namespace QNativeInterface;
|
||||
using namespace QNativeInterface::Private;
|
||||
const auto window = not_null(widget->windowHandle());
|
||||
const auto native = qApp->nativeInterface<QWaylandApplication>();
|
||||
const auto nativeWindow = window->nativeInterface<QWaylandWindow>();
|
||||
if (!native || !nativeWindow) {
|
||||
return;
|
||||
}
|
||||
|
||||
const auto toplevel = nativeWindow->surfaceRole<xdg_toplevel>();
|
||||
const auto seat = native->lastInputSeat();
|
||||
if (!toplevel || !seat) {
|
||||
return;
|
||||
}
|
||||
|
||||
const auto pos = point
|
||||
* window->devicePixelRatio()
|
||||
/ window->handle()->devicePixelRatio();
|
||||
|
||||
wl_proxy_marshal_array(
|
||||
reinterpret_cast<wl_proxy*>(toplevel),
|
||||
4, // XDG_TOPLEVEL_SHOW_WINDOW_MENU
|
||||
std::array{
|
||||
wl_argument{ .o = reinterpret_cast<wl_object*>(seat) },
|
||||
wl_argument{ .u = native->lastInputSerial() },
|
||||
wl_argument{ .i = pos.x() },
|
||||
wl_argument{ .i = pos.y() },
|
||||
}.data());
|
||||
}
|
||||
#endif // wayland
|
||||
|
||||
} // namespace
|
||||
|
||||
bool IsApplicationActive() {
|
||||
return QApplication::activeWindow() != nullptr;
|
||||
}
|
||||
|
||||
bool TranslucentWindowsSupported() {
|
||||
if (::Platform::IsWayland()) {
|
||||
return true;
|
||||
}
|
||||
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
if (::Platform::IsX11()) {
|
||||
const base::Platform::XCB::Connection connection;
|
||||
if (!connection || xcb_connection_has_error(connection)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
const auto atom = base::Platform::XCB::GetAtom(
|
||||
connection,
|
||||
"_NET_WM_CM_S0");
|
||||
|
||||
if (!atom) {
|
||||
return false;
|
||||
}
|
||||
|
||||
const auto cookie = xcb_get_selection_owner(connection, atom);
|
||||
|
||||
const auto result = base::Platform::XCB::MakeReplyPointer(
|
||||
xcb_get_selection_owner_reply(
|
||||
connection,
|
||||
cookie,
|
||||
nullptr));
|
||||
|
||||
if (!result) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return result->owner;
|
||||
}
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
void IgnoreAllActivation(not_null<QWidget*> widget) {
|
||||
}
|
||||
|
||||
void ClearTransientParent(not_null<QWidget*> widget) {
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
if (::Platform::IsX11()) {
|
||||
const base::Platform::XCB::Connection connection;
|
||||
if (!connection || xcb_connection_has_error(connection)) {
|
||||
return;
|
||||
}
|
||||
|
||||
free(
|
||||
xcb_request_check(
|
||||
connection,
|
||||
xcb_delete_property_checked(
|
||||
connection,
|
||||
widget->winId(),
|
||||
XCB_ATOM_WM_TRANSIENT_FOR)));
|
||||
|
||||
return;
|
||||
}
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
}
|
||||
|
||||
std::optional<bool> IsOverlapped(
|
||||
not_null<QWidget*> widget,
|
||||
const QRect &rect) {
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
if (::Platform::IsX11()) {
|
||||
return XCBIsOverlapped(widget, rect);
|
||||
}
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
bool WindowMarginsSupported() {
|
||||
#if defined QT_FEATURE_wayland && QT_CONFIG(wayland)
|
||||
static const auto WaylandResult = [] {
|
||||
using namespace QNativeInterface::Private;
|
||||
QWindow window;
|
||||
window.create();
|
||||
return bool(window.nativeInterface<QWaylandWindow>());
|
||||
}();
|
||||
|
||||
if (WaylandResult) {
|
||||
return true;
|
||||
}
|
||||
#endif // wayland
|
||||
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
namespace XCB = base::Platform::XCB;
|
||||
if (::Platform::IsX11()
|
||||
&& XCB::IsSupportedByWM(
|
||||
XCB::Connection(),
|
||||
kXCBFrameExtentsAtomName)) {
|
||||
return true;
|
||||
}
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
void SetWindowMargins(not_null<QWidget*> widget, const QMargins &margins) {
|
||||
#if defined QT_FEATURE_wayland && QT_CONFIG(wayland)
|
||||
using namespace QNativeInterface::Private;
|
||||
const auto window = not_null(widget->windowHandle());
|
||||
const auto platformWindow = not_null(window->handle());
|
||||
if (const auto native = window->nativeInterface<QWaylandWindow>()) {
|
||||
native->setCustomMargins(
|
||||
margins
|
||||
* window->devicePixelRatio()
|
||||
/ platformWindow->devicePixelRatio());
|
||||
return;
|
||||
}
|
||||
#endif // wayland
|
||||
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
if (::Platform::IsX11()) {
|
||||
SetXCBFrameExtents(widget, margins);
|
||||
return;
|
||||
}
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
}
|
||||
|
||||
void ShowWindowMenu(not_null<QWidget*> widget, const QPoint &point) {
|
||||
#if defined QT_FEATURE_wayland && QT_CONFIG(wayland)
|
||||
if (::Platform::IsWayland()) {
|
||||
ShowWaylandWindowMenu(widget, point);
|
||||
return;
|
||||
}
|
||||
#endif // wayland
|
||||
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
if (::Platform::IsX11()) {
|
||||
ShowXCBWindowMenu(widget, point);
|
||||
return;
|
||||
}
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
}
|
||||
|
||||
} // namespace Platform
|
||||
} // namespace Ui
|
||||
43
Telegram/lib_ui/ui/platform/linux/ui_utility_linux.h
Normal file
43
Telegram/lib_ui/ui/platform/linux/ui_utility_linux.h
Normal file
@@ -0,0 +1,43 @@
|
||||
// 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
|
||||
//
|
||||
#pragma once
|
||||
|
||||
#include "ui/platform/ui_platform_utility.h"
|
||||
|
||||
class QPainter;
|
||||
class QPaintEvent;
|
||||
|
||||
namespace Ui {
|
||||
namespace Platform {
|
||||
|
||||
inline void InitOnTopPanel(not_null<QWidget*> panel) {
|
||||
}
|
||||
|
||||
inline void DeInitOnTopPanel(not_null<QWidget*> panel) {
|
||||
}
|
||||
|
||||
inline void ReInitOnTopPanel(not_null<QWidget*> panel) {
|
||||
}
|
||||
|
||||
inline void ShowOverAll(not_null<QWidget*> widget, bool canFocus) {
|
||||
}
|
||||
|
||||
inline void AcceptAllMouseInput(not_null<QWidget*> widget) {
|
||||
}
|
||||
|
||||
inline void DisableSystemWindowResize(not_null<QWidget*> widget, QSize ratio) {
|
||||
}
|
||||
|
||||
inline constexpr bool UseMainQueueGeneric() {
|
||||
return true;
|
||||
}
|
||||
|
||||
inline void FixPopupMenuNativeEmojiPopup(not_null<PopupMenu*> menu) {
|
||||
}
|
||||
|
||||
} // namespace Platform
|
||||
} // namespace Ui
|
||||
25
Telegram/lib_ui/ui/platform/linux/ui_window_linux.cpp
Normal file
25
Telegram/lib_ui/ui/platform/linux/ui_window_linux.cpp
Normal file
@@ -0,0 +1,25 @@
|
||||
// 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/linux/ui_window_linux.h"
|
||||
|
||||
namespace Ui::Platform {
|
||||
|
||||
std::unique_ptr<BasicWindowHelper> CreateSpecialWindowHelper(
|
||||
not_null<RpWidget*> window) {
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
bool NativeWindowFrameSupported() {
|
||||
return true;
|
||||
}
|
||||
|
||||
rpl::producer<FullScreenEvent> FullScreenEvents(
|
||||
not_null<RpWidget*> window) {
|
||||
return rpl::never<FullScreenEvent>();
|
||||
}
|
||||
|
||||
} // namespace Ui::Platform
|
||||
9
Telegram/lib_ui/ui/platform/linux/ui_window_linux.h
Normal file
9
Telegram/lib_ui/ui/platform/linux/ui_window_linux.h
Normal file
@@ -0,0 +1,9 @@
|
||||
// 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
|
||||
//
|
||||
#pragma once
|
||||
|
||||
#include "ui/platform/ui_platform_window.h"
|
||||
143
Telegram/lib_ui/ui/platform/linux/ui_window_title_linux.cpp
Normal file
143
Telegram/lib_ui/ui/platform/linux/ui_window_title_linux.cpp
Normal file
@@ -0,0 +1,143 @@
|
||||
// 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/linux/ui_window_title_linux.h"
|
||||
|
||||
#include "base/platform/linux/base_linux_xdp_utilities.h"
|
||||
|
||||
#include "base/integration.h"
|
||||
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
#include "base/platform/linux/base_linux_xsettings.h"
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
|
||||
namespace Ui {
|
||||
namespace Platform {
|
||||
namespace {
|
||||
|
||||
class TitleControlsLayoutImpl : public TitleControlsLayout {
|
||||
public:
|
||||
TitleControlsLayoutImpl();
|
||||
|
||||
private:
|
||||
[[nodiscard]] static TitleControls::Layout Get();
|
||||
|
||||
const rpl::lifetime _lifetime;
|
||||
const base::Platform::XDP::SettingWatcher _settingWatcher;
|
||||
};
|
||||
|
||||
TitleControlsLayoutImpl::TitleControlsLayoutImpl()
|
||||
: TitleControlsLayout(Get())
|
||||
, _lifetime([&] {
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
using base::Platform::XCB::XSettings;
|
||||
if (const auto xSettings = XSettings::Instance()) {
|
||||
return xSettings->registerCallbackForProperty(
|
||||
"Gtk/DecorationLayout",
|
||||
[=](xcb_connection_t *, const QByteArray &, const QVariant &) {
|
||||
base::Integration::Instance().enterFromEventLoop([&] {
|
||||
_variable = Get();
|
||||
});
|
||||
});
|
||||
}
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
return rpl::lifetime();
|
||||
}())
|
||||
, _settingWatcher("org.gnome.desktop.wm.preferences", "button-layout", [=] {
|
||||
base::Integration::Instance().enterFromEventLoop([&] {
|
||||
_variable = Get();
|
||||
});
|
||||
}) {}
|
||||
|
||||
TitleControls::Layout TitleControlsLayoutImpl::Get() {
|
||||
const auto convert = [](const QString &keywords) {
|
||||
const auto toControl = [](const QString &keyword) {
|
||||
if (keyword == qstr("minimize")) {
|
||||
return TitleControls::Control::Minimize;
|
||||
} else if (keyword == qstr("maximize")) {
|
||||
return TitleControls::Control::Maximize;
|
||||
} else if (keyword == qstr("close")) {
|
||||
return TitleControls::Control::Close;
|
||||
}
|
||||
return TitleControls::Control::Unknown;
|
||||
};
|
||||
|
||||
TitleControls::Layout result;
|
||||
const auto splitted = keywords.split(':');
|
||||
|
||||
ranges::transform(
|
||||
splitted[0].split(','),
|
||||
ranges::back_inserter(result.left),
|
||||
toControl);
|
||||
|
||||
if (splitted.size() > 1) {
|
||||
ranges::transform(
|
||||
splitted[1].split(','),
|
||||
ranges::back_inserter(result.right),
|
||||
toControl);
|
||||
}
|
||||
|
||||
return result;
|
||||
};
|
||||
|
||||
#ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
const auto xSettingsResult = [&]()
|
||||
-> std::optional<TitleControls::Layout> {
|
||||
using base::Platform::XCB::XSettings;
|
||||
const auto xSettings = XSettings::Instance();
|
||||
if (!xSettings) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
const auto decorationLayout = xSettings->setting(
|
||||
"Gtk/DecorationLayout");
|
||||
|
||||
if (!decorationLayout.isValid()) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
return convert(decorationLayout.toString());
|
||||
}();
|
||||
|
||||
if (xSettingsResult.has_value()) {
|
||||
return *xSettingsResult;
|
||||
}
|
||||
#endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
|
||||
|
||||
const auto portalResult = [&]() -> std::optional<TitleControls::Layout> {
|
||||
auto decorationLayout = base::Platform::XDP::ReadSetting(
|
||||
"org.gnome.desktop.wm.preferences",
|
||||
"button-layout");
|
||||
|
||||
if (!decorationLayout.has_value()) {
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
return convert(
|
||||
QString::fromStdString(decorationLayout->get_string(nullptr)));
|
||||
}();
|
||||
|
||||
if (portalResult.has_value()) {
|
||||
return *portalResult;
|
||||
}
|
||||
|
||||
return TitleControls::Layout{
|
||||
.right = {
|
||||
TitleControls::Control::Minimize,
|
||||
TitleControls::Control::Maximize,
|
||||
TitleControls::Control::Close,
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
std::shared_ptr<TitleControlsLayout> TitleControlsLayout::Create() {
|
||||
return std::make_shared<TitleControlsLayoutImpl>();
|
||||
}
|
||||
|
||||
} // namespace Platform
|
||||
} // namespace Ui
|
||||
@@ -0,0 +1,9 @@
|
||||
// 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
|
||||
//
|
||||
#pragma once
|
||||
|
||||
#include "ui/platform/ui_platform_window_title.h"
|
||||
Reference in New Issue
Block a user