Kivy:标签文本在 for 循环期间不更新

2022-01-15 00:00:00 python kivy for-loop label

问题描述

我在 for 循环期间尝试更新标签文本时遇到问题.有类似的条目(例如:Update properties of a kivy widget while运行代码),但它们似乎并不完全适合我的问题(或者我错过了重点......).我运行以下代码:

I have an issue when I try to update a label text during a for-loop. There are similar entries (e.g.: Update properties of a kivy widget while running code) but they do not seem to fit my issue exactly (or I missed the point…). I run following code:

*.py:

from kivy.app import App
from kivy.uix.boxlayout import BoxLayout
from kivy.properties import StringProperty
#from time import sleep

class MyBox(BoxLayout):
    tobeupd = StringProperty()

    def __init__(self,*args,**kwargs):
        super(MyBox,self).__init__(*args,**kwargs)
        self.tobeupd = '#'

    def upd_ltxt(self):
        for i in range(1,10):
            self.tobeupd = str(i)
            print(self.tobeupd)
            input('Write something: ')  # new line, see edit below
            #sleep(0.5) 

class updApp(App):
    def build(self):
        return MyBox()

if __name__ == '__main__':
    updApp().run() 

*.kv

<MyBox>:
    orientation: 'horizontal'
    cols: 2
    Label:
        text: root.tobeupd
    Button:
        text: 'Start Update'
        on_release: root.upd_ltxt()

虽然打印"语句会定期更新 shell,但标签文本仅在 for 循环结束时更新.谁能向我解释为什么 Kivy 会这样工作以及我如何克服这个问题?

Whereas the ‘print’ statement updates the shell regularly, the label text updates at the end of the for-loop only. Can anyone explain to me why Kivy works this way and how I can overcome this problem?

根据 PM2Ring 和 Gugas,我更改了代码以避免睡眠功能.如果我要求用户在循环继续之前输入一些内容,问题仍然存在.这些值在 shell 中更新,但不在标签上.

According to PM2Ring and Gugas, I changed the code in order to avoid the sleep-function. The problem remains if I ask the user to enter something before the loop can be continued. The values are updated in the shell but not on the label.


解决方案

您可以为此使用 threading.
当您在 kivy 中执行循环或等待输入时,主线程正在等待,并且不会在应用程序上更新任何内容.threading 会阻止这种情况.
使用 threading 在主线程之外创建另一个线程.
示例:

You can use threading for this.
When you do a loop or wait for an input in kivy, the main thread is waiting, and nothing will update on the app. threading will prevent that.
Use threading to make another thread besides the main thread.
Example:

from kivy.app import App
from kivy.uix.boxlayout import BoxLayout
from kivy.properties import StringProperty
from kivy.lang import Builder
import threading

Builder.load_string('''

<MyBox>:
    orientation: 'horizontal'
    cols: 2
    Label:
        text: root.tobeupd
    Button:
        text: 'Start Update'
        on_release: root.upd_ltxt()

''')

class MyBox(BoxLayout):
    tobeupd = StringProperty()

    def __init__(self,*args,**kwargs):
        super(MyBox,self).__init__(*args,**kwargs)
        self.tobeupd = '#'

    def upd_ltxt(self):
        threading.Thread(target=self.update_label).start()

    def update_label(self):
        for i in range(1,10):
            print(self.tobeupd)
            self.tobeupd = str(i)
            input('Write something: ')  # new line, see edit below



class updApp(App):
    def build(self):
        return MyBox()

if __name__ == '__main__':
    updApp().run()

现在值得一提的是,即使第一个尚未完成,您也可以继续按下按钮并启动线程.这可能是不受欢迎的行为.
这可以通过在线程开始时禁用按钮并在结束时再次启用它来防止.

Now its worth mentioning that you can keep pushing the button and start threads, even if the first did not finish yet. This might be an unwanted behavior.
This can be prevented by disabling the button in the beginning of the thread, and enabling it again at the end.

在kv中给按钮一个id:

Give the button an id in kv:

Button:
    id: updatebutton
    text: 'Start Update'
    on_release: root.upd_ltxt()

在线程中这样做:

def update_label(self):

    self.ids.updatebutton.disabled = True

    for i in range(1,10):
        self.tobeupd = str(i)
        input('Write something: ')

    self.ids.updatebutton.disabled = False

相关文章