如何在Kivy App中为ListView的不同实例设置不同的属性?

时间:2017-01-04 12:50:01

标签: android python listview kivy kivy-language

我希望我的Kivy应用程序拥有多个ListView实例。在分配不同的属性时,我很困惑,尤其是回调。以下代码说明了我的问题:

from kivy.app import App
from kivy.uix.listview import ListItemButton
from kivy.uix.screenmanager import ScreenManager, Screen
from kivy.lang import Builder

Builder.load_string('''
#: import ListAdapter kivy.adapters.listadapter.ListAdapter
#: import ListItemButton kivy.uix.listview.ListItemButton
<smRoot>:
    Screen1:
    Screen2:
<ListItemButton>:
    on_press: app.callback1(self)
    height: dp(25)
    size_hint: (1,.1)
<Screen1>:
    name:'screen1'
    BoxLayout:
        orientation: 'vertical'
        Label:
            text: 'Screen1'
            size_hint_y: .1
        ListView:
            id:lstScreen1    
            adapter:
                ListAdapter(data=[{'text':'item1'},{'text':'item2'},{'text':'item3'}],
                args_converter=lambda row_index, rec: {'text': rec['text']},
                cls=ListItemButton,
                selection_mode='single',
                allow_empty_selection=False)
        Button:
            text: 'Switch Screen'
            size_hint_y: .1
            on_press: root.manager.current = 'screen2'
<ListItemButton>:
    on_press: app.callback2(self)
    height: dp(25)
    size_hint: (1,.1)            
<Screen2>:
    name:'screen2'
    BoxLayout:
        orientation: 'vertical'
        Label:
            text: 'Screen2'
            size_hint_y: .1
        ListView:
            id:lstScreen2    
            adapter:
                ListAdapter(data=[{'text':'item1'},{'text':'item2'},{'text':'item3'}],
                args_converter=lambda row_index, rec: {'text': rec['text']},
                cls=ListItemButton,
                selection_mode='single', 
                allow_empty_selection=False)
        Button:
            text: 'Switch Screen'
            size_hint_y: .1
            on_press: root.manager.current = 'screen1'
''')

class smRoot(ScreenManager):
    pass

class Screen1(Screen):
    MyButt = ListItemButton
    pass

class Screen2(Screen):
    pass

class myApp(App):
    def build(self):
        smroot = smRoot()
        return smroot

    def callback1(self, btn_instance):
        index = btn_instance.index
        print 'From Screen 1: ' + str(index)

    def callback2(self, btn_instance):
        index = btn_instance.index
        print 'From Screen 2: ' + str(index)

myApp().run()

在这里,我希望屏幕1和2上的每个ListView实例都有单独的回调。但是当我点击任一屏幕时,输出都是相同的,如下所示:

From Screen 2: 0
From Screen 1: 0

理想情况下,我希望实现像Dynamic Class之类的东西。但是当我尝试这样的事情时:

<MyButt@ListItemButton>:
    #on_press: app.callback1(self)
    height: dp(25)
    size_hint: (1,.1)
<Screen1>:
    name:'screen1'
    MyButt:
        on_press: app.callback1(self)
    BoxLayout:
        orientation: 'vertical'
        Label:
            text: 'Screen1'
            size_hint_y: .1
        ListView:
            id:lstScreen1    
            adapter:
                ListAdapter(data=[{'text':'item1'},{'text':'item2'},{'text':'item3'}],
                args_converter=lambda row_index, rec: {'text': rec['text']},
                cls=MyButt,
                selection_mode='single',
                allow_empty_selection=False)

我收到NameError: name 'MyButt' is not defined错误。据我所知,我在上面链接的文档示例中遵循相同的语法,所以我很困惑。

我没有尝试在Python中创建自定义类,但我宁愿将.iv元素中的UI元素分开(如果我没有弄错,那不是kv背后的基本思想吗?)。< / p>

非常感谢这方面的任何帮助/指示/建议。

1 个答案:

答案 0 :(得分:0)

你不能在kv中使用python端(在“:”之后)的widget类而不导入它,但是因为MyButt没有在任何模块中定义,所以是一个动态类(在纯kv中定义) )您可以使用Factory来使用它。

首先,将工厂导入kv文件的顶部。

#:import Factory kivy.factory.Factory

然后改变

    ListView:
        id:lstScreen1    
        adapter:
            ListAdapter(data=[{'text':'item1'},{'text':'item2'},{'text':'item3'}],
            args_converter=lambda row_index, rec: {'text': rec['text']},
            cls=MyButt,

    ListView:
        id:lstScreen1    
        adapter:
            ListAdapter(data=[{'text':'item1'},{'text':'item2'},{'text':'item3'}],
            args_converter=lambda row_index, rec: {'text': rec['text']},
            cls=Factory.MyButt,