尝试给日期单元格上色时遇到的问题

1 投票
1 回答
3575 浏览
提问于 2025-04-17 05:45

我在使用QT Designer创建的日历控件时遇到了一些问题,想在选中日期后改变单元格的颜色,然后通过pyuic 4转换成Python代码。

到目前为止,我看到了一些关于如何重新上色表格或树形控件单元格的类似问题——不过那些例子都是在代码中实例化之前扩展了QCalendarWidget或树形控件类,而我使用的是QT Designer放置的日历控件,经过pyuic转换后在生成的Python脚本中实例化。

这是我尝试使用QCalendarWidget的paintCell函数来改变日期选择颜色的主窗口文件示例:

import os, sys

from PyQt4 import QtCore, QtGui

from calanderTestWindow import Ui_calanderTestWindow

class Main(QtGui.QMainWindow):
    def __init__(self):
        QtGui.QMainWindow.__init__(self)

        self.ui = Ui_calanderTestWindow()
        self.ui.setupUi(self)
        self.conncectSignals()

    def conncectSignals(self):
        QtCore.QObject.connect(self.ui.testCalander, QtCore.SIGNAL('selectionChanged()'), self.clickDate)

    def clickDate(self):
        painter = QtGui.QPainter()
        painter.setPen(QtGui.QColor(0,255,255))
        date = self.ui.testCalander.selectedDate()
        cellRect = QtCore.QRect(0,0,10,10)
        self.ui.testCalander.paintCell(painter, cellRect, date)


if __name__ == "__main__":
    app = QtGui.QApplication(sys.argv)
    window = Main()
    window.show()
    sys.exit(app.exec_())

这是经过puic转换的Qt Designer脚本:

from PyQt4 import QtCore, QtGui

try:
    _fromUtf8 = QtCore.QString.fromUtf8
except AttributeError:
    _fromUtf8 = lambda s: s

class Ui_calanderTestWindow(object):
    def setupUi(self, calanderTestWindow):
        calanderTestWindow.setObjectName(_fromUtf8("calanderTestWindow"))
        calanderTestWindow.resize(262, 203)
        calanderTestWindow.setWindowTitle(QtGui.QApplication.translate("calanderTestWindow", "Calendar Test Window", None, QtGui.QApplication.UnicodeUTF8))
        self.centralwidget = QtGui.QWidget(calanderTestWindow)
        self.centralwidget.setObjectName(_fromUtf8("centralwidget"))
        self.testCalander = QtGui.QCalendarWidget(self.centralwidget)
        self.testCalander.setGeometry(QtCore.QRect(0, 0, 256, 155))
        self.testCalander.setGridVisible(True)
        self.testCalander.setVerticalHeaderFormat(QtGui.QCalendarWidget.NoVerticalHeader)
        self.testCalander.setNavigationBarVisible(True)
        self.testCalander.setObjectName(_fromUtf8("testCalander"))
        calanderTestWindow.setCentralWidget(self.centralwidget)
        self.menubar = QtGui.QMenuBar(calanderTestWindow)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 262, 21))
        self.menubar.setObjectName(_fromUtf8("menubar"))
        calanderTestWindow.setMenuBar(self.menubar)
        self.statusbar = QtGui.QStatusBar(calanderTestWindow)
        self.statusbar.setObjectName(_fromUtf8("statusbar"))
        calanderTestWindow.setStatusBar(self.statusbar)

        self.retranslateUi(calanderTestWindow)
        QtCore.QMetaObject.connectSlotsByName(calanderTestWindow)

    def retranslateUi(self, calanderTestWindow):
        pass

当我运行这个时,出现了一些日志信息,基本上告诉我出了一些问题:

QPainter::setPen: Painter not active
QPainter::save: Painter not active
QPainter::setClipRect: Painter not active
QPainter::brushOrigin: Painter not active
QPainter::setBrushOrigin: Painter not active
QPainter::setBrushOrigin: Painter not active
QPainter::setPen: Painter not active
QPainter::pen: Painter not active
QPainter::save: Painter not active
QPainter::setBackgroundMode: Painter not active
QPainter::setBrush: Painter not active
QPainter::setBrushOrigin: Painter not active
QPainter::setPen: Painter not active
QPainter::drawRects: Painter not active
QPainter::drawRects: Painter not active
QPainter::drawRects: Painter not active
QPainter::drawRects: Painter not active
QPainter::restore: Unbalanced save/restore
QPainter::restore: Unbalanced save/restore

我算是一个初级程序员(或者更低一点)——我对Python有一定的经验,也对QT在Autodesk Maya中的应用有一点了解,并且有技术艺术的背景,但可能在面向对象编程的核心原则方面经验不足。不过我非常愿意学习。

1 个回答

2

我不知道你是否已经找到这个问题的答案,但我来试试。这个日历小工具其实并不是很擅长显示已选择的日期,更多的是用来选择日期。我觉得它还在寻找当前使用的绘图类。

不过,你可以重新实现 QCalendarWidget,并重写 paintCell 方法,这样就能显示你选择的日期了。

from PyQt4 import QtCore, QtGui

class dateCalendar(QtGui.QCalendarWidget)
    def __init__(self, parent = None):
        super(calendar, self).__init__(parent)
        self.color = QtGui.QColor(self.palette().color(QtGui.QPalette.Highlight))
        self.color.setAlpha(150)
        #self.selectionChanged.connect(self.updateCells)
        self.dateList = []

    def paintCell(self, painter, rect, date):
        #calling original paintCell to draw the actual calendar
        QtGui.QCalendarWidget.paintCell(self, painter, rect, date)

        #highlight a particular date
        if date in self.dateList:
            painter.fillRect(rect, self.color)

    def selectDates(self, qdatesList):
        self.dateList = qdatesList
        #this redraws the calendar with your updated date list
        self.updateCells()

不过现在你需要手动把这些代码写进你的控件里,而不是在 Qt Designer 里使用它(除非你想把它做成一个插件)。

希望这能帮到你,如果你还没解决这个问题的话。

class widget(QtGui.QWidget):
    def __init__(self, parent = None):
        super(empty, self).__init__(parent)
        self.setGeometry(300, 300, 280, 170)

        #no layout
        self.cal = calendar(self)
        self.but = QtGui.QPushButton("Push", self)
        self.but.clicked.connect(self.addDate)

    def addDate(self):
        self.cal.selectDates([QtCore.QDate(2012,10,8), QtCore.QDate(2012,10,5)])

撰写回答