当我在tkinter中一次又一次按下启动和暂停按钮时,倒数计时器的速度增加

时间:2019-05-23 02:30:24

标签: python windows tkinter

我有一个倒计时脚本,当我一次又一次按下开始和暂停按钮时,秒数会快速更新。

起初,我认为在开始按钮和暂停按钮之间快速单击会加快秒速,因此我添加了time.sleep(1)来延迟在无效按钮之间的单击。

如何解决此问题?

我的算法正确吗?还是还有其他更好的方法来倒计时?如果是,该怎么办?

代码

from tkinter import *

PAUSE = False
HOUR, MINUTE, SECOND = 0, 0, 0


def start():
    '''Command for START button'''

    global PAUSE

    PAUSE = False
    start_button['state'] = 'disabled'
    pause_button['state'] = 'normal'
    reset_button['state'] = 'normal'
    # time.sleep(1) to delay time between clicks of pause and start buttons
    Counter()


def pause():
    '''Command for PAUSE button'''

    global PAUSE

    PAUSE = True
    start_button['state'] = 'normal'
    pause_button['state'] = 'disabled'


def reset():
    '''Command for RESET button'''

    global HOUR, MINUTE, SECOND, PAUSE

    PAUSE = True
    start_button['state'] = 'normal'
    pause_button['state'] = 'disabled'
    reset_button['state'] = 'disabled'
    Time['text'] = '00:00:00'

    HOUR, MINUTE, SECOND = 0, 0, 0


def Counter():
    '''Updating hour, minute and seconds'''

    global HOUR, MINUTE, SECOND

    if PAUSE is False:
        if SECOND == 59:
            if MINUTE == SECOND == 59:
                HOUR += 1

            if MINUTE == 59:
                MINUTE = 0

            else:
                MINUTE += 1

            SECOND = -1

        SECOND += 1

        Time.config(text='{}:{}:{}'.format(str(HOUR).zfill(2), str(MINUTE).zfill(2), str(SECOND).zfill(2)))
        root.after(1000, Counter)


root = Tk()
root.title('COUNTER')

width, height = 342, 108
pos_x = root.winfo_screenwidth() // 2 - width // 2
pos_y = root.winfo_screenheight() // 2 - height // 2

root.geometry('{}x{}+{}+{}'.format(width, height, pos_x, pos_y))

Time = Label(root, fg='Black', text='00:00:00', font=("Helvetica", 40))
Time.pack(side='bottom')

start_button = Button(root, text='START', font=("Arial", 16), fg='black', width=8, command=start)
start_button.pack(side='left')

pause_button = Button(root, text='PAUSE', font=("Arial", 16), fg='black', width=8, state='disabled', command=pause)
pause_button.pack(side='left')

reset_button = Button(root, text='RESET', font=("Arial", 16), fg='black', width=10, state='disabled', command=reset)
reset_button.pack(side='left', fill='both')

root.mainloop()

2 个答案:

答案 0 :(得分:1)

您的Counter函数每次单击都会调用root.after(1000,Counter),并在单击后立即修改Time标签。当您足够快地单击按钮时,您可以安排多个root.after并累加秒数。

要修改当前脚本,您可以跟踪当前操作,并调用root.after_cancel暂停操作。

from tkinter import *

PAUSE = False
HOUR, MINUTE, SECOND = 0, 0, 0
action = None #keep track on current action and also avoid click spam


def start():
    '''Command for START button'''

    global PAUSE, action

    PAUSE = False
    ...
    if not action:
        Counter()


def pause():
    '''Command for PAUSE button'''

    global PAUSE, action

    ...
    if action:
        root.after_cancel(action)
        action = None


def reset():
    '''Command for RESET button'''

    global HOUR, MINUTE, SECOND, PAUSE, action

    ...

    HOUR, MINUTE, SECOND = 0, 0, 0
    action = None


def Counter():
    '''Updating hour, minute and seconds'''

    global HOUR, MINUTE, SECOND, action

    if PAUSE is False:
        if SECOND == 59:
            if MINUTE == SECOND == 59:
                HOUR += 1

            if MINUTE == 59:
                MINUTE = 0

            else:
                MINUTE += 1

            SECOND = -1

        SECOND += 1

        Time.config(text='{}:{}:{}'.format(str(HOUR).zfill(2), str(MINUTE).zfill(2), str(SECOND).zfill(2)))
        action = root.after(1000, Counter)

root = Tk()
...
start_button = Button(root, text='START', font=("Arial", 16), fg='black', width=8, command=lambda: root.after(1000,start)) #to avoid the instant second increment

答案 1 :(得分:1)

  

起初,我认为快速单击开始按钮和暂停按钮之间的速度会加快秒数,因此我添加了time.sleep(1)来延迟无效按钮之间的单击。

问题在于,由于.aftert()方法调用自身之前有一个延迟,因此多次单击开始按钮并创建.after(1000, counter)的多个循环。意味着一次发生了许多.after()循环。要解决此问题,您只需要进行修改,这样就在出现.after()循环的情况下无法按下开始按钮。

为此,您必须更改三个功能

  

counter()函数

我更改了代码,因此如果计数器正在运行,则start_button将被禁用,并且当计数器停止运行时,其状态将恢复为正常

def Counter():
    '''Updating hour, minute and seconds'''
    global HOUR, MINUTE, SECOND

    if PAUSE is False:
        start_button['state'] = 'disabled' # setting it to disabled if the counter is running
        if SECOND == 59:
            if MINUTE == SECOND == 59:
                HOUR += 1

            if MINUTE == 59:
                MINUTE = 0

            else:
                MINUTE += 1

            SECOND = -1

        SECOND += 1

        Time.config(text='{}:{}:{}'.format(str(HOUR).zfill(2), str(MINUTE).zfill(2), str(SECOND).zfill(2)))
        root.after(1000, Counter)
    else:
        start_button['state'] = 'normal' # setting it to normal if the counter stops
  

pause()reset()函数

您必须删除将start_button的状态设置为正常的功能,因为这将允许在上一次计数器循环结束之前按下start_button

def pause():
    '''Command for PAUSE button'''

    global PAUSE

    PAUSE = True
    pause_button['state'] = 'disabled'

def reset():
    '''Command for RESET button'''

    global HOUR, MINUTE, SECOND, PAUSE

    PAUSE = True
    pause_button['state'] = 'disabled'
    reset_button['state'] = 'disabled'
    Time['text'] = '00:00:00'

    HOUR, MINUTE, SECOND = 0, 0, 0

更新这些功能将消除此错误。 希望这是您正在寻找的答案:)