Форум программистов, компьютерный форум, киберфорум
iamvic
Войти
Регистрация
Восстановить пароль
Путевые заметки в процессе познания Python и PyQt/PySide.
Рейтинг: 1.00. Голосов: 5.

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

Запись от iamvic размещена 22.12.2020 в 18:04
Метки 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)
Код:
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).
Код:
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)
Размещено в Памятка
Показов 2949 Комментарии 0
Всего комментариев 0
Комментарии
 
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2021, vBulletin Solutions, Inc.