动态创建的 kivy 按钮立即运行 on_press 和 on_release

Posted

技术标签:

【中文标题】动态创建的 kivy 按钮立即运行 on_press 和 on_release【英文标题】:Dynamically created kivy buttons run on_press and on_release immediately 【发布时间】:2014-02-16 03:54:53 【问题描述】:

为什么我的 kivy 按钮在 python 中创建的那一刻就会被按下?

所以,让我先说我知道这里似乎有这个问题的答案:

on_press in Kivy keeps running at start up instead

...但是,没有工作示例。我试图复制那里的示例以理解答案,但缺乏填补示例中缺少的内容的经验。

因此,对于查看其他答案的人来说,这可能是一个简单的答案,可以在这里应用它,并用更简单的英语为菜鸟解释。

这是一个关于该问题的小型工作示例:

import kivy
kivy.require('1.7.2') # replace with your current kivy version !

from kivy.app import App
from kivy.uix.screenmanager import ScreenManager, Screen
from kivy.properties import ObjectProperty
from kivy.uix.button import Button
from kivy.uix.gridlayout import GridLayout

i = ['some', 'words']

class HomeScreen(Screen):
    grid_l = ObjectProperty(None)
    top_lbl = ObjectProperty(None)

    def search_btn_pressed(self):
        grid = self.grid_l
        grid.bind(minimum_height=grid.setter('height'),
                     minimum_width=grid.setter('width'))

        for result in i:

                btn1 = Button(size_hint=(1, None))
                btn1.text = '%r' % result
                btn1.bind(on_release=self.btn1_pressed(result))

                btn2 = Button(size_hint=(1, None))
                btn2.text = 'Remove result buttons'
                btn2.bind(on_release=self.btn2_pressed)

                grid.add_widget(btn1)
                grid.add_widget(btn2)

    def btn1_pressed(self, result, *args):
        new_text = result
        self.top_lbl.text = new_text

    def btn2_pressed(self, *args):
        self.grid_l.clear_widgets()
        #pass

class buttons_pressedApp(App):

    def build(self):

        return HomeScreen()

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

还有kv文件:

#:kivy 1.7.2

<HomeScreen>:
    scroll_view: scrollviewID
    top_lbl: lblID
    grid_l: gridlayoutID
    AnchorLayout:
        size_hint: 1, .1   
        pos_hint: 'x': 0, 'y': .9
        anchor_x: 'center'
        anchor_y: 'center'
        Label:
            id: lblID
            text: 'Button Tester'
    Button:
        size_hint: 1, .1   
        pos_hint: 'x': 0, 'y': .8
        text: 'Add theoretical search results'
        on_release: root.search_btn_pressed()
    ScrollView:
        id: scrollviewID
        orientation: 'vertical'
        pos_hint: 'x': 0, 'y': 0
        size_hint: 1, .8
        bar_width: '8dp'
        GridLayout:
            id: gridlayoutID
            cols: 1
            size_hint: 1, None
            row_default_height: 40
            row_force_default: False

正如您在运行它时所看到的,在 kivy 中创建的第一个按钮非常有效。但是,您也会注意到顶部的标签立即发生了变化,表明第一个动态创建的按钮已经执行了 on_release 函数。

您无法判断第二个动态创建的按钮“删除结果按钮”已被执行,因为它在仍然没有按钮时删除了所有按钮。但是,很明显,当您再次按下“添加理论搜索结果”按钮时,会立即执行“删除结果按钮”按钮。它应该再添加两个按钮,但似乎什么也没发生。这是因为“删除结果按钮”按钮正在删除前两个按钮,然后它们立即被替换。

然后,当然,这两个按钮似乎都没有做任何事情。

给出类似问题的人应该很容易解决这个例子!

提前致谢。

编辑:

我改变了 btn2 以反映恶劣的回答。完美运行。但是,正如他所指出的,当我对 btn1 执行相同操作时,一些默认参数被传递到该方法中并且它给出了错误。因此,我像以前一样将 btn1 与括号和“结果”放在其中作为参数。当然,这会立即运行,并且不会返回任何要绑定的内容(如恶意解释)。我希望能够传入示例中已经定义的“结果”,但自然不会立即运行它。抱歉……我第一次应该这样写。

编辑2,答案:

为了反映最后的评论,我只想再次发布整个工作示例,并附上答案。

import kivy
kivy.require('1.7.2') # replace with your current kivy version !

from kivy.app import App
from kivy.uix.screenmanager import ScreenManager, Screen
from kivy.properties import ObjectProperty
from kivy.uix.button import Button
from kivy.uix.gridlayout import GridLayout
from functools import partial

i = ['some', 'words']

class HomeScreen(Screen):
    grid_l = ObjectProperty(None)
    top_lbl = ObjectProperty(None)

    def search_btn_pressed(self):
        grid = self.grid_l
        grid.bind(minimum_height=grid.setter('height'),
                     minimum_width=grid.setter('width'))

        for result in i:

                btn1 = Button(size_hint=(1, None))
                btn1.text = '%r' % result
                btn1.bind(on_release=partial(self.btn1_pressed, result))

                btn2 = Button(size_hint=(1, None))
                btn2.text = 'Remove result buttons'
                btn2.bind(on_release=self.btn2_pressed)

                grid.add_widget(btn1)
                grid.add_widget(btn2)

    def btn1_pressed(self, result, *args):
        new_text = result
        self.top_lbl.text = new_text

    def btn2_pressed(self, *args):
        self.grid_l.clear_widgets()
        #pass

class buttons_pressedApp(App):

    def build(self):

        return HomeScreen()

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

有效!

【问题讨论】:

【参考方案1】:

btn2.bind(on_release=self.btn2_pressed())

你有绑定错误的语法(这也是你其他问题的问题)。

Bind 接受一个函数,但你不是在传递一个函数,而是在调用这个函数。 bind 永远不会看到你碰巧在它的参数框中写了btn2_pressed,因为 python 调用该函数并且只将结果传递给bind

所以解决方案是,你真的想写类似的东西

btn2.bind(on_release=self.btn2_pressed)

注意删除的括号 - 这些是调用函数的语法,但我们特别不想这样做。相反,我们传递函数本身。

bind 也将一些默认参数传递给函数,而您的函数被定义为只接受一个。由于您不关心此处的额外参数,因此您可以使用

定义您的函数
def btn2_pressed(self, *args):

*args 捕获备用参数。不熟悉的可以查一下这个语法。

【讨论】:

你的回答非常适合我的例子(而且很清楚),我想给你答案。但是,我刚刚编辑了我的问题以反映我确实希望能够传递论点。现在 btn1 应该将列表中的每个“结果”传递给 btn1_pressed。 然后你需要创建一个包含这些参数的函数。一个常见的技巧是使用来自functools 模块的partial,在这种情况下你可以使用bind(on_release=partial(self.btn1_pressed, result))partial 是一个函数,它接受一个函数和一些参数,并返回一个新函数,该函数将首先传递这些参数。 这就是票!谢谢!

以上是关于动态创建的 kivy 按钮立即运行 on_press 和 on_release的主要内容,如果未能解决你的问题,请参考以下文章

按下时将动态创建的按钮信息传递给函数。 kivymd, kivy, 蟒蛇

Kivy - 通过引用根属性动态添加按钮

带有 Kivy 图像的动态按钮

KIVY:如何清除所有子项的网格布局

引用 Kivy 中动态创建的小部件的 id

如何使用On_Press更改动态创建的小部件的BG颜色并使用Pickle保存? (与Kivy的Python)