Как мне изменить QLayout QGroupBox, чтобы удалить или добавить параметр QTextEdit вход? (Python, PyQt)
Я пытаюсь реализовать окно, которое может получить параметры от пользователя. В этом окне пользователь должен указать количество строк (чтобы иметь N наборов параметров). Из этого числа я хотел бы иметь автоматическое поколение QTextEdit
наборы для каждого N
набор параметров. Так что теперь у меня есть этот код
# -*- coding: utf-8 -*-
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
from __future__ import unicode_literals
from PyQt4 import QtGui
import sys
import numpy as np
class SurfViewer(QtGui.QMainWindow):
def __init__(self, parent=None):
super(SurfViewer, self).__init__()
# General Window
self.centralWidget = QtGui.QWidget()
self.setCentralWidget(self.centralWidget)
self.mainHBOX_param_scene = QtGui.QVBoxLayout()
# parameter2
self.horizontalGroupBox_Param2 = QtGui.QGroupBox("Set range Parameter 2 for matrix method")
self.horizontalGroupBox_Param2.setMaximumHeight(80)
self.Param2_layout = QtGui.QVBoxLayout()
grid2 = QtGui.QGridLayout()
self.Param2_layout.addLayout(grid2)
self.Param2_Label = QtGui.QLabel('Name')
self.Param2 = QtGui.QComboBox()
self.Param2.addItems(["a","b","c"])
self.Param2.setMinimumWidth(100)
# self.Param2_layout.addWidget(self.Param2)
self.Param2_min_Label = QtGui.QLabel('min')
self.Param2_min_Edit = QtGui.QTextEdit('1')
self.Param2_min_Edit.setMaximumWidth(40)
self.Param2_max_Label = QtGui.QLabel('max')
self.Param2_max_Edit = QtGui.QTextEdit('5')
self.Param2_max_Edit.setMaximumWidth(40)
self.Param2_step_Label = QtGui.QLabel('step')
self.Param2_step_Edit = QtGui.QTextEdit('1')
self.Param2_step_Edit.setMaximumWidth(40)
grid2.addWidget(self.Param2_Label, 0, 0)
grid2.addWidget(self.Param2, 1, 0)
grid2.addWidget(self.Param2_min_Label, 0, 1)
grid2.addWidget(self.Param2_min_Edit, 1, 1)
grid2.addWidget(self.Param2_max_Label, 0, 2)
grid2.addWidget(self.Param2_max_Edit, 1, 2)
grid2.addWidget(self.Param2_step_Label, 0, 3)
grid2.addWidget(self.Param2_step_Edit, 1, 3)
self.horizontalGroupBox_Param2.setLayout(self.Param2_layout)
#action button
self.horizontalGroupBox_Actions = QtGui.QGroupBox("Actions")
self.horizontalGroupBox_Actions.setMaximumHeight(100)
self.layout_Actions = QtGui.QHBoxLayout()
self.Button_Run=QtGui.QPushButton('Run')
self.nbparam = QtGui.QTextEdit('2')
self.nbparam.setMaximumHeight(30)
self.nbparam.setMaximumWidth(50)
self.layout_Actions.addWidget(self.Button_Run)
self.layout_Actions.addWidget(self.nbparam)
self.horizontalGroupBox_Actions.setLayout(self.layout_Actions)
# set layout
self.mainHBOX_param_scene.addWidget(self.horizontalGroupBox_Actions)
self.mainHBOX_param_scene.addWidget(self.horizontalGroupBox_Param2)
self.centralWidget.setLayout(self.mainHBOX_param_scene)
self.Button_Run.clicked.connect(self.Runclick)
def Runclick(self):
# parameter2
self.horizontalGroupBox_Param = QtGui.QGroupBox("Set range Parameter 2 for matrix method")
nbparam = int(self.nbparam.toPlainText())
self.Param_layout = QtGui.QVBoxLayout()
grid2 = QtGui.QGridLayout()
self.Param_layout.addLayout(grid2)
self.Param_Label = QtGui.QLabel('Names')
self.Param_min_Label = QtGui.QLabel('mins')
self.Param_max_Label = QtGui.QLabel('maxs')
self.Param_step_Label = QtGui.QLabel('steps')
grid2.addWidget(self.Param_Label, 0, 0)
grid2.addWidget(self.Param_min_Label, 0, 1)
grid2.addWidget(self.Param_max_Label, 0, 2)
grid2.addWidget(self.Param_step_Label, 0, 3)
self.listofedit = []
text=["a","b","c"]
for i in np.arange(nbparam):
print(i)
self.listofedit.append([QtGui.QComboBox(),QtGui.QTextEdit('10'),QtGui.QTextEdit('50'),QtGui.QTextEdit('10')])
self.listofedit[i][0].addItems(text)
self.listofedit[i][0].setMinimumWidth(100)
self.listofedit[i][1].setMaximumWidth(40)
self.listofedit[i][2].setMaximumWidth(40)
self.listofedit[i][3].setMaximumWidth(40)
grid2.addWidget(self.listofedit[i][0], i+1, 0)
grid2.addWidget(self.listofedit[i][1], i+1, 1)
grid2.addWidget(self.listofedit[i][2], i+1, 2)
grid2.addWidget(self.listofedit[i][3], i+1, 3)
self.horizontalGroupBox_Param.setLayout(self.Param_layout)
# while self.horizontalGroupBox_Param2.layout().count():
# self.horizontalGroupBox_Param2.layout().itemAt(0).widget().deleteLater()
self.horizontalGroupBox_Param2.layout().clear()
self.horizontalGroupBox_Param2.setLayout(self.Param_layout)
return
if __name__ == '__main__':
app = QtGui.QApplication(sys.argv)
ex = SurfViewer(app)
ex.show()
sys.exit(app.exec_())
Которые генерируют мне это окно:
Поэтому я хотел бы, чтобы, нажимая кнопку "Выполнить" (число в верхнем правом углу - это число строк N), я изменял количество наборов параметров в нижней части окна. Так что в этом случае у меня должно быть две строки для ввода двух имен, двух минут, двух максимумов и двух значений шага. Я начал обновлять окно, но изо всех сил пытаюсь найти правильный код для добавления в def Runclick(self):
Я уже пишу код, чтобы сгенерировать правильное количество строк набора параметров, но мне не удается обновить их в окне.
Я чётко что-то написал с финальной частью:
self.horizontalGroupBox_Param.setLayout(self.Param_layout)
# while self.horizontalGroupBox_Param2.layout().count():
# self.horizontalGroupBox_Param2.layout().itemAt(0).widget().deleteLater()
self.horizontalGroupBox_Param2.setLayout(self.Param_layout)
Я думаю, что я не принимаю вещи правильно. Так что если у кого-то есть решение, я буду благодарен
ОБНОВИТЬ
Привет @jgorosdev, спасибо за ваш ответ. Хотя, это не совсем то, что я хотел бы. В коде, который вы мне дали, он работает правильно, потому что наборы параметров всегда обновляются внизу окна, но в моем более продвинутом программном обеспечении, есть некоторые другие QGroupBox после. Например, если я добавлю консоль, как часть:
# -*- coding: utf-8 -*-
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
from __future__ import unicode_literals
from PyQt4 import QtGui
import sys
import numpy as np
class SurfViewer(QtGui.QMainWindow):
def __init__(self, parent=None):
super(SurfViewer, self).__init__()
# General Window
self.centralWidget = QtGui.QWidget()
self.setCentralWidget(self.centralWidget)
self.mainHBOX_param_scene = QtGui.QVBoxLayout()
#action button
self.horizontalGroupBox_Actions = QtGui.QGroupBox("Actions")
self.horizontalGroupBox_Actions.setMaximumHeight(100)
self.layout_Actions = QtGui.QHBoxLayout()
self.Button_Run=QtGui.QPushButton('Run')
self.nbparam = QtGui.QTextEdit('3')
self.nbparam.setMaximumHeight(30)
self.nbparam.setMaximumWidth(50)
self.layout_Actions.addWidget(self.Button_Run)
self.layout_Actions.addWidget(self.nbparam)
self.horizontalGroupBox_Actions.setLayout(self.layout_Actions)
self.mainHBOX_param_scene.addWidget(self.horizontalGroupBox_Actions)
self.centralWidget.setLayout(self.mainHBOX_param_scene)
self.updateWidget = None
# Param2 layout
self.horizontalGroupBox_Param2 = QtGui.QGroupBox("Set range Parameter 2 for matrix method")
self.Param2_layout = QtGui.QVBoxLayout()
self.Button_Run.clicked.connect(self.UpdateLayout)
self.UpdateLayout()
# console
self.horizontalGroupBox_Console2 = QtGui.QGroupBox("Console ")
self.layout_Consol_Label2 = QtGui.QVBoxLayout()
self.Consol_Label2 = QtGui.QTextEdit('Console :')
self.layout_Consol_Label2.addWidget(self.Consol_Label2)
self.horizontalGroupBox_Console2.setLayout(self.layout_Consol_Label2)
self.mainHBOX_param_scene.addWidget(self.horizontalGroupBox_Console2)
# parameter2
# From now on everything well be created inside a function that updates the layout
def UpdateLayout(self):
if (self.updateWidget == None):
self.updateWidget = QtGui.QWidget()
else:
self.mainHBOX_param_scene.removeWidget(self.updateWidget)
self.updateWidget.deleteLater()
self.updateWidget = None
cnt = self.mainHBOX_param_scene.layout().count()
print("Widget count:" +str(cnt)+ "")
nbparam = int(self.nbparam.toPlainText())
#self.horizontalGroupBox_Param2.setMaximumHeight(150)
grid2 = QtGui.QGridLayout()
self.Param2_Label = QtGui.QLabel('Name')
self.Param2 = QtGui.QComboBox()
self.Param2.addItems(["a","b","c"])
self.Param2.setMinimumWidth(100)
# self.Param2_layout.addWidget(self.Param2)
self.Param2_min_Label = QtGui.QLabel('min')
self.Param2_max_Label = QtGui.QLabel('max')
self.Param2_step_Label = QtGui.QLabel('step')
grid2.addWidget(self.Param2_Label, 0, 0)
grid2.addWidget(self.Param2_min_Label, 0, 1)
grid2.addWidget(self.Param2_max_Label, 0, 2)
grid2.addWidget(self.Param2_step_Label, 0, 3)
self.listofedit = []
text=["a","b","c"]
for i in np.arange(nbparam):
print(i)
self.listofedit.append([QtGui.QComboBox(),QtGui.QTextEdit('20'),QtGui.QTextEdit('50'),QtGui.QTextEdit('10')])
self.listofedit[i][0].addItems(text)
self.listofedit[i][0].setMinimumWidth(100)
self.listofedit[i][1].setMaximumWidth(40)
self.listofedit[i][1].setMaximumHeight(40)
self.listofedit[i][2].setMaximumWidth(40)
self.listofedit[i][2].setMaximumHeight(40)
self.listofedit[i][3].setMaximumWidth(40)
self.listofedit[i][3].setMaximumHeight(40)
grid2.addWidget(self.listofedit[i][0], i+1, 0)
grid2.addWidget(self.listofedit[i][1], i+1, 1)
grid2.addWidget(self.listofedit[i][2], i+1, 2)
grid2.addWidget(self.listofedit[i][3], i+1, 3)
self.updateWidget = QtGui.QWidget()
self.updateWidget.setLayout(grid2)
#self.updateWidget.setMaximumHeight(nbparam*60)
self.Param2_layout.addWidget(self.updateWidget)
self.horizontalGroupBox_Param2.setLayout(self.Param2_layout)
self.mainHBOX_param_scene.addWidget(self.horizontalGroupBox_Param2)
return
if __name__ == '__main__':
app = QtGui.QApplication(sys.argv)
ex = SurfViewer(app)
ex.show()
sys.exit(app.exec_())
У меня есть это окно в начале:
Это то, что я хотел бы, но я получаю это изображение после нажатия на кнопку Run:
Таким образом, консоль и набор параметров инвертированы. Мне интересно, можно ли заменить часть макета без повторной сборки всего макета?
1 ответ
Я сделал несколько небольших изменений в коде и теперь работает хорошо.
Вы два раза создавали нижнюю группу в своем коде, и я думаю, что это лучший способ. Я упаковал все в UpdateLayout
функция.
Вам, вероятно, придется повторно проверить политику изменения размера qt, чтобы правильно изменить размер главного окна.
Код с ОБНОВЛЕНИЕМ:
# -*- coding: utf-8 -*-
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
from __future__ import unicode_literals
from PyQt4 import QtGui
import sys
import numpy as np
class SurfViewer(QtGui.QMainWindow):
def __init__(self, parent=None):
super(SurfViewer, self).__init__()
# General Window
self.centralWidget = QtGui.QWidget()
self.setCentralWidget(self.centralWidget)
self.mainHBOX_param_scene = QtGui.QVBoxLayout()
#action button
self.horizontalGroupBox_Actions = QtGui.QGroupBox("Actions")
self.horizontalGroupBox_Actions.setMaximumHeight(100)
self.layout_Actions = QtGui.QHBoxLayout()
self.Button_Run=QtGui.QPushButton('Run')
self.nbparam = QtGui.QTextEdit('3')
self.nbparam.setMaximumHeight(30)
self.nbparam.setMaximumWidth(50)
self.layout_Actions.addWidget(self.Button_Run)
self.layout_Actions.addWidget(self.nbparam)
self.horizontalGroupBox_Actions.setLayout(self.layout_Actions)
self.mainHBOX_param_scene.addWidget(self.horizontalGroupBox_Actions)
self.centralWidget.setLayout(self.mainHBOX_param_scene)
self.updateWidget = None
# Param2 layout
self.horizontalGroupBox_Param2 = QtGui.QGroupBox("Set range Parameter 2 for matrix method")
self.Param2_layout = QtGui.QVBoxLayout()
self.Button_Run.clicked.connect(self.UpdateLayout)
self.UpdateLayout()
# console
self.horizontalGroupBox_Console2 = QtGui.QGroupBox("Console ")
self.layout_Consol_Label2 = QtGui.QVBoxLayout()
self.Consol_Label2 = QtGui.QTextEdit('Console :')
self.layout_Consol_Label2.addWidget(self.Consol_Label2)
self.horizontalGroupBox_Console2.setLayout(self.layout_Consol_Label2)
self.mainHBOX_param_scene.addWidget(self.horizontalGroupBox_Console2)
# parameter2
# From now on everything well be created inside a function that updates the layout
def UpdateLayout(self):
if (self.updateWidget == None):
self.updateWidget = QtGui.QWidget()
else:
self.mainHBOX_param_scene.removeWidget(self.updateWidget)
self.updateWidget.deleteLater()
self.updateWidget = None
cnt = self.mainHBOX_param_scene.layout().count()
print("Widget count:" +str(cnt)+ "")
nbparam = int(self.nbparam.toPlainText())
#self.horizontalGroupBox_Param2.setMaximumHeight(150)
grid2 = QtGui.QGridLayout()
self.Param2_Label = QtGui.QLabel('Name')
self.Param2 = QtGui.QComboBox()
self.Param2.addItems(["a","b","c"])
self.Param2.setMinimumWidth(100)
# self.Param2_layout.addWidget(self.Param2)
self.Param2_min_Label = QtGui.QLabel('min')
self.Param2_max_Label = QtGui.QLabel('max')
self.Param2_step_Label = QtGui.QLabel('step')
grid2.addWidget(self.Param2_Label, 0, 0)
grid2.addWidget(self.Param2_min_Label, 0, 1)
grid2.addWidget(self.Param2_max_Label, 0, 2)
grid2.addWidget(self.Param2_step_Label, 0, 3)
self.listofedit = []
text=["a","b","c"]
for i in np.arange(nbparam):
print(i)
self.listofedit.append([QtGui.QComboBox(),QtGui.QTextEdit('20'),QtGui.QTextEdit('50'),QtGui.QTextEdit('10')])
self.listofedit[i][0].addItems(text)
self.listofedit[i][0].setMinimumWidth(100)
self.listofedit[i][1].setMaximumWidth(40)
self.listofedit[i][1].setMaximumHeight(40)
self.listofedit[i][2].setMaximumWidth(40)
self.listofedit[i][2].setMaximumHeight(40)
self.listofedit[i][3].setMaximumWidth(40)
self.listofedit[i][3].setMaximumHeight(40)
grid2.addWidget(self.listofedit[i][0], i+1, 0)
grid2.addWidget(self.listofedit[i][1], i+1, 1)
grid2.addWidget(self.listofedit[i][2], i+1, 2)
grid2.addWidget(self.listofedit[i][3], i+1, 3)
self.updateWidget = QtGui.QWidget()
self.updateWidget.setLayout(grid2)
#self.updateWidget.setMaximumHeight(nbparam*60)
self.Param2_layout.addWidget(self.updateWidget,)
self.horizontalGroupBox_Param2.setLayout(self.Param2_layout)
#self.mainHBOX_param_scene.addWidget(self.horizontalGroupBox_Param2)
# NEW
self.mainHBOX_param_scene.insertWidget(1,self.horizontalGroupBox_Param2)
return
if __name__ == '__main__':
app = QtGui.QApplication(sys.argv)
ex = SurfViewer(app)
ex.show()
sys.exit(app.exec_())
Ответ на ОБНОВЛЕНИЕ
Если вы хотите обновить QGroupBox без изменения позиции, вместо addWidget()
функция, которая вам понадобится, это insertWidget()
функция.
Я обновил строку в UpdateLayout
функция, которая добавляла виджет. Надеюсь, что это хорошо для вас.
Ссылка на функцию следующая: