PyQt5: уведомлять об изменении значения атрибута - PullRequest
0 голосов
/ 05 ноября 2018

Прежде всего, пожалуйста, посмотрите на пример кода, который приведен ниже. Как я могу получить доступ, например, .setDisabled(...) к QPushButton, когда значение атрибута self.markup_points внутри QGraphicsView изменено? Как я могу реализовать это, используя pyqt сигналы или ... использовать singleton ?

class ImageView(QtWidgets.QGraphicsView):
    def __init__(self, parent):
        super(ImageView, self).__init__(parent)
        self.markup_points = []
        ...
        ...

    def set_image(self, pixmap):
        foo()

    def mousePressEvent(self, event):
        foo()
        self.markup_points.append(QtCore.QPointF(bar()))
        super(ImageView, self).mousePressEvent(event)
    ...

    def keyPressEvent(self, event):
        key = event.key()
        modifiers = int(event.modifiers())
        if (modifiers and modifiers & MOD_MASK == modifiers and
                key > 0 and key != QtCore.Qt.Key_Control and key != QtCore.Qt.Key_Meta):
            if key == 88:
                self.remove_point()

    def remove_point(self):
        if len(self.markup_points):
            self.markup_points.pop()
    ...

class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
    def __init__(self):
        super(MainWindow, self).__init__()
        self.setupUi(self)
        ...
        self.imageView = ImageView()
        self.btnLoad.clicked.connect(self._load_combination)
        self.btnSkip.clicked.connect(self._skip_combination)
        self.btnSave.clicked.connect(self._save_objects)
        # qpushbutton that I want to access later
        self.btnRemove.clicked.connect(self.imageView.remove_point)
    ...
    def event_if_something_is_changed_in_image_view(self):
        self.btnRemove.setDisabled(True)

1 Ответ

0 голосов
/ 05 ноября 2018

Почему вы думаете, что синглтон - это решение? Синглтон является анти-паттерном, поэтому его следует избегать, и только в определенных случаях он необходим, кроме того, он не имеет ничего общего с уведомлением об изменениях, поэтому откажитесь от него.

Решение состоит в том, чтобы создать сигнал, который выдается при изменении, и подключить его к слоту, который получает уведомления:

class ImageView(QtWidgets.QGraphicsView):
    markupPointsChanged = QtCore.pyqtSignal(list) # <---

    def __init__(self, parent):
        super(ImageView, self).__init__(parent)
        self.markup_points = []
        # ...

    def mousePressEvent(self, event):
        foo()
        self.append_point(QtCore.QPointF(bar()))
        super(ImageView, self).mousePressEvent(event)

    def keyPressEvent(self, event):
        key = event.key()
        modifiers = int(event.modifiers())
        if (modifiers and modifiers & MOD_MASK == modifiers and
                key > 0 and key not in (QtCore.Qt.Key_Control, QtCore.Qt.Key_Meta)):
            if key == QtCore.Qt.Key_X:
                self.remove_point()

    def append_point(self, p):
        self.markup_points.append(p)
        self.markupPointsChanged.emit(self.markup_points)  # <---

    def remove_point(self):
        if self.markup_points:
            self.markup_points.pop()
        self.markupPointsChanged.emit(self.markup_points) # <---
    # ...

class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
    def __init__(self):
        super(MainWindow, self).__init__()
        self.setupUi(self)
        # ...
        self.imageView = ImageView()
        self.btnLoad.clicked.connect(self._load_combination)
        self.btnSkip.clicked.connect(self._skip_combination)
        self.btnSave.clicked.connect(self._save_objects)
        self.btnRemove.clicked.connect(self.imageView.remove_point)
        self.imageView.markupPointsChanged.connect(self.on_markupPointsChanged) # <---

    @QtCore.pyqtSlot(list)
    def on_markupPointsChanged(self, points):
        print(points)
        self.btnRemove.setDisabled(True)
...