Я нашел отличный ресурс здесь для создания 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
Спасибо, что заглянули!