1

リストビューで辞書のリストを並べ替えようとしています。私は ListAdapter.data を変更することによってそれを行っています。

kv ファイルでは、リストビューの上のスピナー ボタンがソート値を選択し、on_release が root.beer_sort を呼び出して ListAdapter (beer_la という名前) データを変更します。コンソールにbeer_la.dataを出力すると、変わったように見えますが、アプリの表示順が更新されません。

class CellarDoor(BoxLayout):
    def __init__(self, **kwargs):
        self.beer_archive = []
        self.wine_archive = []
        with open(join(FILE_ROOT, 'beer_archive.csv'), 'rb', 1) as beer_csv:
            self.beer_archive = list(csv.DictReader(beer_csv))

        self.beer_la = ListAdapter(data=self.beer_archive, 
                          args_converter=self.beer_formatter,
                          cls=CompositeListItem,         
                          selection_mode='single',
                          allow_empty_selection=True)
        super(CellarDoor, self).__init__(**kwargs)

    def beer_formatter(self, row_index, beer_data):
        return {'text': beer_data['Beer'],
                'size_hint_y': None,
                'height': 50,
                'cls_dicts': [{'cls': ListItemImage,
                               'kwargs': {'size_hint_x': .2,
                                          'size': self.parent.size,
                                          'cellar': 'Beer',
                                          'style': beer_data['Style']}},
                              {'cls': ListItemLabel,
                               'kwargs': {'text': beer_data['Beer']}},
                              {'cls': ListItemButton,
                               'kwargs': {'text': str(beer_data['Stock'])}}]}

    def beer_sort(self, sort, reverse):
        self.beer_la.data = sorted(self.beer_archive,
                                   key=lambda k: k[sort],
                                   reverse=reverse)
        print "new sort = %s" % sort
        print self.beer_la.data

kv ファイルの関連部分

<CellarDoor>
    ...
    Spinner:
        id: beer_sort_spinner
        text: 'Brewery'
        values: ['Brewery', 'Beer', 'Year', 'Style', 'Stock', 'Rating', 'Price']
        on_release: root.beer_sort(self.text, (True if future_search_button.state=='down' else False))
    ToggleButton:
        id: future_search_button
        text: 'Reverse'
    Button:
        text: 'Add Beer'
    ListView:
        adapter: root.beer_la
4

3 に答える 3

1

あなたのお悩みに少しでもお役に立てれば幸いです。CellarDoor(Kivy言語を使用する代わりに)コンストラクターを呼び出した後にリストビューを追加すると、sortメソッドが機能し始めることがわかりました。私の例はあなたの単純化されたバージョンですが、あなたが説明した動作はまだ存在していました.

コメントを外した場合:

#    ListView:
#        adapter: root.beer_la

とコメント:

        self.add_widget(ListView(adapter=self.beer_la))

その後、あなたが説明したようにソートが機能しなくなります。

不思議なことに、大きなリスト (100 としましょう) がある場合、ListViewそれに対するアクション (スクロールまたは選択) とアイテムの実行後に更新されます。どういうわけか、これはアダプタのプロパティが画面の更新にバインドされていないことを意味します。これは、 Githubで開発者に報告するに値する問題である可能性があります。うまく説明できませんが、少なくとも問題が解決することを願っています。Kivy Language でアダプターを直接作成して追加することもできます。

from kivy.adapters.listadapter import ListAdapter
from kivy.uix.listview import ListView, ListItemButton
from kivy.uix.boxlayout import BoxLayout
from kivy.lang import Builder
from kivy.app import App

Builder.load_string("""
<CellarDoor>:
    ToggleButton:
        text: 'Reverse'
        on_state: root.beer_sort((True if self.state=='down' else False))
#    ListView:
#        adapter: root.beer_la
""")

class CellarDoor(BoxLayout):
    def __init__(self, **kwargs):
        self.beer_archive = ["Item #{0}".format(i) for i in range(10)]
        self.beer_la = ListAdapter(data=self.beer_archive, 
                          cls=ListItemButton,
                          selection_mode='single',
                          allow_empty_selection=True)

        super(CellarDoor, self).__init__(**kwargs)
        self.add_widget(ListView(adapter=self.beer_la))


    def beer_sort(self, reverse):
        self.beer_la.data = sorted(self.beer_archive,
                                   reverse=reverse)
        print self.beer_la.data

class TestApp(App):
    def build(self):
        return CellarDoor()

if __name__ == '__main__':
    TestApp().run()
于 2013-07-17T02:29:14.473 に答える
0

O'Reillyの「Kivyでアプリを作成する_trigger_reset_populate()」という本では、次の機能が提案されています。

    def update_data(self):
        self.list_adapter.data = mydata
        self.list_view._trigger_reset_populate()
于 2017-01-06T03:24:10.363 に答える