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

时间:2014-01-25 00:12:59

标签: python button dynamic kivy pressed

为什么我的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做同样的事情时,一些默认参数被传递给方法并且它给出了一个错误。所以,我像前面一样用括号和'result'作为参数留下了btn1。当然,这是立即运行并且不返回任何内容(如解释中所述)。我希望能够传递示例中已定义的'result',但自然不会立即运行它。道歉......我应该第一次这样写。

编辑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 个答案:

答案 0 :(得分: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捕获备用参数。如果您不熟悉它,可以查看此语法。