mafulechkaMarch 18, 2020, 6:01 a.m.

Пользовательские оформления окон на стороне клиента в Qt 5.15

Это свежие новости о новой функции в Qt 5.15, которая очень порадовала.

Традиционно оформление окон были довольно скучной вещью. Заголовок, ограничить, минимизировать, максимизировать, изменить размер, выйти... и всё.

ADS

Однако, в последнее время приложения всё чаще и чаще включают Пользовательский интерфейс приложения (application specific UI) и оформление тем. Пара скриншотов для объяснения, о чем здесь говорится:

MacOS занимается этим довольно давно.

... и Chrome, и любой другой веб-браузер.

Встраивание меню в оформление может сэкономить много места на экране.

... или это может быть важно для брендинга или дизайна.
К сожалению, подобные вещи ранее не были возможны с Qt.
Однако можно было убрать оформление с окна, т. е.:

Window {
    flags: Qt.FramelessWindowHint
}

Но это только оставило вас с окном без оформлея. Таким образом, это не могло быть перемещено или изменено. Если бы вы затем попытались реализовать движение окна или изменить размер, схватив мышь и вручную установив размер и положение окна, вы быстро обнаружили бы, что это не очень-то хорошо. Менеджеры окон, как правило, имеют очень специфическое поведение при перемещении или изменении размеров окон. Обычные действия: перетаскивание вверх, чтобы максимизировать, перетаскивание влево/вправо на плитку, привязка к другим окнам или панели задач, изменение размеров двух окон одновременно, если они расположены рядом друг с другом и так далее.

Справедливости ради, ранее был предоставлен один помощник - QSizeGrip. Он позволяет изменить размер любого заданного угла окна, но работает только с углами, а не с границами окна, и доступен только для приложений виджетов.

В Qt 5.15 было добавлено два новых метода в QWindow: startSystemMove и startSystemResize. Эти методы просят оконный менеджер вступить во владение и начать собственную операцию изменения размера или перемещения. Это означает, что привязка, разбиение на тайлы и так далее, должны работать просто волшебным образом, а реализация заголовка в QML становится почти однострочной:

DragHandler {
    onActiveChanged: if (active) window.startSystemMove();
    target: null
}

Поместив этот фрагмент кода в элемент QtQuick, все триггер операции перетаскивания вызовут операцию перемещения собственного окна.
startSystemResize работает аналогично, за исключением того, что он принимает аргумент Qt::Edges, который является битовым полем границ окна, которые вы захватили, то есть для нижнего правого угла, вы бы вызвали:

startSystemResize(Qt.RightEdge | Qt.BottomEdge)

Это также очень удобно, так как вы можете легко иметь один обработчик для всех четырех границ окна и просто создать аргумент границ следующим образом:

DragHandler {
    id: resizeHandler
    grabPermissions: TapHandler.TakeOverForbidden
    target: null
    onActiveChanged: if (active) {
        const p = resizeHandler.centroid.position;
        let e = 0;
        if (p.x < border) e |= Qt.LeftEdge;
        if (p.x >= width - border) e |= Qt.RightEdge;
        if (p.y < border) e |= Qt.TopEdge;
        if (p.y >= height - border) e |= Qt.BottomEdge;
        window.startSystemResize(e);
    }
}

Если вы хотите получить полный пример того, как это можно использовать, был сделан макет веб-браузера с использованием нового API.

Обратите внимание, что, хотя это кроссплатформенный API, не все платформы поддерживают его. startSystemMove в настоящее время поддерживается в Wayland, X11, macOS и Windows, в то время как startSystemResize, с другой стороны, поддерживается в Wayland, X11 и Windows, но не в macOS.

Чтобы справиться с этим, оба метода возвращают логическое значение, которое указывает, была ли операция поддержана или нет. Это означает, что если вы хотите реализовать изменение размера и в macOS, вам придется проверить возвращаемое значение startSystemResize и постараться сделать все возможное, чтобы реализовать запасной вариант в случае сбоя, т. е.:

if (!window.startSystemResize(edges)) {
    // your fallback code for setting window.width/height manually
}

Дальнейшая работа в Qt будет заключаться в том, чтобы предоставить запасные варианты или абстракции, которые бы сделали эту работу за вас, но, по крайней мере, ничто не должно помешать вам сделать это самостоятельно.

Другая область улучшения - согласование с оконным менеджером в том, следует ли использовать оформление на стороне клиента или на стороне сервера. Некоторые приложения могут захотеть поддерживать оба режима и позволить оконному менеджеру решать, но в настоящее время это невозможно. После установки FramelessWindowHint оформления на стороне сервера не будет.

Третья область - тень окна. По крайней мере, на Wayland тень должна быть нарисована, как часть оформления окна. И хотя можно рисовать тени с помощью QtQuick, в настоящее время нет способа сообщить плагину QPA, какая часть поверхности является тенью, а какая - оконной рамкой, что означает, что, если вы попытаетесь нарисовать тени, менеджер окон в настоящее время будет рассмотрите теневую часть окна, и это испортит мозаику и привязку с другими окнами. На других платформах диспетчер окон обычно рисует тени, даже для окон, оформленных на стороне клиента, поэтому эту проблему сложно решить.

We recommend hosting TIMEWEB
We recommend hosting TIMEWEB
Stable hosting, on which the social network EVILEG is located. For projects on Django we recommend VDS hosting.
Support the author Donate

Comments

Only authorized users can post comments.
Please, Log in or Sign up
Timeweb

Let me recommend you the excellent hosting on which EVILEG is located.

For many years, Timeweb has been proving his stability.

For projects on Django I recommend VDS hosting

View Hosting
s

C++ - Тест 003. Условия и циклы

  • Result:42points,
  • Rating points-8
e
  • enfant
  • Oct. 14, 2021, 2:59 p.m.

C++ - Test 002. Constants

  • Result:75points,
  • Rating points2
e
  • enfant
  • Oct. 14, 2021, 2:54 p.m.

C++ - Test 001. The first program and data types

  • Result:93points,
  • Rating points8
Popular publications in the last 90 Days
Last comments
s

Qt/C++ - Lesson 060. Configuring the appearance of the application in runtime

Добрый вечер, на "лету" не работает, только перезапуск
s

Qt/C++ - Lesson 060. Configuring the appearance of the application in runtime

Спасибо, завтра опробую и отпишусь по результату

Django - Tutorial 007. Adding Pagination based on django-bootstrap3

Просто список каких-нибудь объектов передавайте, который дёрнули ищ api стороннего сервера from django.core.paginator import Paginatorobjects = ['john', 'paul', 'george', 'ringo']p = Pagina…
b

PyQt5 - Tutorial 009. Using QThread with MoveToThread

Спасибо большое

PyQt5 - Tutorial 009. Using QThread with MoveToThread

Вызвать либо метод quit() либо эквивалентный его вариант - метод exit(0)
Now discuss on the forum
  • Nomad
  • Oct. 15, 2021, 6:39 a.m.

Вопрос из раздела "как реализовать"

Всем привет. Есть бизнес логика которую надо реализовать на джанге, она состоит в следующем: надо реализовать функционал регистрации/авторизации компаний у которого есть свой дашборд …

Проблема с созданием файлов перевода для составного проекта

Я имею ввиду, если у вас был старые ts файлы, то написать парсер, который составил бы словарь переводов. Например. "Hello world" - "Привет мир" "Hello dev" - "Привет dev" и…
k

QPsql Компиляция драйвера

при компиляции драйвера из окна Qt 5.15.2 (MSVC2019 64 ) надо было запустить C:\Program Files (x86)\Microsoft Visual Studio\2019\Professional\VC\Auxiliary\Build\vcvars64.bat тогда компиляция вып…
P
  • Pisych
  • Oct. 12, 2021, 10:59 p.m.

Фильтр в Select формы Django

Добрый день! Подскажите, как можно сделать? Есть форма, связанная с моделью. В форме Select, выбор типа документа (Приход, Расход,Списание). Этот Select берет данные из таблицы типов документов.…

Вызов функции Python с Qml

Чтобы onResult в QML подключилось к чему-то в Python, нужно чтобы result было сигналом, а у вас это слот. В качестве сигнала определено takeFunc и в данном случае в QML должно ра…
About
Services
© EVILEG 2015-2021
Recommend hosting TIMEWEB