How to duplicate blocks of widgets in kv file (lowercase-only rule)

前端 未结 1 1054
心在旅途
心在旅途 2021-01-26 18:26

I try to understand how kivy .kv files work, so I created a small application with a horizontial BoxLayout which contains three GridLayouts, as shown:

my_widget:         


        
相关标签:
1条回答
  • 2021-01-26 18:58

    Yeah, this gets me badly everytime I encounter it e.g. copy&paste word into <> (and forget about uppercase). I still wonder if I should treat it as a bug or as a feature, because it forces user to name widgets in a correct style/case, which makes it easier for reading too.

    The thing is that widgets/rules in kv language should use ThisWordStyle or something similar, but the first capital letter seems to be important. I encountered even a case when a capital letter inside the word was enough and the rest lowercase, but can't reproduce, sadly.

    Words with lowercase only are mostly used as properties or variables, so maybe the my_widget was handled as a property or variable too like a global, or was completely ignored when run through the language parser.

    Let's see:

    1) your kv layout put into App:

    from kivy.app import App
    from kivy.lang import Builder
    from kivy.uix.boxlayout import BoxLayout
    Builder.load_string('''
    <my_widget@BoxLayout>:
        my_widget2:
        Button:
            text: 'bla'
    
    <MyWidget>:
        my_widget:
        Button:
            text: 'bad'
    
    <my_widget2@GridLayout>:
        rows: 3
        ToggleButton:
        Image:
        Label:
    
    ''')
    class MyWidget(BoxLayout):
        pass
    
    class Test(App):
        def build(self):
            return MyWidget()
    Test().run()
    

    The only visible thing will be a Button with 'bad' string

    2) a little change in naming - my_widget -> My_widget

    <My_widget@BoxLayout>:
        my_widget2:
        Button:
            text: 'bla'
    
    <MyWidget>:
        My_widget:
        Button:
            text: 'bad'
    
    <my_widget2@GridLayout>:
        rows: 3
        ToggleButton:
        Image:
        Label:
    

    and there's one more visible widget!

    3) a working layout with all stuff (my_widget2 -> My_widget2)

    <My_widget@BoxLayout>:
        My_widget2:
        My_widget2:
        Button:
            text: 'bla'
    
    <MyWidget>:
        My_widget:
        Button:
            text: 'bad'
    
    <My_widget2@GridLayout>:
        rows: 3
        ToggleButton:
        Image:
        Label:
    

    Also to answer passing arguments into such widgets (<My@Widget>), use Factory to access such widget and then it's only passing (kw)args:

    #:import Factory kivy.factory.Factory
    <MyWidget>:
        Button:
            on_release: root.add_widget(Factory.My_widget2(text='hi'))
    
    <My_widget2@Label>:
        size_hint: [None, None]
        size: [200, 50]
    
    0 讨论(0)
提交回复
热议问题