Как отлаживать QComboBox, испускающий несколько «активированных» сигналов - PullRequest
1 голос
/ 26 сентября 2019

Я нашел отличный ресурс здесь для создания QComboBox, который дает отфильтрованный список предлагаемых.Это работает хорошо, за исключением того факта, что сигналы «activated» и «currentIndexChanged» испускаются три раза каждый раз, когда я выбираю предложенную опцию в выпадающем списке.Поведение отличается в зависимости от того, выбран ли параметр мышью или с помощью клавиш со стрелками и кнопки ввода.

У меня вопрос, как мне это отладить?В коде нет смысла ловить и предотвращать излучение первых двух сигналов.Есть ли способ переопределить «активированный» сигнал QComboBox, чтобы попытаться поймать его в действии?Или я должен определить свой собственный сигнал и использовать его вместо этого?

Вот код:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
from PySide2 import QtCore, QtGui, QtWidgets
from PySide2.QtCore import Qt, QSortFilterProxyModel
from PySide2.QtWidgets import QCompleter, QComboBox

class ExtendedComboBox(QComboBox):
    def __init__(self, parent=None):
        super(ExtendedComboBox, self).__init__(parent)

        self.setFocusPolicy(Qt.StrongFocus)
        self.setEditable(True)

        # add a filter model to filter matching items
        self.pFilterModel = QSortFilterProxyModel(self)
        self.pFilterModel.setFilterCaseSensitivity(Qt.CaseInsensitive)
        self.pFilterModel.setSourceModel(self.model())

        # add a completer, which uses the filter model
        self.completer = QtWidgets.QCompleter(self)
        self.completer.setModel(self.pFilterModel)

        # always show all (filtered) completions
        self.completer.setCompletionMode(QCompleter.UnfilteredPopupCompletion)
        self.setCompleter(self.completer)

        # connect signals
        self.lineEdit().textEdited.connect(self.pFilterModel.setFilterFixedString)
        self.completer.activated.connect(self.on_completer_activated)

    # on selection of an item from the completer, select the corresponding item from combobox 
    def on_completer_activated(self, text):
        if text:
            index = self.findText(text)
            self.setCurrentIndex(index)
            # self.activated.emit(self.itemText(index))


    # on model change, update the models of the filter and completer as well 
    def setModel(self, model):
        super(ExtendedComboBox, self).setModel(model)
        self.pFilterModel.setSourceModel(model)
        self.completer.setModel(self.pFilterModel)


    # on model column change, update the model column of the filter and completer as well
    def setModelColumn(self, column):
        self.completer.setCompletionColumn(column)
        self.pFilterModel.setFilterKeyColumn(column)
        super(ExtendedComboBox, self).setModelColumn(column)    

def change_option(text):
    print(text)

if __name__ == "__main__":
    import sys
    from PySide2.QtWidgets import QApplication
    from PySide2.QtCore import QStringListModel

    app = QApplication(sys.argv)

    string_list = ['hola muchachos', 'adios amigos', 'hello world', 'good bye']

    combo = ExtendedComboBox()

    # either fill the standard model of the combobox
    combo.addItems(string_list)
    combo.currentIndexChanged[str].connect(change_option)
    # or use another model
    #combo.setModel(QStringListModel(string_list))

    combo.resize(300, 40)
    combo.show()

    sys.exit(app.exec_())

Вы заметите, если вы запустите код и начнете вводить «привет» втекстовое поле, затем нажмите на предложенный «привет мир», сигнал activated возвращает правильный «привет мир».Хотя если вы начнете вводить «hello», но в этот раз используйте клавиши со стрелками для прокрутки вниз до «hello world», он издаст три раза.

Я пробовал несколько реализаций одной и той же идеи, все с тем жерезультат.Я даже заметил подобное поведение с немодифицированным QComboBox после замены модели на новую.

PySide2 5.6.0a1 Windows 10.0.18362 Build 18362

Спасибо, что заглянули!

Ответы [ 2 ]

1 голос
/ 27 сентября 2019

Я использовал PySide2 5.6.0a1, потому что это тот, который Anaconda устанавливает в средах Python 2.7.@eyllanesc указал, что это ранняя и устаревшая версия, и, вероятно, с ошибками.

Когда я пробовал один и тот же код в среде Python 3.7 с PySide2-5.13.1, все работало как ожидалось.

0 голосов
/ 26 сентября 2019

У меня нет PySide2, но по большей части я думаю, что все, что вам нужно сделать, это заменить мою ссылку PyQt5 на PySide2 - поскольку это почти все, что я сделал, чтобы переключить вашу программу с PySide2 на PyQt5 - что нарядус небольшим количеством реструктуризации и тонкой настройки, которая дала мне следующий функциональный фрагмент кода:

from sys import exit as sysExit

from PyQt5.QtCore import Qt, QSortFilterProxyModel, QStringListModel, pyqtSlot
from PyQt5.QtWidgets import QApplication, QWidget, QCompleter, QComboBox, QCompleter, QHBoxLayout

class ExtendedComboBox(QComboBox):
    def __init__(self):
        QComboBox.__init__(self)

        self.setFocusPolicy(Qt.StrongFocus)
        self.setEditable(True)

        # add a filter model to filter matching items
        self.pFilterModel = QSortFilterProxyModel(self)
        self.pFilterModel.setFilterCaseSensitivity(Qt.CaseInsensitive)
        self.pFilterModel.setSourceModel(self.model())

        # add a completer, which uses the filter model
        self.completer = QCompleter(self)
        self.completer.setModel(self.pFilterModel)

        # always show all (filtered) completions
        self.completer.setCompletionMode(QCompleter.UnfilteredPopupCompletion)
        self.setCompleter(self.completer)

        # connect signals
        self.lineEdit().textEdited.connect(self.pFilterModel.setFilterFixedString)
        self.completer.activated.connect(self.on_completer_activated)

    # on selection of an item from the completer, select the corresponding item from combobox 
    def on_completer_activated(self, text):
        if text:
            index = self.findText(text)
            self.setCurrentIndex(index)
            # self.activated.emit(self.itemText(index))


    # on model change, update the models of the filter and completer as well 
    def setModel(self, model):
        self.setModel(model)
        self.pFilterModel.setSourceModel(model)
        self.completer.setModel(self.pFilterModel)


    # on model column change, update the model column of the filter and completer as well
    def setModelColumn(self, column):
        self.completer.setCompletionColumn(column)
        self.pFilterModel.setFilterKeyColumn(column)
        self.setModelColumn(column)    

class MainApp(QWidget):
    def __init__(self):
        QWidget.__init__(self)

        string_list = ['hola muchachos', 'adios amigos', 'hello world', 'good bye']

        self.combo = ExtendedComboBox()

        # either fill the standard model of the combobox
        self.combo.addItems(string_list)
        self.combo.currentIndexChanged[str].connect(self.change_option)
        # or use another model
        #combo.setModel(QStringListModel(string_list))

        self.resize(300, 100)
        self.combo.resize(300, 50)

        HBox = QHBoxLayout()
        HBox.addWidget(self.combo)

        self.setLayout(HBox)

    @pyqtSlot(str)
    def change_option(self, text):
        print(text)

if __name__ == "__main__":
    MainThred = QApplication([])

    MainGui = MainApp()
    MainGui.show()

    sysExit(MainThred.exec_())

Я думаю, что проблема заключалась в том, что вы пытались использовать сигналы / слоты с функцией не-QObject (иначе)Ваша функция change_option не связана напрямую ни с чем, что наследуется от QObject, поэтому я не уверен, что он делал или не делал, но это только предположение, поскольку все, что я сделал, было помещено в обычную структуру Qt, и она работала просто отлично

...