Python - QTableWidget.selectedRanges() 返回选择的多个范围

标签 python qt multiple-columns qtablewidget

我是Python新手,正在开发一个带有表格的GUI。应该可以通过上下文菜单连接行和列。这有效。 我可以使用 ctrl + 鼠标左键单击选择多行。

但是表中是否存在一个或多个跨度,并且我想添加更多跨度,如果选择 nXm 个选择,则 QTableWidget.selectedRanges() 返回 n*m 选择范围。 例如,如果我在一列中选择了 3 行,则有 3 个选择范围。

并且在控制台上写了3次

QTableView::setSpan: single cell span won't be added 

已安装 Python 2.7.5+。

谢谢,再见。

编辑于 2014 年 3 月 17 日

这是我的小程序的完整代码。

程序.py

import sys
from PyQt4 import QtGui, QtCore
from tableGui import Ui_Dialog as Dlg


class MarkDownTableGui(QtGui.QDialog,Dlg):
  def __init__(self):
    QtGui.QDialog.__init__(self)
    self.setupUi(self)
    ColumnCount = None
    ColumnRow = None    
    # Context menu connect cell/reset connected/selected cells
    self.menu = QtGui.QMenu()           
    self.menu.addAction("Connect",self.addSpan)
    self.menu.addAction("Reset",self.resetConnected)     

    self.setContextMenuPolicy(QtCore.Qt.CustomContextMenu)

    self.customContextMenuRequested.connect(self.showContextMenu)            
    self.connect(self.generateButton,QtCore.SIGNAL("clicked()"),self.onGenerate)

  def onGenerate(self):    
    """
      Generate a m x n table
    """
    self.tableWidget.setRowCount(self.RowCount.value())
    self.tableWidget.setColumnCount(self.ColumnCount.value())

  def addSpan(self):    
    selectedElements = self.tableWidget.selectedRanges()       

    for slrange in selectedElements:      
      self.tableWidget.setSpan(slrange.topRow(),slrange.leftColumn(),slrange.rowCount(),slrange.columnCount())  


  def showContextMenu(self,pos):             
    self.menu.exec_(QtGui.QCursor.pos())    

  def resetConnected(self):
    #TODO
    print "TODO"


app = QtGui.QApplication(sys.argv)
dialog = MarkDownTableGui()
dialog.show()
sys.exit(app.exec_())

这是tableGui.py,由pyuic4生成

# -*- coding: utf-8 -*-

# Form implementation generated from reading ui file 'dialog_Table.ui'
#
# Created: Mon Mar 17 08:59:43 2014
#      by: PyQt4 UI code generator 4.10.3
#
# WARNING! All changes made in this file will be lost!

from PyQt4 import QtCore, QtGui

try:
    _fromUtf8 = QtCore.QString.fromUtf8
except AttributeError:
    def _fromUtf8(s):
        return s

try:
    _encoding = QtGui.QApplication.UnicodeUTF8
    def _translate(context, text, disambig):
        return QtGui.QApplication.translate(context, text, disambig, _encoding)
except AttributeError:
    def _translate(context, text, disambig):
        return QtGui.QApplication.translate(context, text, disambig)

class Ui_Dialog(object):
    def setupUi(self, Dialog):
        Dialog.setObjectName(_fromUtf8("Dialog"))
        Dialog.setWindowModality(QtCore.Qt.ApplicationModal)
        Dialog.resize(509, 426)
        Dialog.setModal(True)
        self.buttonBox = QtGui.QDialogButtonBox(Dialog)
        self.buttonBox.setGeometry(QtCore.QRect(140, 370, 341, 32))
        self.buttonBox.setOrientation(QtCore.Qt.Horizontal)
        self.buttonBox.setStandardButtons(QtGui.QDialogButtonBox.Cancel|QtGui.QDialogButtonBox.Ok)
        self.buttonBox.setObjectName(_fromUtf8("buttonBox"))
        self.ColumnCount = QtGui.QSpinBox(Dialog)
        self.ColumnCount.setGeometry(QtCore.QRect(90, 10, 51, 23))
        self.ColumnCount.setMinimum(1)
        self.ColumnCount.setProperty("value", 5)
        self.ColumnCount.setObjectName(_fromUtf8("ColumnCount"))
        self.labelChapterCount = QtGui.QLabel(Dialog)
        self.labelChapterCount.setGeometry(QtCore.QRect(10, 10, 74, 23))
        self.labelChapterCount.setObjectName(_fromUtf8("labelChapterCount"))
        self.RowCount = QtGui.QSpinBox(Dialog)
        self.RowCount.setGeometry(QtCore.QRect(250, 10, 51, 23))
        self.RowCount.setMinimum(1)
        self.RowCount.setProperty("value", 5)
        self.RowCount.setObjectName(_fromUtf8("RowCount"))
        self.labelChapterCount_2 = QtGui.QLabel(Dialog)
        self.labelChapterCount_2.setGeometry(QtCore.QRect(160, 10, 74, 23))
        self.labelChapterCount_2.setObjectName(_fromUtf8("labelChapterCount_2"))
        self.generateButton = QtGui.QPushButton(Dialog)
        self.generateButton.setGeometry(QtCore.QRect(340, 10, 111, 24))
        self.generateButton.setObjectName(_fromUtf8("generateButton"))
        self.tableWidget = QtGui.QTableWidget(Dialog)
        self.tableWidget.setGeometry(QtCore.QRect(0, 40, 491, 321))
        self.tableWidget.setSelectionMode(QtGui.QAbstractItemView.ExtendedSelection)
        self.tableWidget.setRowCount(5)
        self.tableWidget.setColumnCount(5)
        self.tableWidget.setObjectName(_fromUtf8("tableWidget"))

        self.retranslateUi(Dialog)
        QtCore.QObject.connect(self.buttonBox, QtCore.SIGNAL(_fromUtf8("accepted()")), Dialog.accept)
        QtCore.QObject.connect(self.buttonBox, QtCore.SIGNAL(_fromUtf8("rejected()")), Dialog.reject)
        QtCore.QMetaObject.connectSlotsByName(Dialog)

    def retranslateUi(self, Dialog):
        Dialog.setWindowTitle(_translate("Dialog", "Table editor", None))
        self.labelChapterCount.setText(_translate("Dialog", "Column", None))
        self.labelChapterCount_2.setText(_translate("Dialog", "Rows", None))
        self.generateButton.setText(_translate("Dialog", "generate Table", None))

问题

  1. 如果程序正在运行,您可以通过按住鼠标左键来选择单元格
  2. 左键单击打开上下文菜单
  3. 选择连接,创建列/行跨度
  4. 如果我想添加额外的跨度并从点 1 开始,那么我会再次收到消息

    QTableView::setSpan:不会添加单个单元格跨度

问题是为什么 selectedRanges() 返回一个包含 n 个 SelectionRange 的列表,而不是仅返回 SelectionRange。我已阅读 QTableWidget 的文档和 QTableSelectionRange 。但没有任何想法

最佳答案

表格中的跨单元格似乎改变了选择模型构建其选择范围的方式。我不太清楚为什么会这样,但即使这没有发生,也不能消除所有问题。

默认情况下,表格小部件允许进行多项选择,因此您最终可能会得到无法跨越的选择(作为一个整体)。但即使多个选择彼此相邻并形成一个可以跨越的 block ,选择模型总是将它们记录为选择范围列表,而不是将它们全部合并到一个选择中。

因此,为了使其正常工作,我认为您必须在允许设置跨度之前切换到不同的选择模式。最好的选择似乎是这样的:

    self.tableWidget.setSelectionMode(
        QtGui.QAbstractItemView.ContiguousSelection)

它只允许选择单个矩形 block 。您可能需要一个按钮来在不同的选择模式之间切换,并根据需要禁用“连接”菜单项。

这仍然留下了在设置第一个范围后处理选择范围列表的问题。要解决此问题,您可以执行以下操作:

    def addSpan(self): 
        selection = list(self.tableWidget.selectionModel().selection())
        selection.sort()
        top = selection[0].top()
        left = selection[0].left()
        bottom = selection[-1].bottom()
        right = selection[-1].right()
        self.tableWidget.setSpan(
            top, left, bottom - top + 1, right - left + 1)

我不确定对选择进行排序是否确实有必要,但它确实保证选择范围始终处于正确的顺序。

关于Python - QTableWidget.selectedRanges() 返回选择的多个范围,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22382736/

相关文章:

python - 如何在 pandas 中合并和格式化包含 int 数据类型的多个列?

python - 运行 Python 脚本时出现名称错误

c++ - 连接 QDomNode 和 QStandardItem

c++ - 使用 C++ 对象 Q_PROPERTY 绑定(bind)复选框 'checked' 属性

c++ - 向 Q_Object 构造函数添加参数

html - Bootstrap 4 textarea 填充剩余的第二列高度

bash - 什么时候以及为什么我需要一个 's' 标志来按列排序

python - sqlite3.OperationalError : no such table: main. 源

python - 在 sqlalchemy 中动态生成过滤器,混合使用 and/or 和 not/like

python - 有没有办法将 mysql 列值保存在 python 变量中?