为了账号安全,请及时绑定邮箱和手机立即绑定

如何在 QCalendarWidget 中禁用其他月份

如何在 QCalendarWidget 中禁用其他月份

梵蒂冈之花 2021-09-14 10:47:06
我的目标是禁用用户可以单击 aQCalendarWidget中非当前月份的日期,因此我将小部件子类化以执行此操作。到目前为止,我可以让那些日子根本不渲染任何文本(很棒)。这是代码:class QCustomCalendar(QCalendarWidget):    """Create my own Calendar with my own options."""    def __init__(self, parent=None):        """Initializing functions"""        QCalendarWidget.__init__(self, parent)        self.setEnabled(True)        self.setGeometry(QRect(0, 0, 320, 250))        self.setGridVisible(False)        self.setHorizontalHeaderFormat(QCalendarWidget.SingleLetterDayNames)        self.setVerticalHeaderFormat(QCalendarWidget.NoVerticalHeader)        self.setNavigationBarVisible(True)        self.setDateEditEnabled(True)        self.setObjectName("calendarWidget")    def paintCell(self, painter, rect, date):        """Sub-class this and repaint the cells"""        # Render only this-month days        month = "{0}-{1}".format(str(self.yearShown()), str(self.monthShown()).zfill(2))        day = str(date.toPython())        if not day.startswith(month):            return        QCalendarWidget.paintCell(self, painter, rect, date)但是,如果我单击未渲染的一天,它仍然会计数并触发clicked事件。示例:我对一个红色方块进行了 photoshop,点击它,它会选择 6 月 4 日(即使我们在屏幕截图中是在 5 月)。我如何禁用那些日子不被选择?我尝试setDateRange了currentPageChanged事件,但它没有按预期工作:def __init__(self, parent=None):    # some code    self.currentPageChanged.connect(self.store_current_month)    self.clicked.connect(self.calendar_itemchosen)def store_current_month(self):    self.CURRENT_MONTH = "{0}-{1}".format(str(self.yearShown()), str(self.monthShown()).zfill(2))def calendar_itemchosen(self):    day = str(self.selectedDate().toPython())    print(day)    if day.startswith(self.CURRENT_MONTH):        selection = self.selectedDate()        # some code        self.close()用这个代码点击那个红色方块的结果是:2018-062018-06-04所以我猜currentPageChanged当您选择另一个月份的日期时,Qt 首先会触发该事件。setDateRange将不起作用,因为如果我添加它以将选择限制在本月,那么日历顶部的“转到下个月或上个月”按钮将不起作用,我需要用户能够更改月份. 我只是不想让日历显示不属于这个月页面的日子。
查看完整描述

1 回答

?
倚天杖

TA贡献1828条经验 获得超3个赞

一种解决方案是过滤 QCalendarWidget 内部具有的 QTableView 的 mousePressEvent 事件。为此,我们使用事件过滤器:


from PyQt5 import QtCore, QtWidgets


class CalendarWidget(QtWidgets.QCalendarWidget):

    def __init__(self, parent=None):

        super(CalendarWidget, self).__init__(parent, gridVisible=False,

            horizontalHeaderFormat=QtWidgets.QCalendarWidget.SingleLetterDayNames,

            verticalHeaderFormat=QtWidgets.QCalendarWidget.NoVerticalHeader,

            navigationBarVisible=True,

            dateEditEnabled=True)       

        self.setEnabled(True)

        self.setGeometry(QtCore.QRect(0, 0, 320, 250))

        self.clicked.connect(print)


        self.table_view = self.findChild(QtWidgets.QTableView, "qt_calendar_calendarview")

        self.table_view.viewport().installEventFilter(self)

        self.setFirstDayOfWeek(QtCore.Qt.Monday)


    def referenceDate(self):

        refDay = 1

        while refDay <= 31:

            refDate = QtCore.QDate(self.yearShown(), self.monthShown(), refDay)

            if refDate.isValid(): return refDate

            refDay += 1

        return QtCore.QDate()


    def columnForDayOfWeek(self, day):

        m_firstColumn = 1 if self.verticalHeaderFormat() != QtWidgets.QCalendarWidget.NoVerticalHeader else 0

        if day < 1 or day > 7: return -1

        column = day - int(self.firstDayOfWeek())

        if column < 0:

            column += 7

        return column + m_firstColumn


    def columnForFirstOfMonth(self, date):

        return (self.columnForDayOfWeek(date.dayOfWeek()) - (date.day() % 7) + 8) % 7


    def dateForCell(self, row, column):

        m_firstRow = 1 if self.horizontalHeaderFormat() != QtWidgets.QCalendarWidget.NoHorizontalHeader else 0

        m_firstColumn = 1 if self.verticalHeaderFormat() != QtWidgets.QCalendarWidget.NoVerticalHeader else 0

        rowCount = 6

        columnCount = 7

        if row < m_firstRow or row > (m_firstRow + rowCount -1) or column < m_firstColumn or column > (m_firstColumn + columnCount -1):

            return QtCore.QDate()

        refDate = self.referenceDate()

        if not refDate.isValid():

            return QtCore.QDate()

        columnForFirstOfShownMonth = self.columnForFirstOfMonth(refDate)

        if (columnForFirstOfShownMonth - m_firstColumn) < 1:

            row -= 1

        requestedDay = 7*(row - m_firstRow) +  column  - columnForFirstOfShownMonth - refDate.day() + 1

        return refDate.addDays(requestedDay)


    def eventFilter(self, obj, event):

        if obj is self.table_view.viewport() and event.type() == QtCore.QEvent.MouseButtonPress:    

            ix = self.table_view.indexAt(event.pos())

            date = self.dateForCell(ix.row(), ix.column())

            d_start = QtCore.QDate(self.yearShown(), self.monthShown(), 1)

            d_end = QtCore.QDate(self.yearShown(), self.monthShown(), d_start.daysInMonth())

            if d_start > date or date > d_end:

                return True

        return super(CalendarWidget, self).eventFilter(obj, event)


    def paintCell(self, painter, rect, date):

        d_start = QtCore.QDate(self.yearShown(), self.monthShown(), 1)

        d_end = QtCore.QDate(self.yearShown(), self.monthShown(), d_start.daysInMonth())

        if d_start <= date <= d_end:

            super(CalendarWidget, self).paintCell(painter, rect, date)


if __name__ == '__main__':

    import sys

    app = QtWidgets.QApplication(sys.argv)

    w = CalendarWidget()

    w.show()

    sys.exit(app.exec_())



查看完整回答
反对 回复 2021-09-14
  • 1 回答
  • 0 关注
  • 393 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信