3

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

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

Kivy 中的 on_press 在启动时一直运行

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

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

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

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()

有用!

4

1 回答 1

4

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抓住多余的论点。如果您不熟悉此语法,可以查找它。

于 2014-01-25T00:19:50.707 回答