Форум программистов, компьютерный форум, киберфорум
iamvic
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
Путевые заметки в процессе познания Python и PyQt/PySide.
Помни - только тег CODE не портит код добавлением пробела в начало пустой строки.

К вопросу об определении текущего экрана при старте.

Запись от iamvic размещена 22.12.2020 в 18:04
Показов 4188 Комментарии 0
Метки linux, pyqt5, python, python 3, windows

Как я уже писал тут
Как определить во время инициализации на какой экран отрисуется окно?,
вся неправота употребления этого кода в главном окне на старте приложения
Python
1
2
3
4
screen = QDesktopWidget().availableGeometry()
self.setGeometry(
    screen.width()//4, screen.height()//4,
    screen.width()//2, screen.height()//2)
познаётся, когда сам обзаводишься многоэкранной системой.

Ведь запуская задание на дополнительном экране, ожидаешь, что и окно приложения появится на нём же,
а не на основном экране. ИЧСХ, по умолчанию, так оно и произойдёт, если не мудрить с геометрией окна.

Оказывается, чтобы не зарываться в глубины Qt, надо поймать момент завершения отрисовки окна,
как это рекомендуют знающие люди, что и сделано в новом варианте болванки.
Судя по всему, этот способ хорошо подходит под Windows, но вот в Linux всё же не обойтись без костыля.
Там как-то и событие showEvent уже произошло, и self.isVisible() возвращает True, а окно-то всё ещё
не отрисовано, судя по всему.
Но может помочь QTimer с задержкой (100 мсек - перебор, конечно, но чтоб наверняка).

screeprobe.py
Python
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
#!/usr/bin/python3
# -*- coding: utf-8 -*-
 
import sys
from PyQt5.QtCore import PYQT_VERSION_STR, QT_VERSION_STR
from PyQt5.QtCore import pyqtSignal, pyqtSlot
from PyQt5.QtCore import QTimer
from PyQt5.QtWidgets import QAction, QApplication
from PyQt5.QtWidgets import QMainWindow, QMessageBox
 
APP_NAME = 'The QScreen Probe v0.002'
 
def get_current_screen(point):
    """Получить текущий экран
    """
    screens = QApplication.screens()
    if len(screens) > 0:
        for i in range(len(screens)):
            if screens[i].availableGeometry().contains(point):
                return screens[i]
        return None
    else:
        return None
 
 
def print_screen_info(client, name):
    """Печать информации
    """
    print(name, '(primeryScreen)=',
          QApplication.primaryScreen().availableGeometry())
    print(name, '(self.center)=', client.geometry().center())
 
    if (int(QT_VERSION_STR.split('.')[1]) >= 10 and
        int(PYQT_VERSION_STR.split('.')[1]) >= 10):
 
        print(name, '(self.screen)=', client.screen().availableGeometry())
        print(name, '(screenAt)=',
              QApplication.screenAt(
                  client.geometry().center()).availableGeometry())
    print('')
 
 
 
class ProbeMainWindow(QMainWindow):
    """Главное окно приложения
    """
    window_showed = pyqtSignal()
    is_first_show = True
 
    def __init__(self, parent=None):
        super(ProbeMainWindow, self).__init__(parent)
        self.initUI()
 
    def initUI(self):
        """Строим графический интерфейс
        """
        # это неправильно для многоэкранных
        # screen = QDesktopWidget().availableGeometry()
        # self.setGeometry(
        #     screen.width()//4, screen.height()//4,
        #     screen.width()//2, screen.height()//2)
 
        screens = QApplication.screens()
        for i in range(len(screens)):
            print('[{!s}] --> {!s}'.format(i, screens[i].availableGeometry()))
        print('')
        print_screen_info(self, 'init')
 
        self.setWindowTitle(APP_NAME)
        self.statusBar()
 
        exitAction = QAction(
            'Выход', self, shortcut='Ctrl+Q',
            statusTip='Завершить работу приложения',
            triggered=self.to_close, enabled=True)
 
        taskMenu = self.menuBar().addMenu('Файл')
        taskMenu.addAction(exitAction)
 
        self.window_showed.connect(self.to_custom)
 
    def to_custom(self):
        """Обработка после инициализации GUI
        """
        print_screen_info(self, 'before_custom')
 
        if (int(QT_VERSION_STR.split('.')[1]) >= 10 and
            int(PYQT_VERSION_STR.split('.')[1]) >= 10):
 
            current = QApplication.screenAt(self.geometry().center())
        else:
            current = get_current_screen(self.geometry().center())
 
        if current is None:
            # нет устройств отображения или оно кривое
            # (было c какой-то из быстрых настроек экрана в Lubunta 18.04)
            print('Current screen not determitated or invlalid.')
            self.to_close()
        else:
            current_geo = current.availableGeometry()
            self.setGeometry(
                current_geo.x() + current_geo.width() // 4,
                current_geo.y() + current_geo.height() // 4,
                current_geo.width() // 2,
                current_geo.height() // 2)
 
            print_screen_info(self, 'after_custom')
 
    def to_close(self):
        """Завершаем работу
        """
        print_screen_info(self, 'close')
        self.close()
 
    def showEvent(self, event):
        """Обработка событий отрисовки окна
        """
        if self.isVisible() and self.is_first_show:
            print('self.isVisible() =', self.isVisible())
            print_screen_info(self, 'in_showEvent')
            self.is_first_show = False
            if sys.platform == 'win32':
                self.window_showed.emit()
            else:
                QTimer.singleShot(100, self.to_custom)
 
        super(ProbeMainWindow, self).showEvent(event)
 
 
if __name__ == '__main__':
    app = QApplication(sys.argv)
    mwin = ProbeMainWindow()
    mwin.show()
    sys.exit(app.exec_())
Протокол запуска приложения на дополнительном экране (Windows)
Code
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
C:\Tools>py64 screenprobe.py
[0] --> PyQt5.QtCore.QRect(0, 0, 1024, 728)
[1] --> PyQt5.QtCore.QRect(1024, -10, 1280, 1024)
 
init (primeryScreen)= PyQt5.QtCore.QRect(0, 0, 1024, 728)
init (self.center)= PyQt5.QtCore.QPoint(319, 239)
init (self.screen)= PyQt5.QtCore.QRect(0, 0, 1024, 728)
init (screenAt)= PyQt5.QtCore.QRect(0, 0, 1024, 728)
 
self.isVisible() = True
in_showEvent (primeryScreen)= PyQt5.QtCore.QRect(0, 0, 1024, 728)
in_showEvent (self.center)= PyQt5.QtCore.QPoint(1443, 311)   <--- !!!
in_showEvent (self.screen)= PyQt5.QtCore.QRect(1024, -10, 1280, 1024)
in_showEvent (screenAt)= PyQt5.QtCore.QRect(1024, -10, 1280, 1024)
 
before_custom (primeryScreen)= PyQt5.QtCore.QRect(0, 0, 1024, 728)
before_custom (self.center)= PyQt5.QtCore.QPoint(1443, 311)   <--- !!!
before_custom (self.screen)= PyQt5.QtCore.QRect(1024, -10, 1280, 1024)
before_custom (screenAt)= PyQt5.QtCore.QRect(1024, -10, 1280, 1024)
 
after_custom (primeryScreen)= PyQt5.QtCore.QRect(0, 0, 1024, 728)
after_custom (self.center)= PyQt5.QtCore.QPoint(1663, 501)
after_custom (self.screen)= PyQt5.QtCore.QRect(1024, -10, 1280, 1024)
after_custom (screenAt)= PyQt5.QtCore.QRect(1024, -10, 1280, 1024)
 
close (primeryScreen)= PyQt5.QtCore.QRect(0, 0, 1024, 728)
close (self.center)= PyQt5.QtCore.QPoint(1663, 501)
close (self.screen)= PyQt5.QtCore.QRect(1024, -10, 1280, 1024)
close (screenAt)= PyQt5.QtCore.QRect(1024, -10, 1280, 1024)
Протокол запуска приложения на дополнительном экране (Linux).
Code
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
user@lubunta18-home:~/Tools$ ./screenprobe.py
QApplication: invalid style override passed, ignoring it.
[0] --> PyQt5.QtCore.QRect(0, 0, 1024, 576)
[1] --> PyQt5.QtCore.QRect(1024, 0, 1280, 576)
 
init (primeryScreen)= PyQt5.QtCore.QRect(0, 0, 1024, 576)
init (self.center)= PyQt5.QtCore.QPoint(319, 239)
 
self.isVisible() = True
in_showEvent (primeryScreen)= PyQt5.QtCore.QRect(0, 0, 1024, 576)
in_showEvent (self.center)= PyQt5.QtCore.QPoint(99, 49)   <--- WTF ???
 
before_custom (primeryScreen)= PyQt5.QtCore.QRect(0, 0, 1024, 576)
before_custom (self.center)= PyQt5.QtCore.QPoint(1663, 82)   <--- !!!
 
after_custom (primeryScreen)= PyQt5.QtCore.QRect(0, 0, 1024, 576)
after_custom (self.center)= PyQt5.QtCore.QPoint(1663, 287)
 
close (primeryScreen)= PyQt5.QtCore.QRect(0, 0, 1024, 576)
close (self.center)= PyQt5.QtCore.QPoint(1867, 234)
Метки linux, pyqt5, python, python 3, windows
Размещено в Памятка
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Всего комментариев 0
Комментарии
 
Новые блоги и статьи
[golang] Двоичная куча, min-heap
alhaos 20.05.2026
Двоичная куча Двоичная куча — структура данных, которая всегда держит самый важный элемент наготове. Представьте очередь к хилеру в игре, и очередь из игроков в приоритете те у кого меньше. . .
[golang] Breadth-First Search
alhaos 19.05.2026
BFS (Breadth-First Search) — это базовый алгоритм обхода графа в ширину, который поуровнево исследует все связанные вершины. Он начинает с выбранной точки и проверяет всех соседей, прежде чем. . .
[golang] Алгоритм «Хак Госпера»
alhaos 17.05.2026
Алгоритм «Хак Госпера» Хак Госпера (Gosper's Hack) — алгоритм нахождения следующего по величине числа с тем же количеством установленных бит. Придуман Биллом Госпером в 1970-х, опубликован в. . .
Рисование бинарного древа до 6-го колена на js, svg.
russiannick 17.05.2026
<svg width="335" height="240" viewBox="0 0 335 240" fill="#e5e1bb"> <style> <!]> </ style> <g id="bush"> </ g> </ svg> function fn(){ let rost;/ / высота древа let xx=165,yy=210,w=256;
FSharp: interface of module
DevAlt 16.05.2026
Интерфейс модуля F# позволяет управлять доступностью членов, содержащихся в реализации модуля. По-умолчанию все члены модуля доступны: module Foo let x = 10 let boo () = printfn "boo" . . .
Хитросплетение родственных связей пантеона греческих богов.
russiannick 14.05.2026
Однооконник, позволяющий узреть и изучить отдельных героев древней Греции. <!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible". . .
[golang] Угол между стрелками часов
alhaos 12.05.2026
По заданным значениям часа и минуты необходимо определить значение меньшего угла между стрелками аналогового циферблата часов. import "math" func angleClock(hour int, minutes int) float64 { . . .
Debian 13: Установка Lazarus QT5
ВитГо 09.05.2026
Эта инструкция моя компиляция инструкций volvo https:/ / www. cyberforum. ru/ blogs/ 203668/ 10753. html и его же старой инструкции по установке Lazarus с gtk2. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru