动态添加时如何设置对象的名称?

Posted

技术标签:

【中文标题】动态添加时如何设置对象的名称?【英文标题】:How to set the name of an object when adding dynamically? 【发布时间】:2021-12-30 21:41:47 【问题描述】:

我正在创建一个卡路里计算器,我需要动态添加食物。我正在动态添加小部件。例如,我添加了一个标签。但在 setObjectName 之后,我无法使用该名称访问。例如,我将标签的名称设置为“名称”。我正在尝试使用以下图像 name.text() 按名称获取标签的文本,但代码崩溃了。我做错了什么?

代码如下:

import sys  # interaction with Python

from PyQt5.QtWidgets import *  # for classic application based on widgets
from PyQt5 import uic  # to read ui file
from PyQt5 import QtWidgets  # to create gui


class MyWin(QtWidgets.QMainWindow):  # create class witch inherit QMainWindow
    def __init__(self):  # constructor
        QtWidgets.QMainWindow.__init__(self)  # constructor of parent class
        uic.loadUi("gui.ui", self)  # load ui

        self.add_product.clicked.connect(self.add)
        self.remove_product.clicked.connect(self.remove)

    def add(self):
        h1 = QHBoxLayout()
        self.label = QLabel()
        self.label.setObjectName("name")  # set name
        self.label.text = "L"
        h1.addWidget(self.label)
        h1.addWidget(QLabel('Weight'))
        h2 = QHBoxLayout()
        h2.addWidget(QLineEdit())
        h2.addWidget(QLineEdit())
        i = self.verticalLayout_2.count()
        self.verticalLayout_2.insertLayout(i - 2, h1)
        self.verticalLayout_2.insertLayout(i - 1, h2)
        print(self.name.text())  # ERROR
    def remove(self):
        i = self.verticalLayout_2.count()
        if i > 3:
            QWidget().setLayout(self.verticalLayout_2.takeAt(i - 3))
            QWidget().setLayout(self.verticalLayout_2.takeAt(i - 4))


if __name__ == '__main__':  # for check non import module
    app = QApplication(sys.argv)  # create app
    mw = MyWin()  # create object of MyWin class
    mw.show()  # to show gui
    sys.exit(app.exec_())  # execute app

用户界面:

<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
 <class>MainWindow</class>
 <widget class="QMainWindow" name="MainWindow">
  <property name="geometry">
   <rect>
    <x>0</x>
    <y>0</y>
    <width>458</width>
    <height>234</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>MainWindow</string>
  </property>
  <widget class="QWidget" name="centralwidget">
   <layout class="QVBoxLayout" name="verticalLayout">
    <item>
     <widget class="QGroupBox" name="breakfest">
      <property name="title">
       <string>Breakfest</string>
      </property>
      <layout class="QVBoxLayout" name="verticalLayout_3">
       <item>
        <layout class="QVBoxLayout" name="verticalLayout_2">
         <item>
          <layout class="QHBoxLayout" name="horizontalLayout_2">
           <item>
            <widget class="QLabel" name="label_product">
             <property name="text">
              <string>Product</string>
             </property>
            </widget>
           </item>
           <item>
            <widget class="QLabel" name="label_weight">
             <property name="text">
              <string>Weight</string>
             </property>
            </widget>
           </item>
          </layout>
         </item>
         <item>
          <layout class="QHBoxLayout" name="horizontalLayout">
           <item>
            <widget class="QLineEdit" name="lineEdit_product"/>
           </item>
           <item>
            <widget class="QLineEdit" name="lineEdit_weight"/>
           </item>
          </layout>
         </item>
         <item>
          <spacer name="verticalSpacer">
           <property name="orientation">
            <enum>Qt::Vertical</enum>
           </property>
           <property name="sizeHint" stdset="0">
            <size>
             <width>20</width>
             <height>40</height>
            </size>
           </property>
          </spacer>
         </item>
         <item>
          <layout class="QHBoxLayout" name="horizontalLayout_3">
           <item>
            <widget class="QPushButton" name="add_product">
             <property name="text">
              <string>Add product</string>
             </property>
            </widget>
           </item>
           <item>
            <widget class="QPushButton" name="remove_product">
             <property name="text">
              <string>Remove product</string>
             </property>
            </widget>
           </item>
          </layout>
         </item>
        </layout>
       </item>
      </layout>
     </widget>
    </item>
   </layout>
  </widget>
  <widget class="QStatusBar" name="statusbar"/>
 </widget>
 <resources/>
 <connections/>
</ui>

【问题讨论】:

与其使用动态属性,不如将所有标签放在一个dict中;或者也许直接通过布局访问小部件。创建一个单独的Product 小部件类(可能在 Qt 设计器中)可能也是一个好主意,这样您就可以使用更连贯的 API。 (PS:还有getattr/setattr,但我认为这对您的用例来说不是一个好的长期解决方案)。 @ekhumoro 我如何通过布局访问?例如,我创建了一个 label = QLabel ("W") 对象。将其命名为 label.setObjectName(“名称”)。但是当我用这个名字 self.name.text = "L" 调用时,就会发生错误。 【参考方案1】:

您不能简单地使用 objectName 访问对象。您需要使用findChild 方法。

选项 1:

self.widget_i_want = self.findChild(QLabel, "name")

但正如@ekhumoro 所说,最好使用列表或字典来存储所有添加的标签。

选项 2:

class MyWin(QtWidgets.QMainWindow):
    def __init__(self):
        QtWidgets.QMainWindow.__init__(self)
        uic.loadUi("gui.ui", self)  # load ui
        
        self.counter = 1  #this is going to be used to set unique names
        self.add_product.clicked.connect(self.add)
        self.remove_product.clicked.connect(self.remove)
        
        self.dynamically_added_widgets = list() 

    def add(self):
        global counter
        self.counter += 1
        h1 = QHBoxLayout()
        self.label = QLabel()
        self.label.setObjectName(f"nameself.counter")  # set a new, unique name
        self.label.text = "L"
        h1.addWidget(self.label)
        h1.addWidget(QLabel('Weight'))
        h2 = QHBoxLayout()
        h2.addWidget(QLineEdit())
        h2.addWidget(QLineEdit())
        i = self.verticalLayout_2.count()
        self.verticalLayout_2.insertLayout(i - 2, h1)
        self.verticalLayout_2.insertLayout(i - 1, h2)

        self.dynamically_added_widgets.append(self.label) # add the new label to list
        print(self.dynamically_added_widgets)
        print(self.dynamically_added_widgets[-1].objectName())  # print the last added label's objectName
    def remove(self):
        i = self.verticalLayout_2.count()
        if i > 3:
            QWidget().setLayout(self.verticalLayout_2.takeAt(i - 3))
            QWidget().setLayout(self.verticalLayout_2.takeAt(i - 4))

【讨论】:

以上是关于动态添加时如何设置对象的名称?的主要内容,如果未能解决你的问题,请参考以下文章

js里面怎么动态的为对象添加属性

如何在FastReport报表中动态添加数据集

如何在FastReport报表中动态添加数据集

如何在FastReport报表中动态添加数据集

react 对象动态添加属性 setState

如何设置在加载所有动态添加的 iframe 时触发的 window.onload 事件?