mirror of https://github.com/kivy/kivy.git
313 lines
12 KiB
Python
313 lines
12 KiB
Python
from kivy.uix.boxlayout import BoxLayout
|
|
from kivy.uix.gridlayout import GridLayout
|
|
from kivy.uix.label import Label
|
|
from kivy.uix.button import Button
|
|
from kivy.properties import ObjectProperty, \
|
|
NumericProperty, ListProperty, \
|
|
StringProperty
|
|
from kivy.uix.listview import ListView, ListAdapter
|
|
from kivy.uix.mixins.selection import SelectionObserver, SelectableItem
|
|
|
|
# [TODO] NOTE -- This is a copy of the old version of list_master_detail.py,
|
|
# because it contains an example of how to make a custom
|
|
# ListItem.
|
|
#
|
|
# Calling this example "disclosure" because plan is to try
|
|
# expanding a row to show a detail view when the disclosure
|
|
# button for a row is clicked.
|
|
#
|
|
# Something like this:
|
|
#
|
|
# http://www.zkoss.org/zkdemo/grid/master_detail
|
|
#
|
|
# Master-Detail view for showing a list on the left (the master) and a view
|
|
# on the right for the detail view.
|
|
|
|
# A "master-detail" view is a good way to experiment with a listview
|
|
# (the master) and another view (detail view) that gets updated upon selection.
|
|
|
|
|
|
# MASTER list
|
|
|
|
# For the master list, we need to create a custom "list item" type that
|
|
# subclasses SelectableItem.
|
|
|
|
# "Sub" to indicate that this class is for "sub" list items -- a list item
|
|
# could consist of a button on the left, several labels in the middle, and
|
|
# another button on the right. Not sure of the merit of allowing, perhaps,
|
|
# some "sub" list items to react to touches and others not, if that were to
|
|
# be enabled.
|
|
|
|
class ListItemSubButton(SelectableItem, Button):
|
|
selected_color = ListProperty([1., 0., 0., 1])
|
|
deselected_color = None
|
|
|
|
def __init__(self, list_adapter, **kwargs):
|
|
self.list_adapter = list_adapter
|
|
super(ListItemSubButton, self).__init__(**kwargs)
|
|
|
|
# Set deselected_color to be default Button bg color.
|
|
self.deselected_color = self.background_color
|
|
|
|
self.bind(on_release=self.handle_selection)
|
|
|
|
def handle_selection(self, button):
|
|
# Not this "sub" list item, but the list item.
|
|
self.list_adapter.handle_selection(self.parent)
|
|
|
|
# [TODO] At least there is some action on this set, but
|
|
# the color gets somehow composited.
|
|
def select(self, *args):
|
|
self.background_color = self.selected_color
|
|
|
|
# [TODO] No effect seen, but it is grey, so might be happening.
|
|
def deselect(self, *args):
|
|
self.background_color = self.deselected_color
|
|
|
|
def __repr__(self):
|
|
return self.text
|
|
|
|
|
|
class ListItemSubLabel(SelectableItem, Label):
|
|
# Same idea as "sub" for button above.
|
|
selected_color = ListProperty([1., 0., 0., 1])
|
|
deselected_color = ListProperty([.33, .33, .33, 1])
|
|
|
|
def __init__(self, list_adapter, **kwargs):
|
|
self.list_adapter = list_adapter
|
|
super(ListItemSubLabel, self).__init__(**kwargs)
|
|
|
|
self.bind(on_release=self.handle_selection)
|
|
|
|
def handle_selection(self, button):
|
|
# Not this "sub" list item, but the list item (parent).
|
|
self.list_adapter.handle_selection(self.parent)
|
|
|
|
# [TODO] Should Label have background_color, like Button, etc.?
|
|
# [TODO] Not tested yet.
|
|
def select(self, *args):
|
|
self.bold = True
|
|
|
|
def deselect(self, *args):
|
|
self.bold = False
|
|
|
|
def __repr__(self):
|
|
return self.text
|
|
|
|
|
|
class ListItem(SelectableItem, BoxLayout):
|
|
# ListItem (BoxLayout) by default uses orientation='horizontal',
|
|
# but could be used also for a side-to-side display of items.
|
|
#
|
|
# ListItemSubButton sublasses Button, which has background_color.
|
|
# Here we must add this property.
|
|
background_color = ListProperty([1, 1, 1, 1])
|
|
|
|
selected_color = ListProperty([1., 0., 0., 1])
|
|
deselected_color = ListProperty([.33, .33, .33, 1])
|
|
|
|
icon_button = ObjectProperty(None)
|
|
content_button = ObjectProperty(None)
|
|
|
|
def __init__(self, list_adapter, **kwargs):
|
|
self.list_adapter = list_adapter
|
|
super(ListItem, self).__init__(size_hint_y=None, height=25)
|
|
|
|
# Now this button just has text '>', but it would be neat to make the
|
|
# left button hold icons -- the list would be heterogeneous, containing
|
|
# different ListItem types that could be filtered perhaps (an option
|
|
# for selecting all of a given type, for example).
|
|
self.icon_button = ListItemSubButton(
|
|
list_adapter=list_adapter,
|
|
text='>', size_hint_x=.05, size_hint_y=None, height=25)
|
|
self.content_button = ListItemSubButton(
|
|
list_adapter=list_adapter, **kwargs)
|
|
self.add_widget(self.icon_button)
|
|
self.add_widget(self.content_button)
|
|
|
|
self.bind(on_release=self.handle_selection)
|
|
|
|
def handle_selection(self, item):
|
|
self.list_adapter.handle_selection(self)
|
|
|
|
def select(self, *args):
|
|
self.background_color = self.selected_color
|
|
|
|
def deselect(self, *args):
|
|
self.background_color = self.deselected_color
|
|
|
|
def __repr__(self):
|
|
if self.content_button is not None:
|
|
return self.content_button.text
|
|
else:
|
|
return 'empty'
|
|
|
|
|
|
class DetailView(SelectionObserver, GridLayout):
|
|
fruit_name = StringProperty('')
|
|
|
|
def __init__(self, **kwargs):
|
|
kwargs['cols'] = 2
|
|
super(DetailView, self).__init__(**kwargs)
|
|
self.bind(fruit_name=self.redraw)
|
|
|
|
def redraw(self, *args):
|
|
self.clear_widgets()
|
|
self.add_widget(Label(text="Name:", halign='right'))
|
|
self.add_widget(Label(text=self.fruit_name))
|
|
for category in descriptors:
|
|
self.add_widget(Label(text="{0}:".format(category),
|
|
halign='right'))
|
|
self.add_widget(
|
|
Label(text=str(fruit_data[self.fruit_name][category])))
|
|
|
|
def observed_selection_changed(self, list_adapter, selection):
|
|
if len(list_adapter.selection) == 0:
|
|
return
|
|
|
|
selected_object = list_adapter.selection[0]
|
|
|
|
if type(selected_object) is str:
|
|
self.fruit_name = selected_object
|
|
else:
|
|
self.fruit_name = str(selected_object)
|
|
print 'just added or updated detail label'
|
|
|
|
|
|
class MasterDetailView(GridLayout):
|
|
'''Implementation of an MasterDetailView with a vertical scrollable list
|
|
on the left (the master, or source list) and a detail view on the right.
|
|
'''
|
|
|
|
list_adapter = ObjectProperty(None)
|
|
|
|
master_list_view = ObjectProperty(None)
|
|
|
|
divider = ObjectProperty(None)
|
|
|
|
divider_width = NumericProperty(2)
|
|
|
|
detail_view = ObjectProperty(None)
|
|
|
|
def __init__(self, items, **kwargs):
|
|
#kwargs['orientation'] = 'horizontal'
|
|
kwargs['cols'] = 2
|
|
kwargs['size_hint'] = (1.0, 1.0)
|
|
super(MasterDetailView, self).__init__(**kwargs)
|
|
|
|
list_item_args_converter = lambda x: {'text': x,
|
|
'size_hint_y': None,
|
|
'height': 25}
|
|
self.list_adapter = ListAdapter(items,
|
|
args_converter=list_item_args_converter,
|
|
selection_mode='single',
|
|
allow_empty_selection=False,
|
|
cls=ListItem)
|
|
self.master_list_view = ListView(adapter=self.list_adapter,
|
|
size_hint=(.3, 1.0))
|
|
self.add_widget(self.master_list_view)
|
|
|
|
self.detail_view = DetailView(size_hint=(.7, 1.0))
|
|
self.add_widget(self.detail_view)
|
|
|
|
self.list_adapter.bind(
|
|
selection=self.detail_view.observed_selection_changed)
|
|
|
|
self.list_adapter.initialize_selection()
|
|
|
|
# Data from http://www.fda.gov/Food/LabelingNutrition/\
|
|
# FoodLabelingGuidanceRegulatoryInformation/\
|
|
# InformationforRestaurantsRetailEstablishments/\
|
|
# ucm063482.htm
|
|
fruit_categories = \
|
|
{'Melons': ['Cantaloupe', 'Honeydew Melon', 'Watermelon'],
|
|
'Tree Fruits': ['Apple', 'Avocado, California', 'Banana', 'Nectarine',
|
|
'Peach', 'Pear', 'Pineapple', 'Plums',
|
|
'Sweet Cherries'],
|
|
'Citrus Fruits': ['Grapefruit', 'Lemon', 'Lime', 'Orange',
|
|
'Tangerine'],
|
|
'Miscellaneous Fruits': ['Grapes', 'Kiwifruit', 'Strawberries']}
|
|
descriptors = """(gram weight/ ounce weight) Calories Calories from Fa
|
|
t Total Fat Sodium Potassium Total Carbo-hydrate Dietary Fiber Suga
|
|
rs Protein Vitamin A Vitamin C Calcium Iron""".replace('\n', '')
|
|
descriptors = [item.strip() for item in descriptors.split('\t')]
|
|
units = """(g) (%DV) (mg) (%DV) (mg) (%DV) (g) (%DV) (g)
|
|
(%DV) (g) (g) (%DV) (%DV) (%DV) (%DV)""".replace('\n', '')
|
|
units = [item.strip() for item in units.split('\t')]
|
|
raw_fruit_data = [
|
|
{'name':'Apple',
|
|
'Serving Size': '1 large (242 g/8 oz)',
|
|
'data': [130, 0, 0, 0, 0, 0, 260, 7, 34, 11, 5, 20, 25, 1, 2, 8, 2, 2]},
|
|
{'name':'Avocado, California',
|
|
'Serving Size': '1/5 medium (30 g/1.1 oz)',
|
|
'data': [50, 35, 4.5, 7, 0, 0, 140, 4, 3, 1, 1, 4, 0, 1, 0, 4, 0, 2]},
|
|
{'name':'Banana',
|
|
'Serving Size': '1 medium (126 g/4.5 oz)',
|
|
'data': [110, 0, 0, 0, 0, 0, 450, 13, 30, 10, 3, 12, 19, 1, 2, 15, 0, 2]},
|
|
{'name':'Cantaloupe',
|
|
'Serving Size': '1/4 medium (134 g/4.8 oz)',
|
|
'data': [50, 0, 0, 0, 20, 1, 240, 7, 12, 4, 1, 4, 11, 1, 120, 80, 2, 2]},
|
|
{'name':'Grapefruit',
|
|
'Serving Size': '1/2 medium (154 g/5.5 oz)',
|
|
'data': [60, 0, 0, 0, 0, 0, 160, 5, 15, 5, 2, 8, 11, 1, 35, 100, 4, 0]},
|
|
{'name':'Grapes',
|
|
'Serving Size': '3/4 cup (126 g/4.5 oz)',
|
|
'data': [90, 0, 0, 0, 15, 1, 240, 7, 23, 8, 1, 4, 20, 0, 0, 2, 2, 0]},
|
|
{'name':'Honeydew Melon',
|
|
'Serving Size': '1/10 medium melon (134 g/4.8 oz)',
|
|
'data': [50, 0, 0, 0, 30, 1, 210, 6, 12, 4, 1, 4, 11, 1, 2, 45, 2, 2]},
|
|
{'name':'Kiwifruit',
|
|
'Serving Size': '2 medium (148 g/5.3 oz)',
|
|
'data': [90, 10, 1, 2, 0, 0, 450, 13, 20, 7, 4, 16, 13, 1, 2, 240, 4, 2]},
|
|
{'name':'Lemon',
|
|
'Serving Size': '1 medium (58 g/2.1 oz)',
|
|
'data': [15, 0, 0, 0, 0, 0, 75, 2, 5, 2, 2, 8, 2, 0, 0, 40, 2, 0]},
|
|
{'name':'Lime',
|
|
'Serving Size': '1 medium (67 g/2.4 oz)',
|
|
'data': [20, 0, 0, 0, 0, 0, 75, 2, 7, 2, 2, 8, 0, 0, 0, 35, 0, 0]},
|
|
{'name':'Nectarine',
|
|
'Serving Size': '1 medium (140 g/5.0 oz)',
|
|
'data': [60, 5, 0.5, 1, 0, 0, 250, 7, 15, 5, 2, 8, 11, 1, 8, 15, 0, 2]},
|
|
{'name':'Orange',
|
|
'Serving Size': '1 medium (154 g/5.5 oz)',
|
|
'data': [80, 0, 0, 0, 0, 0, 250, 7, 19, 6, 3, 12, 14, 1, 2, 130, 6, 0]},
|
|
{'name':'Peach',
|
|
'Serving Size': '1 medium (147 g/5.3 oz)',
|
|
'data': [60, 0, 0.5, 1, 0, 0, 230, 7, 15, 5, 2, 8, 13, 1, 6, 15, 0, 2]},
|
|
{'name':'Pear',
|
|
'Serving Size': '1 medium (166 g/5.9 oz)',
|
|
'data': [100, 0, 0, 0, 0, 0, 190, 5, 26, 9, 6, 24, 16, 1, 0, 10, 2, 0]},
|
|
{'name':'Pineapple',
|
|
'Serving Size': '2 slices, 3" diameter, 3/4" thick (112 g/4 oz)',
|
|
'data': [50, 0, 0, 0, 10, 0, 120, 3, 13, 4, 1, 4, 10, 1, 2, 50, 2, 2]},
|
|
{'name':'Plums',
|
|
'Serving Size': '2 medium (151 g/5.4 oz)',
|
|
'data': [70, 0, 0, 0, 0, 0, 230, 7, 19, 6, 2, 8, 16, 1, 8, 10, 0, 2]},
|
|
{'name':'Strawberries',
|
|
'Serving Size': '8 medium (147 g/5.3 oz)',
|
|
'data': [50, 0, 0, 0, 0, 0, 170, 5, 11, 4, 2, 8, 8, 1, 0, 160, 2, 2]},
|
|
{'name':'Sweet Cherries',
|
|
'Serving Size': '21 cherries; 1 cup (140 g/5.0 oz)',
|
|
'data': [100, 0, 0, 0, 0, 0, 350, 10, 26, 9, 1, 4, 16, 1, 2, 15, 2, 2]},
|
|
{'name':'Tangerine',
|
|
'Serving Size': '1 medium (109 g/3.9 oz)',
|
|
'data': [50, 0, 0, 0, 0, 0, 160, 5, 13, 4, 2, 8, 9, 1, 6, 45, 4, 0]},
|
|
{'name':'Watermelon',
|
|
'Serving Size': '1/18 medium melon; 2 cups diced pieces (280 g/10.0 oz)',
|
|
'data': [80, 0, 0, 0, 0, 0, 270, 8, 21, 7, 1, 4, 20, 1, 30, 25, 2, 4]}]
|
|
|
|
fruit_data = {}
|
|
descriptors_and_units = dict(zip(descriptors, units))
|
|
for row in raw_fruit_data:
|
|
fruit_data[row['name']] = {}
|
|
fruit_data[row['name']] = dict({'Serving Size': row['Serving Size']},
|
|
**dict(zip(descriptors_and_units.keys(), row['data'])))
|
|
|
|
if __name__ == '__main__':
|
|
|
|
from kivy.base import runTouchApp
|
|
|
|
master_detail = MasterDetailView(fruit_data.keys(), width=800)
|
|
|
|
runTouchApp(master_detail)
|