动态创建的kivy按钮在按下并立即释放时运行

2024-10-01 09:30:13 发布

您现在位置:Python中文网/ 问答频道 /正文

为什么我的kivy按钮在python中创建时就被按下了呢?

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

on_press in Kivy keeps running at start up instead

……然而,没有有效的例子。我试着在那里复制这个例子来理解答案,但是缺乏经验来填补这个例子中缺少的东西。

所以,对于那些看到另一个答案,可以在这里应用它,并用更简单的英语解释noob的人来说,这可能是一个简单的答案。

下面是这个问题的一个小的工作示例:

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,并在其中插入括号和“result”作为参数。当然,这是立即运行,并返回无需约束(正如恶劣解释)。我希望能够传入示例中已定义的“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()

它起作用了!


Tags: textfromimportselfnonedefbuttonresult
1条回答
网友
1楼 · 发布于 2024-10-01 09:30:13

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捕获备用参数。如果您不熟悉这个语法,可以查找它。

相关问题 更多 >