В этом случае лучше всего реализовать собственный QListWidget, который позволяет вам получить строку, виджет не должен знать, к какой строке он принадлежит, но список-виджет должен знать это.
Так что я создамниже приведен пользовательский QListWidget, который испускает сигнал clicked_widget, указывающий номер строки виджета, испускающего нажатый сигнал:
clickablelistwidget.py
from PyQt5 import QtCore, QtWidgets
class ClickableListWidget(QtWidgets.QListWidget):
clicked_widget = QtCore.pyqtSignal(int)
def append_widget(self, widget_clickable):
item = QtWidgets.QListWidgetItem()
self.addItem(item)
item.setSizeHint(widget_clickable.sizeHint())
self.setItemWidget(item, widget_clickable)
if hasattr(widget_clickable, "clicked"):
if isinstance(widget_clickable.clicked, QtCore.pyqtBoundSignal):
widget_clickable.clicked.connect(self.on_clicked)
@QtCore.pyqtSlot()
def on_clicked(self):
widget = self.sender()
gp = widget.mapToGlobal(QtCore.QPoint())
lp = self.viewport().mapFromGlobal(gp)
item = self.itemAt(lp)
row = self.row(item)
self.clicked_widget.emit(row)
Затем изменитефайл Main_GUI.py, чтобы заменить QListWidget новым классом:
Main_GUI.py
from PyQt5 import QtCore, QtGui, QtWidgets
from clickablelistwidget import ClickableListWidget
class Ui_MainWindow(object):
def setupUi(self, MainWindow):
MainWindow.setObjectName("MainWindow")
MainWindow.resize(353, 237)
self.centralwidget = QtWidgets.QWidget(MainWindow)
self.centralwidget.setObjectName("centralwidget")
self.gridLayout = QtWidgets.QGridLayout(self.centralwidget)
self.gridLayout.setObjectName("gridLayout")
self.listWidget = ClickableListWidget(self.centralwidget) # <---
self.listWidget.setObjectName("listWidget")
self.gridLayout.addWidget(self.listWidget, 0, 0, 1, 1)
self.pushButton = QtWidgets.QPushButton(self.centralwidget)
self.pushButton.setObjectName("pushButton")
self.gridLayout.addWidget(self.pushButton, 1, 0, 1, 1)
MainWindow.setCentralWidget(self.centralwidget)
self.menubar = QtWidgets.QMenuBar(MainWindow)
self.menubar.setGeometry(QtCore.QRect(0, 0, 353, 21))
self.menubar.setObjectName("menubar")
MainWindow.setMenuBar(self.menubar)
self.statusbar = QtWidgets.QStatusBar(MainWindow)
self.statusbar.setObjectName("statusbar")
MainWindow.setStatusBar(self.statusbar)
self.retranslateUi(MainWindow)
QtCore.QMetaObject.connectSlotsByName(MainWindow)
def retranslateUi(self, MainWindow):
_translate = QtCore.QCoreApplication.translate
MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))
self.pushButton.setText(_translate("MainWindow", "Add Item"))
Ваш пользовательский виджет изменен так, что он излучает сигнал, по которому нажималикогда нажата кнопка:
CustomWidget.py
import sys
from PyQt5 import QtCore, QtWidgets
from CustomWidget_GUI import Ui_Table_Widget_Insert
class CustomListEntry(QtWidgets.QWidget, Ui_Table_Widget_Insert):
clicked = QtCore.pyqtSignal()
def __init__(self, parent=None):
super(CustomListEntry, self).__init__(parent)
self.setupUi(self)
self.setUpMainUiFunction()
def setUpMainUiFunction(self):
self.pushButton.clicked.connect(self.clicked)
def setlabel(self,label):
self.label.setText(label)
Теперь ее можно использовать в основном файле:
Main.py
import sys
from PyQt5 import QtCore, QtWidgets
from Main_GUI import Ui_MainWindow
from CustomWidget import CustomListEntry
class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
def __init__(self, parent=None):
super(MainWindow, self).__init__(parent)
self.setupUi(self)
self.setUpMainUiFunction()
def setUpMainUiFunction(self):
self.pushButton.clicked.connect(self.addCustomEntry)
self.listWidget.clicked_widget.connect(self.on_clicked_widget)
@QtCore.pyqtSlot()
def addCustomEntry(self):
listEntryNumber = self.listWidget.count()
custom_widget = CustomListEntry()
custom_widget.setlabel("Entry Number= {}".format(listEntryNumber))
self.listWidget.append_widget(custom_widget)
@QtCore.pyqtSlot(int)
def on_clicked_widget(self, row):
print("row number of found item = {}".format(row))
if __name__ == "__main__":
#os.environ["QT_AUTO_SCREEN_SCALE_FACTOR"] = "1"
app = QtWidgets.QApplication(sys.argv)
MainWindow = MainWindow()
MainWindow.show()
sys.exit(app.exec_())
В конце ваша папка должна иметь следующую структуру:
├── clickablelistwidget.py
├── CustomListWidget.py
├── CustomWidget_GUI.py
├── CustomWidget.py
├── Main_GUI.py
└── Main.py
Преимущество этого метода состоит в том, что он отсоединяет классы, что делает его легко повторно используемым,Вам не нужно прибегать к objectName или вставлять QListWidgetItem в виджет, который может быть утомительным, если у нас много типов виджетов.: -.)