Python 来自SQL模型的Qt自定义多选QComboBox
我想要一个下拉类别选择框,允许用户单击多个选择,最好连接到SQL查询。我并不特别喜欢组合框,每次点击都消失是没有帮助的,任何东西都可以完成任务 目前,我已经基本上把jerry装配在一起,技术上允许多个选择,但它只是基于随机的,如果鼠标被拖动到它上面或没有Python 来自SQL模型的Qt自定义多选QComboBox,python,pyqt,pyqt5,qcombobox,qsqlquerymodel,Python,Pyqt,Pyqt5,Qcombobox,Qsqlquerymodel,我想要一个下拉类别选择框,允许用户单击多个选择,最好连接到SQL查询。我并不特别喜欢组合框,每次点击都消失是没有帮助的,任何东西都可以完成任务 目前,我已经基本上把jerry装配在一起,技术上允许多个选择,但它只是基于随机的,如果鼠标被拖动到它上面或没有 self.catbx=QComboBox() ... self.catq=QtSql.QSqlQuery(conn) self.catq.exec("SELECT name FROM categories") self.catmo=QtSql.
self.catbx=QComboBox()
...
self.catq=QtSql.QSqlQuery(conn)
self.catq.exec("SELECT name FROM categories")
self.catmo=QtSql.QSqlQueryModel()
self.catmo.setQuery(self.catq)
self.catbx.setModel(self.catmo)
...
self.catview=QListView()
self.catview.setModel(self.catmo)
self.catbx.setView(self.catview)
self.catview.setSelectionMode(QAbstractItemView.MultiSelection)
希望这足够清楚,有人能帮忙!: 基本上,项目是不可选择的,因为这是触发关闭弹出窗口的事件,所以解决方案是在模型中消除该标志,就像我在 另一方面,可检查的选项在QSqlQueryModel中默认不存在,因此我们必须基于另一个实现它 最后,创建一个从QComboBox继承的类,我们覆盖hidePopup方法以发出一个发送所选项目的信号 更新: 如果要在按下项目的任何部分时也进行标记,则必须创建一个委托并覆盖editorEvent方法,以便它处理MouseButtonRelease事件。但这带来了一个小问题:要打开弹出窗口,你必须按下显示的项目,这样它就会打开
from PyQt5 import QtCore, QtGui, QtWidgets, QtSql
class CheckSqlQueryModel(QtSql.QSqlQueryModel):
def __init__(self, *args, **kwargs):
QtSql.QSqlQueryModel.__init__(self, *args, **kwargs)
self.checks = {}
def checkState(self, pindex):
if pindex not in self.checks.keys():
self.checks[pindex] = QtCore.Qt.Unchecked
return self.checks[pindex]
def data(self, index, role=QtCore.Qt.DisplayRole):
if role == QtCore.Qt.CheckStateRole and index.isValid():
return self.checkState(QtCore.QPersistentModelIndex(index))
return QtSql.QSqlQueryModel.data(self, index, role)
def setData(self, index, value, role=QtCore.Qt.EditRole):
if role == QtCore.Qt.CheckStateRole and index.isValid():
self.checks[QtCore.QPersistentModelIndex(index)] = value
return True
return QtSql.QSqlQueryModel(self, index, value, role)
def flags(self, index):
fl = QtSql.QSqlQueryModel.flags(self, index) & ~QtCore.Qt.ItemIsSelectable
fl |= QtCore.Qt.ItemIsEditable | QtCore.Qt.ItemIsUserCheckable
return fl
class CheckComboBox(QtWidgets.QComboBox):
selectedChanged = QtCore.pyqtSignal(list)
def hidePopup(self):
results = []
for i in range(self.count()):
if self.itemData(i, QtCore.Qt.CheckStateRole) == QtCore.Qt.Checked:
results.append(self.itemText(i))
self.selectedChanged.emit(results)
QtWidgets.QComboBox.hidePopup(self)
class CheckDelegate(QtWidgets.QStyledItemDelegate):
def editorEvent(self, event, model, option, index):
if event.type() == QtCore.QEvent.MouseButtonRelease:
val = index.data(QtCore.Qt.CheckStateRole)
new_val = QtCore.Qt.Checked if val == QtCore.Qt.Unchecked else QtCore.Qt.Unchecked
model.setData(index, new_val, QtCore.Qt.CheckStateRole)
return True
return QtWidgets.QStyledItemDelegate.editorEvent(self, event, model, option, index)
class Widget(QtWidgets.QWidget):
def __init__(self, *args, **kwargs):
QtWidgets.QWidget.__init__(self, *args, **kwargs)
lay = QtWidgets.QVBoxLayout(self)
combo = CheckComboBox()
combo.setView(QtWidgets.QListView())
combo.setItemDelegate(CheckDelegate(combo))
model = CheckSqlQueryModel()
model.setQuery("SELECT name FROM categories")
combo.setModel(model)
self.lw = QtWidgets.QListWidget()
combo.selectedChanged.connect(self.on_selectedChanged)
lay.addWidget(combo)
lay.addWidget(self.lw)
def on_selectedChanged(self, items):
self.lw.clear()
self.lw.addItems(items)
def createConnection():
db = QtSql.QSqlDatabase.addDatabase("QSQLITE")
db.setDatabaseName(":memory:")
if not db.open():
QtWidgets.QMessageBox.critical(None, "Cannot open database",
"Unable to establish a database connection.\n"
"This example needs SQLite support. Please read "
"the Qt SQL driver documentation for information how "
"to build it.\n\n"
"Click Cancel to exit.", QMessageBox.Cancel)
return False
query = QtSql.QSqlQuery()
query.exec_("create table categories (id int primary key, name varchar(20))");
for i in range(1, 10):
query.exec_("insert into categories values({i}, 'categories-{i}')".format(i=i));
return True
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
if not createConnection():
sys.exit(-1)
w = Widget()
w.show()
sys.exit(app.exec_())
您想要一个带有复选框的组合框吗?复选框列表中有一个按钮,该按钮控制列表在该按钮下的显示/隐藏@DmitrySazonov只要不是每次点击都消失就行了!甚至只是一个组合框,其项目是checkable@user26472你想让弹出窗口关闭什么事件?@eyllanesc不完全确定你在问什么,但我正在尝试的是一个弹出类别选择菜单,它将附加到我的应用程序主表的过滤器上。有相当多的类别可供选择多个选项,因此我想要qcombobox的下拉功能,其中有一个窗口,只有当用户在窗口外单击时,该窗口才会消失,并在该窗口上过滤数据。明白了吗?谢谢!!是否有任何方法可以单击标签名称以同时计入复选框?我试着摆弄旗帜之类的东西,但运气不好。除此之外,尽管它现在可以正常工作了,谢谢: