Форум программистов, компьютерный форум, киберфорум
Python: GUI, графика
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.62/13: Рейтинг темы: голосов - 13, средняя оценка - 4.62
0 / 0 / 0
Регистрация: 22.06.2016
Сообщений: 22
PyQt5

Открывание окон (переход между ними), учусь разбираться в классах

27.03.2019, 10:20. Показов 2502. Ответов 6

Студворк — интернет-сервис помощи студентам
Набросал небольшую прогу, на основе учебника. Приложение, которое будет управлять бд.
Вопрос 1
Почему в функцию(open или openZvuk) нам нужно передавать self, ведь по сути я просто создаю объект другого класса
Вопрос 2
После нажатия на кнопку, которая открывает третье окно ( class WindowZvuk) просто мгновенно закрывается и консоль пишет рестарт и ничего не происходит. Пробовал отладить этот фрагмент просто, без создания класса, всё отлично работает. Из этого стоит вывод, что я накосячил в создании класса или чём-то подобном, но сам разобраться не могу. За критику и подсказки буду благодарен, только начал изучать
Код:
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
from PyQt5 import QtWidgets, QtSql, QtCore
import sys
 
app = QtWidgets.QApplication(sys.argv)
class Windows(QtWidgets.QWidget):# класс окна "Студия звукозаписи"
    def __init__(self, parent = None):#parent = None for works 
        QtWidgets.QWidget.__init__(self, parent)# for works
        btn1 = QtWidgets.QPushButton("Просмотреть таблицы")
        self.resize(400, 300)
        self.setWindowTitle("Студия звукозаписи")
        vbox = QtWidgets.QVBoxLayout()
        vbox.addWidget(btn1)
        self.setLayout(vbox)
        btn1.clicked.connect(self.open)
    def open(self):
        self.Twowin = Window2()
        self.Twowin.show()
        self.hide()
 
                     
class Window2(QtWidgets.QWidget):# класс окна "Выбор таблиц"
     def __init__(self, parent = None):#parent = None for works 
        QtWidgets.QWidget.__init__(self, parent)# for works
        self.resize(400, 300)
        self.setWindowTitle("Выбор таблиц")
        btnTableZvuk = QtWidgets.QPushButton("Таблица Звукорежиссёров")
        vbox2 = QtWidgets.QVBoxLayout()
        vbox2.addWidget(btnTableZvuk)
        self.setLayout(vbox2)
        btnTableZvuk.clicked.connect(self.openZvuk)
     def openZvuk(self):
         self.WindowZv = WindowZvuk()# вот тут проверить, как рабоатет self.
         self.WindowZv.show()#
         
 
 
         
class WindowZvuk(QtWidgets.QWidget):# класс окна "Выбор таблиц"
     def __init__(self, parent = None):#parent = None for works 
        QtWidgets.QWidget.__init__(self, parent)# for works
        self.resize(1000, 1000)
        con = QtSql.QSqlDatabase.addDatabase('QSQLITE')
        con.setDatabaseName('путь')#здесь специально нет пути, в оригинале он есть
        con.open()
        stm = QtSql.QSqlTableModel(parent = WindowZv)
        stm.setTable('ZvukRez')
        stm.setSort(1, QtCore.Qt.AscendingOrder)
        stm.select()
        stm.setHeaderData(1, QtCore.Qt.Horizontal, 'Столбец 1')
        stm.setHeaderData(2, QtCore.Qt.Horizontal, 'Столбец 2')
        stm.setHeaderData(3, QtCore.Qt.Horizontal, 'Столбец 3j')
        stm.setHeaderData(4, QtCore.Qt.Horizontal, 'Столбец 4')       
        stm.setHeaderData(5, QtCore.Qt.Horizontal, 'Столбец 5')
        vbox3 = QtWidgets.QVBoxLayout()
        tv = QtWidgets.QTableView()
        tv.setmodel(stm)
        tv.hideColumn(0)
        tv.setColumnWidth(1, 150)
        tv.setColumnWidth(2, 150)
        tv.setColumnWidth(3, 150)
        tv.setColumnWidth(4, 150)
        tv.setColumnWidth(5, 150)
        vbox3.addWidget(tv)
        self.setLayout(self.vbox3)
        
    # def addRecord():
     #   stm.insertRow(stm.rowCount())
 
    
     #def delRecord():
      #  stm.removeRow(tv.currentIndex().row())
      #  stm.select()
 
    
 
        
MainWindow = Windows()
MainWindow.show()
0
Лучшие ответы (1)
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
27.03.2019, 10:20
Ответы с готовыми решениями:

Две ОС одновременно - переход между ними
Доброго времени суток всем. Я установил себе на машину Linux и Виндовс. Подскажите , пожалуйста , как мне из Линукса попасть в виндовс и...

Двойное меню и переход между ними
Добрый вечер, подскажите пожалуйста, как организовать возврат из одного меню в другое? Точнее: я написал код, и в нем содержится двойное...

Переход на следующую страницу(между ними скрытая )
Подскажите, пожалуйста, у меня есть ComboBox, в котором содержится 8 элементов. Все листы изначально скрыты. В зависимости от выбранного...

6
1293 / 677 / 367
Регистрация: 07.01.2019
Сообщений: 2,302
27.03.2019, 11:46
Цитата Сообщение от Mododel Посмотреть сообщение
После нажатия на кнопку, которая открывает третье окно ( class WindowZvuk) просто мгновенно закрывается и консоль пишет рестарт и ничего не происходит. Пробовал отладить этот фрагмент просто, без создания класса, всё отлично работает.
У меня не работает с ошибкой

NameError: name 'WindowZv' is not defined
Цитата Сообщение от Mododel Посмотреть сообщение
Почему в функцию(open или openZvuk) нам нужно передавать self, ведь по сути я просто создаю объект другого класса
self для приема аргумента, во все методы класса передается аргумент контекста и его надо принять, обычно переменная для приема называется self, хотя имя может быть любым, без этой переменной будет ошибка

takes 0 positional arguments but 1 was given
0
963 / 718 / 276
Регистрация: 10.12.2016
Сообщений: 1,764
27.03.2019, 12:44
Лучший ответ Сообщение было отмечено Mododel как решение

Решение

self в функции значит что метод принадлежит классу, в данных - хранимая ссылка
вы ошибки БД не проверяете нигде и зачем так сложно?
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
from PyQt5.QtCore import Qt
from PyQt5.QtWidgets import QApplication, QTableView, QWidget, QGridLayout, QPushButton, QComboBox
from PyQt5.QtSql import QSqlTableModel, QSqlDatabase, QSqlQuery
 
class View(QWidget):
    def __init__(self):
        super().__init__()
        self.view = QTableView()
        self.view.horizontalHeader().setSectionResizeMode(1)
        self.btnAdd = QPushButton('Добавить запись')
        self.btnAdd.clicked.connect(self.add_record)
        self.box = QComboBox()
        grid = QGridLayout(self)
        grid.setContentsMargins(0,0,0,0)
        grid.addWidget(self.view,0,0,4,4)
        grid.addWidget(self.box,4,0,1,1)
        grid.addWidget(self.btnAdd,4,1,1,1)
        self.db = self.open_db()
        self.box.addItems(self.db.tables())
        self.box.currentIndexChanged.connect(self.on_index)
        self.model = QSqlTableModel()
        self.on_index()
        self.model.setEditStrategy(QSqlTableModel.OnFieldChange)
        self.model.setHeaderData(0, Qt.Horizontal, "Вопрос")
        self.model.setHeaderData(1, Qt.Horizontal, "Ответ")
        self.view.setModel(self.model)
        self.model.select()
        
    def open_db(self):
        db = QSqlDatabase.addDatabase('QSQLITE')
        db.setDatabaseName('tst.db')
        if not(db.open()): 
            print(db.lastErrorText())
            exit(1)
        q = QSqlQuery('create table if not exists tbl(quest text,answ text)',db)
        if not q.exec_(): 
            print(q.lastErrorText())
        return db
    
    def on_index(self):
        self.model.setTable(self.box.currentText())
        
    def add_record(self):
        idx = self.model.rowCount()
        self.model.insertRow(idx)
 
if __name__ == '__main__':
 
    import sys
    app = QApplication(sys.argv)
    w = View()
    w.resize(600,400)
    w.show()
    sys.exit(app.exec_())
Добавлено через 40 минут
звиняюсь, не тот код приложил
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
from PyQt5.QtCore import Qt
from PyQt5.QtWidgets import (QApplication, QTableView, QWidget, QGridLayout, QPushButton, QComboBox,
QFileDialog, QMessageBox)
from PyQt5.QtSql import QSqlTableModel, QSqlDatabase, QSqlQuery
 
class View(QWidget):
    def __init__(self):
        super().__init__()
        self.dbname = ''
        self.db = QSqlDatabase.addDatabase('QSQLITE')
        self.view = QTableView()
        self.view.horizontalHeader().setSectionResizeMode(1)
        self.btnAdd = QPushButton('Добавить запись')
        self.btnAdd.clicked.connect(self.add_record)
        self.btnOpen = QPushButton('Открыть базу данных')
        self.btnOpen.clicked.connect(self.open_db)
        self.box = QComboBox()
        self.box.currentIndexChanged.connect(self.on_index)
        grid = QGridLayout(self)
        grid.setContentsMargins(0,0,0,0)
        grid.addWidget(self.view,0,0,4,4)
        grid.addWidget(self.box,4,0,1,1)
        grid.addWidget(self.btnOpen,4,1,1,1)
        grid.addWidget(self.btnAdd,4,2,1,1)
        self.model = QSqlTableModel()
        self.model.setEditStrategy(QSqlTableModel.OnFieldChange)
        self.view.setModel(self.model)
        
    def open_db(self):
        dbname,_ = QFileDialog.getOpenFileName(self,'Открыть базу данных',self.dbname,'*.db *.sqlite')
        if not dbname: return
        self.dbname = dbname
        self.db.setDatabaseName(self.dbname)
        if not(self.db.open()):
            QMessageBox.critical(self,"",self.db.lastError().text())
            return
        self.box.clear()
        self.box.addItems(self.db.tables())
    
    def on_index(self):
        table = self.box.currentText()
        self.model.setTable(table)
        if table == 'tbl':
            self.model.setHeaderData(0, Qt.Horizontal, "Вопрос")
            self.model.setHeaderData(1, Qt.Horizontal, "Ответ")
        self.model.select()
        
    def add_record(self):
        idx = self.model.rowCount()
        self.model.insertRow(idx)
 
if __name__ == '__main__':
 
    import sys
    app = QApplication(sys.argv)
    w = View()
    w.resize(600,400)
    w.show()
    sys.exit(app.exec_())
1
0 / 0 / 0
Регистрация: 22.06.2016
Сообщений: 22
27.03.2019, 15:34  [ТС]
Добавлено через 5 минут
Спасибо, vic5710, работает, вот только пытаюсь добавить кнопку удаления записи к этому же коду, но что-то не хочет работать
Python
1
2
3
def del_record():
        self.model.removeRow()
        #self.model.select()
Добавлено через 52 минуты
еще вот так пробовал, результат тот же - вылет

Python
1
2
        self.model.removeRow(self.view.currentIndex().row())
        self.model.select()
0
963 / 718 / 276
Регистрация: 10.12.2016
Сообщений: 1,764
27.03.2019, 16:02
опять же - ничего не проверяете. с БД так нельзя
Python
1
2
3
4
5
    try:
        self.model.removeRow(self.view.currentIndex().row())
     except Exception as e:
        print(e)   
     self.model.select()
0
0 / 0 / 0
Регистрация: 22.06.2016
Сообщений: 22
27.03.2019, 16:21  [ТС]
Извиняюсь за свою корявость. Просто сначала пытаюсь сделать чтобы хотя бы работала, а потом осуществлять проверки. Кстати, функция всё равно вылетает.
И ещё вопрос, шарю документация QGridLayout, но никак не могу понять, что делают последние две цифры, когда мы добавляем виджет. Первые две - координаты

Добавлено через 12 минут
"И ещё вопрос, шарю документация QGridLayout, но никак не могу понять, что делают последние две цифры, когда мы добавляем виджет. Первые две - координаты"

с этим разобрался
0
963 / 718 / 276
Регистрация: 10.12.2016
Сообщений: 1,764
27.03.2019, 22:30
я контекстное меню юзаю
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
from PyQt5.QtCore import Qt
from PyQt5.QtWidgets import (QApplication, QTableView, QWidget, QGridLayout, QPushButton, QComboBox,
QFileDialog, QMessageBox, QMenu)
from PyQt5.QtSql import QSqlTableModel, QSqlDatabase, QSqlQuery, QSqlRecord
 
class View(QWidget):
    def __init__(self):
        super().__init__()
        self.dbname = ''
        self.db = QSqlDatabase.addDatabase('QSQLITE')
        self.view = QTableView()
        self.view.horizontalHeader().setSectionResizeMode(1)
        self.btnOpen = QPushButton('Открыть базу данных')
        self.btnOpen.clicked.connect(self.open_db)
        self.box = QComboBox()
        self.box.currentIndexChanged.connect(self.on_index)
        grid = QGridLayout(self)
        grid.setContentsMargins(0,0,0,0)
        grid.addWidget(self.view,0,0,4,4)
        grid.addWidget(self.box,4,0,1,1)
        grid.addWidget(self.btnOpen,4,1,1,1)
        self.model = QSqlTableModel()
        self.model.setEditStrategy(QSqlTableModel.OnManualSubmit)
        
    def open_db(self):
        dbname,_ = QFileDialog.getOpenFileName(self,'Открыть базу данных',self.dbname,'*.db *.sqlite')
        if not dbname: return
        self.dbname = dbname
        self.db.setDatabaseName(self.dbname)
        if not(self.db.open()):
            QMessageBox.critical(self,"",self.db.lastError().text())
            return
        self.box.clear()
        self.box.addItems(self.db.tables())
    
    def on_index(self):
        table = self.box.currentText()
        self.model.setTable(table)
        if table == 'tbl':
            self.model.setHeaderData(0, Qt.Horizontal, "Вопрос")
            self.model.setHeaderData(1, Qt.Horizontal, "Ответ")
        self.view.setModel(self.model)
        self.model.select()
 
    def contextMenuEvent(self,event):
        pos = self.view.mapToGlobal(event.pos())
        mnu = QMenu()
        mnu.addAction('Add record').setObjectName('add')
        mnu.addAction('Delete record').setObjectName('del')
        mnu.addAction('Submit all').setObjectName('submit')
        ret = mnu.exec_(pos)
        if not ret: return
        obj = ret.objectName()
        if obj == 'add':
            row = self.model.rowCount()
            self.model.insertRow(row)
            self.model.setData(self.model.index(row,0),None)
            self.model.submitAll()
        elif obj == 'del':
            idx = self.view.currentIndex()
            self.model.removeRow(idx.row())
            self.model.submitAll()
        else:
            self.model.submitAll()
        self.model.select()
 
if __name__ == '__main__':
    import sys
    app = QApplication(sys.argv)
    w = View()
    w.resize(600,400)
    w.show()
    sys.exit(app.exec_())
1
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
27.03.2019, 22:30
Помогаю со студенческими работами здесь

Создать два activity, организовать переход между ними
Нужно создать два activity. Первое должно содержать кнопку btn1. Второе должно содержать текст TextView с текстом "Переданный...

Telegram бот и «Сценарии» как правильно организовать переход между ними?
У меня на главной странице index.js при старте уводит на файл Scenegenerator(там розмещены сцены) const currScene = new...

Автоматическое открывание окон рекламы в Гугл
реклама вылетает по 3-4 окна. Помогите удалить заразу.

Самостоятельное открывание окон и вкладок браузера с рекламой
Самостоятельное открывание окон и вкладок браузера с рекламой вот лог

Само открывание окон рекламы во всех браузерах
Добрый день помогите пожалуйста! Вот логи


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
7
Ответ Создать тему
Новые блоги и статьи
SDL3 для Web (WebAssembly): Реализация движения на Box2D v3 - трение и коллизии с повёрнутыми стенами
8Observer8 20.02.2026
Содержание блога Box2D позволяет легко создать главного героя, который не проходит сквозь стены и перемещается с заданным трением о препятствия, которые можно располагать под углом, как верхнее. . .
Конвертировать закладки radiotray-ng в m3u-плейлист
damix 19.02.2026
Это можно сделать скриптом для PowerShell. Использование . \СonvertRadiotrayToM3U. ps1 <path_to_bookmarks. json> Рядом с файлом bookmarks. json появится файл bookmarks. m3u с результатом. # Check if. . .
Семь CDC на одном интерфейсе: 5 U[S]ARTов, 1 CAN и 1 SSI
Eddy_Em 18.02.2026
Постепенно допиливаю свою "многоинтерфейсную плату". Выглядит вот так: https:/ / www. cyberforum. ru/ blog_attachment. php?attachmentid=11617&stc=1&d=1771445347 Основана на STM32F303RBT6. На борту пять. . .
Камера Toupcam IUA500KMA
Eddy_Em 12.02.2026
Т. к. у всяких "хикроботов" слишком уж мелкий пиксель, для подсмотра в ESPriF они вообще плохо годятся: уже 14 величину можно рассмотреть еле-еле лишь на экспозициях под 3 секунды (а то и больше),. . .
И ясному Солнцу
zbw 12.02.2026
И ясному Солнцу, и светлой Луне. В мире покоя нет и люди не могут жить в тишине. А жить им немного лет.
«Знание-Сила»
zbw 12.02.2026
«Знание-Сила» «Время-Деньги» «Деньги -Пуля»
SDL3 для Web (WebAssembly): Подключение Box2D v3, физика и отрисовка коллайдеров
8Observer8 12.02.2026
Содержание блога Box2D - это библиотека для 2D физики для анимаций и игр. С её помощью можно определять были ли коллизии между конкретными объектами и вызывать обработчики событий столкновения. . . .
SDL3 для Web (WebAssembly): Загрузка PNG с прозрачным фоном с помощью SDL_LoadPNG (без SDL3_image)
8Observer8 11.02.2026
Содержание блога Библиотека SDL3 содержит встроенные инструменты для базовой работы с изображениями - без использования библиотеки SDL3_image. Пошагово создадим проект для загрузки изображения. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru