Search code examples
pythonfor-loopbuttonkivytogglebutton

Binding a function with togglebutton in Kivy


I'm new in Python and Kivy, so probably my problem is "easy", but I can't find solution.. please help.

I'm trying to create several togglebuttons in for-loop, and I have problem with binding each togglebutton with a function from another class (normally in my code it was: on_state: root.on_state(self.state, self.text)

I've tried to do this with lambda function. I have no errros, but when togglebutton is pressed (state = "down") nothing happens.

If anyone could show me where the problem is, I would be very grateful. Thanks!

class MainScreen(BoxLayout):
    def __init__(self, **kwargs):
        super().__init__(**kwargs)
        Window.size = (1400, 800)
        self.ingredients_sites = []
        self.count = 0

    def display_text(self, count):
        # displaying result in TextInput from ingredient_sites
        result = zielone_koktajle.result(self.ingredients_sites, count)
        if result:
            self.display.text = ", ".join(result)
        else:
            self.display.text = ""
            self.display.hint_text = "brak wspolnych stron.."

    def on_state(self, state, ingredient_name):
        if state == "down":
            # dd sites to the list "ingredient_sites" by extracting value of the key from dictionary
            self.ingredients_sites.extend(zielone_koktajle.get_ing_from_csv().get(ingredient_name))
            self.count += 1
            self.display_text(self.count)

        if state == "normal":
            for i in (zielone_koktajle.get_ing_from_csv().get(ingredient_name)):
                if i in self.ingredients_sites:
                    self.ingredients_sites.remove(i)
            self.count -= 1
            self.display_text(self.count)


class ToggleButtons(StackLayout):
    lista = ["aloes", "blonnik", "herbata biala", "herbata czerwona", "herbata zielona", "mleko kokosowe", "mleko migdalowe", "mleko owsiane", "mleko ryzowe", "mleko sojowe", "woda kokosowa"]
    def __init__(self, **kwargs):
        super().__init__(**kwargs)
        self.add_togglebuttons()

    def add_togglebuttons(self):
        for i in ToggleButtons.lista:
            tgbtn = ToggleButton(id = "togglebtn",
                                 text = i,
                                 size_hint = (.1, .30)
                                 )

            tgbtn.bind(on_state = lambda x: MainScreen.on_state(self, tgbtn.state, i))

            self.add_widget(tgbtn)


class ZieloneKoktajleApp(App):
     def build(self):
        self.title = "Zielone koktajle - index"
        return MainScreen()

and short sample of my KV file:

<MainScreen>:
    orientation: "vertical"
    id_togglebtn: togglebtn

    BoxLayout:
        id: togglebtn
        orientation: "vertical"

        Label:
            text: "Zielone koktajle"

    BoxLayout:
        id: id_togglebuttons
        orientation: "vertical"
        spacing: 10

        CustLabel:
            text: "WARZYWA"
        StackLayout:
            id: warzywa
            CustWidget:
                text: "awokado"
                on_state: root.on_state(self.state, self.text)

        CustLabel:
            text: "NAPOJE"
        StackLayout:
            id: napoje
            ToggleButtons:

Solution

  • You can bind it in .kv.
    Try this:

    from kivy.app import App
    from kivy.lang import Builder
    from kivy.uix.stacklayout import StackLayout
    from kivy.uix.boxlayout import BoxLayout
    from kivy.uix.togglebutton import ToggleButton
    
    
    Builder.load_string('''
    
    <ToggleButton>:
        on_state: app.root.on_state(self)
    
    <MainScreen>:
    
        BoxLayout:
    
            ToggleButtons:
    
    
    
    ''')
    
    class MainScreen(BoxLayout):
    
        def on_state(self, togglebutton):
            tb = togglebutton
            print(tb,tb.state,tb.text)
    
    
    class ToggleButtons(StackLayout):
        lista = ["aloes", "blonnik", "herbata biala", "herbata czerwona", "herbata zielona", "mleko kokosowe", "mleko migdalowe", "mleko owsiane", "mleko ryzowe", "mleko sojowe", "woda kokosowa"]
    
        def __init__(self,**kwargs):
            super(ToggleButtons,self).__init__(**kwargs)
            for i in self.lista:
                tgbtn = ToggleButton(text = i,size_hint = (.1, .30))
    
                self.add_widget(tgbtn)
    
    
    class MyApp(App):
         def build(self):
            return MainScreen()
    
    
    MyApp().run()