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

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

Запись от iamvic размещена 22.12.2020 в 18:04
Показов 4157 Комментарии 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
Комментарии
 
Новые блоги и статьи
Нейросеть на алгоритме "эстафета хвоста" как перспектива.
Hrethgir 06.05.2026
На десерт, когда запущу сервер. Статья тут https:/ / habr. com/ ru/ articles/ 1030914/ . Автор я сам, нейросеть только помогает в вопросах которые мне не известны - не знаю людей которые знали-бы. . .
Асинхронный приём данных из COM-порта
Argus19 01.05.2026
Асинхронный приём данных из COM-порта Купил на aliexpress термопринтер QR701. Он оказался странным. Поключил к Arduino Nano. Был очень удивлён. Наотрез отказывается печатать русские буквы. Чтобы. . .
попытка написать игровой сервер на C++
pyirrlicht 29.04.2026
попытка написать игровой сервер на плюсах с открытым бесконечным миром. возможно получится прикрутить интерпретатор питон для кастомизации игровой логики. что есть на текущий момент:. . .
Контроль уникальности выбранного документа-основания при изменении реквизита
Maks 28.04.2026
Алгоритм из решения ниже разработан на примере нетипового документа "ЗаявкаНаРемонтСпецтехники", разработанного в КА2. Задача: уведомлять пользователя, если указанная заявка (документ-основание). . .
Благородство как наказание
Maks 24.04.2026
У хорошего человека отношения с женщинами всегда складываются трудно. А я человек хороший. Заявляю без тени смущения, потому что гордиться тут нечем. От хорошего человека ждут соответствующего. . .
Валидация и контроль данных табличной части документа перед записью
Maks 22.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа, разработанного в КА2. Задача: контроль и валидация данных табличной части документа перед записью с учетом регламента компании. . .
Отчёт о затраченных материалах за определенный период с макетом печатной формы
Maks 21.04.2026
Отчёт из решения ниже размещён в конфигурации КА2. Задача: разработка отчёта по затраченным материалам за определённый период, с возможностью вывода печатной формы отчёта с шапкой и подвалом. В. . .
Отчёт о спецтехнике находящейся в ремонте
Maks 20.04.2026
Отчёт из решения ниже размещен в конфигурации КА2. Задача: отобразить спецтехнику, которая на данный момент находится в ремонте. Есть нетиповой документ "Заявка на ремонт спецтехники" который. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru