2012-08-13 18:42:40 +00:00
|
|
|
from kivy.adapters.listadapter import ListAdapter
|
|
|
|
from kivy.adapters.mixins.selection import SelectableDataItem
|
2012-07-13 11:07:23 +00:00
|
|
|
from kivy.uix.gridlayout import GridLayout
|
2012-07-31 03:39:15 +00:00
|
|
|
from kivy.uix.listview import ListView, ListItemButton
|
2012-08-01 12:29:11 +00:00
|
|
|
|
2012-08-13 18:42:40 +00:00
|
|
|
from fixtures import fruit_categories, fruit_data_list_of_dicts
|
2012-08-01 12:29:11 +00:00
|
|
|
|
2012-08-03 01:53:41 +00:00
|
|
|
from fruit_detail_view import FruitDetailView
|
2012-07-13 11:07:23 +00:00
|
|
|
|
2012-07-13 11:18:05 +00:00
|
|
|
# This is an expansion on the "master-detail" example to illustrate
|
2012-07-30 03:47:32 +00:00
|
|
|
# cascading from the selection of one list view to another. In this
|
|
|
|
# example the lists are restricted to single selection. The list on the
|
|
|
|
# left is a simple list. The list in the middle is specialized for
|
|
|
|
# observing the selection in the first, and using that item as the key
|
|
|
|
# into a dict providing its own list items. The view on the right is
|
|
|
|
# the sames as the DetailView in the master-detail example.
|
2012-07-13 11:07:23 +00:00
|
|
|
|
2012-08-06 20:58:49 +00:00
|
|
|
# A custom adapter is needed here, because we must transform the selected
|
|
|
|
# fruit category into the list of fruits for that category.
|
2012-08-13 18:42:40 +00:00
|
|
|
|
|
|
|
|
2012-08-03 01:53:41 +00:00
|
|
|
class FruitsListAdapter(ListAdapter):
|
|
|
|
|
|
|
|
def fruit_category_changed(self, fruit_categories_adapter, *args):
|
|
|
|
if len(fruit_categories_adapter.selection) == 0:
|
|
|
|
self.data = []
|
|
|
|
return
|
|
|
|
|
2012-08-13 18:42:40 +00:00
|
|
|
category = \
|
|
|
|
fruit_categories[str(fruit_categories_adapter.selection[0])]
|
|
|
|
|
|
|
|
# We are responsible with resetting the data. In this example, we are
|
|
|
|
# using lists of instances of the classes defined below, CategoryItem
|
|
|
|
# and FruitItem. We assume that the names of the fruits are unique,
|
|
|
|
# so we look up items by name.
|
|
|
|
#
|
|
|
|
self.data = \
|
|
|
|
[f for f in fruit_data_objects if f.name in category['fruits']]
|
|
|
|
|
|
|
|
# Also, see the examples that use dict records.
|
|
|
|
|
|
|
|
|
|
|
|
# FruitsListAdapter subclasses ListAdapter, which has SelectionSupport mixed
|
|
|
|
# in. SelectionSupport requires that data items handle selection operations.
|
|
|
|
# This means that we can't have simple strings as data items, nor can we have
|
|
|
|
# items that don't comply with SelectionSupport needs. It is not difficult to
|
|
|
|
# make your own data items, however, because you can define custom data item
|
|
|
|
# classes that subclass SelectableDataItem:
|
|
|
|
#
|
|
|
|
class CategoryItem(SelectableDataItem):
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
super(CategoryItem, self).__init__(**kwargs)
|
|
|
|
self.name = kwargs.get('name', '')
|
|
|
|
self.fruits = kwargs.get('fruits', [])
|
|
|
|
self.is_selected = kwargs.get('is_selected', False)
|
|
|
|
|
|
|
|
|
|
|
|
class FruitItem(SelectableDataItem):
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
super(FruitItem, self).__init__(**kwargs)
|
|
|
|
self.name = kwargs.get('name', '')
|
|
|
|
self.serving_size = kwargs.get('Serving Size', '')
|
|
|
|
self.data = kwargs.get('data', [])
|
|
|
|
self.is_selected = kwargs.get('is_selected', False)
|
|
|
|
|
|
|
|
|
|
|
|
# To instantiate CategoryItem and FruitItem instances, we use the dictionary-
|
|
|
|
# style fixtures data in fruit_data (See import above), which is
|
|
|
|
# also used by other list examples. The double asterisk usage here is for
|
|
|
|
# setting arguments from a dict in calls to instantiate the custom data item
|
|
|
|
# classes defined above.
|
|
|
|
|
|
|
|
# fruit_categories is a dict of dicts.
|
|
|
|
category_data_objects = \
|
|
|
|
[CategoryItem(**fruit_categories[c]) for c in sorted(fruit_categories)]
|
|
|
|
|
|
|
|
# fruit_data_list_of_dicts is a list of dicts, already sorted.
|
|
|
|
fruit_data_objects = \
|
|
|
|
[FruitItem(**fruit_dict) for fruit_dict in fruit_data_list_of_dicts]
|
|
|
|
|
|
|
|
# We end up with two normal lists of objects, to be used for two list views
|
|
|
|
# defined below.
|
2012-08-03 01:53:41 +00:00
|
|
|
|
|
|
|
|
2012-07-13 11:18:05 +00:00
|
|
|
class CascadingView(GridLayout):
|
|
|
|
'''Implementation of a master-detail style view, with a scrollable list
|
2012-08-13 18:42:40 +00:00
|
|
|
of fruit categories on the left, a list of fruits for the selected
|
|
|
|
category in the middle, and a detail view on the right.
|
2012-07-13 11:07:23 +00:00
|
|
|
'''
|
|
|
|
|
2012-07-15 15:55:04 +00:00
|
|
|
def __init__(self, **kwargs):
|
2012-07-13 11:07:23 +00:00
|
|
|
kwargs['cols'] = 3
|
|
|
|
kwargs['size_hint'] = (1.0, 1.0)
|
2012-07-13 11:18:05 +00:00
|
|
|
super(CascadingView, self).__init__(**kwargs)
|
2012-07-13 11:07:23 +00:00
|
|
|
|
2012-08-13 18:42:40 +00:00
|
|
|
list_item_args_converter = \
|
|
|
|
lambda selectable: {'text': selectable.name,
|
|
|
|
'size_hint_y': None,
|
|
|
|
'height': 25}
|
|
|
|
|
|
|
|
# Add a fruit categories list on the left. We use ListAdapter, for
|
|
|
|
# which we set the data argument to the list of CategoryItem
|
|
|
|
# instances from above. The args_converter only pulls the name
|
|
|
|
# property from these instances, adding also size_hint_y and height.
|
|
|
|
# selection_mode is single, because this list will "drive" the second
|
|
|
|
# list defined below. allow_empty_selection is False, because we
|
|
|
|
# always want a selected category, so that the second list will be
|
|
|
|
# populated. Finally, we instruct ListAdapter to build list item views
|
|
|
|
# using the provided cls, ListItemButton.
|
2012-07-13 11:07:23 +00:00
|
|
|
#
|
2012-07-30 03:47:32 +00:00
|
|
|
fruit_categories_list_adapter = \
|
2012-08-13 18:42:40 +00:00
|
|
|
ListAdapter(data=category_data_objects,
|
2012-07-14 15:55:14 +00:00
|
|
|
args_converter=list_item_args_converter,
|
2012-07-13 11:07:23 +00:00
|
|
|
selection_mode='single',
|
|
|
|
allow_empty_selection=False,
|
2012-07-31 03:39:15 +00:00
|
|
|
cls=ListItemButton)
|
2012-08-13 18:42:40 +00:00
|
|
|
|
2012-07-30 03:47:32 +00:00
|
|
|
fruit_categories_list_view = \
|
|
|
|
ListView(adapter=fruit_categories_list_adapter,
|
2012-08-13 18:42:40 +00:00
|
|
|
size_hint=(.2, 1.0))
|
|
|
|
|
2012-07-30 03:47:32 +00:00
|
|
|
self.add_widget(fruit_categories_list_view)
|
2012-07-13 11:07:23 +00:00
|
|
|
|
2012-08-13 18:42:40 +00:00
|
|
|
# Fruits, for a given category, are in a list in the middle, which
|
|
|
|
# uses FruitsListsAdapter, defined above. FruitsListAdapter has a
|
|
|
|
# fruit_changed() method that updates the data list. The binding
|
|
|
|
# to the fruit_categories_list_adapter is set up after
|
|
|
|
# instantiation of the fruit_list_adapter.
|
2012-07-13 11:07:23 +00:00
|
|
|
#
|
2012-08-13 18:42:40 +00:00
|
|
|
first_category_fruits = \
|
|
|
|
fruit_categories[fruit_categories.keys()[0]]['fruits']
|
|
|
|
|
|
|
|
first_category_fruit_data_objects = \
|
|
|
|
[f for f in fruit_data_objects if f.name in first_category_fruits]
|
2012-08-03 01:53:41 +00:00
|
|
|
|
2012-07-30 03:47:32 +00:00
|
|
|
fruits_list_adapter = \
|
2012-08-13 18:42:40 +00:00
|
|
|
FruitsListAdapter(data=first_category_fruit_data_objects,
|
|
|
|
args_converter=list_item_args_converter,
|
|
|
|
selection_mode='single',
|
|
|
|
allow_empty_selection=False,
|
|
|
|
cls=ListItemButton)
|
2012-08-06 18:35:27 +00:00
|
|
|
|
2012-08-03 01:53:41 +00:00
|
|
|
fruit_categories_list_adapter.bind(
|
|
|
|
on_selection_change=fruits_list_adapter.fruit_category_changed)
|
|
|
|
|
2012-07-30 03:47:32 +00:00
|
|
|
fruits_list_view = \
|
2012-08-13 18:42:40 +00:00
|
|
|
ListView(adapter=fruits_list_adapter, size_hint=(.2, 1.0))
|
2012-08-03 22:50:22 +00:00
|
|
|
|
2012-07-30 03:47:32 +00:00
|
|
|
self.add_widget(fruits_list_view)
|
2012-07-13 11:07:23 +00:00
|
|
|
|
|
|
|
# Detail view, for a given fruit, on the right:
|
|
|
|
#
|
2012-08-03 01:53:41 +00:00
|
|
|
detail_view = FruitDetailView(size_hint=(.6, 1.0))
|
2012-08-06 18:35:27 +00:00
|
|
|
|
2012-07-31 19:08:00 +00:00
|
|
|
fruits_list_adapter.bind(
|
2012-08-03 01:53:41 +00:00
|
|
|
on_selection_change=detail_view.fruit_changed)
|
2012-07-30 03:47:32 +00:00
|
|
|
self.add_widget(detail_view)
|
2012-07-13 11:07:23 +00:00
|
|
|
|
2012-08-01 12:04:36 +00:00
|
|
|
# Force triggering of on_selection_change() for the DetailView, for
|
2012-08-05 17:41:06 +00:00
|
|
|
# correct initial display.
|
2012-08-01 12:04:36 +00:00
|
|
|
fruits_list_adapter.touch_selection()
|
2012-07-31 19:08:00 +00:00
|
|
|
|
2012-07-13 11:07:23 +00:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
|
|
|
|
from kivy.base import runTouchApp
|
|
|
|
|
2012-07-15 15:55:04 +00:00
|
|
|
runTouchApp(CascadingView(width=800))
|