Show More
The requested changes are too big and content was truncated. Show full diff
@@ -1,123 +1,138 b'' | |||
|
1 | 1 | """Interact with functions using widgets. |
|
2 | 2 | """ |
|
3 | 3 | |
|
4 | 4 | #----------------------------------------------------------------------------- |
|
5 | 5 | # Copyright (c) 2013, the IPython Development Team. |
|
6 | 6 | # |
|
7 | 7 | # Distributed under the terms of the Modified BSD License. |
|
8 | 8 | # |
|
9 | 9 | # The full license is in the file COPYING.txt, distributed with this software. |
|
10 | 10 | #----------------------------------------------------------------------------- |
|
11 | 11 | |
|
12 | 12 | #----------------------------------------------------------------------------- |
|
13 | 13 | # Imports |
|
14 | 14 | #----------------------------------------------------------------------------- |
|
15 | 15 | |
|
16 |
from IPython.html.widgets import (Widget, Text |
|
|
17 |
FloatSliderWidget, IntSliderWidget, Check |
|
|
16 | from IPython.html.widgets import (Widget, TextWidget, | |
|
17 | FloatSliderWidget, IntSliderWidget, CheckboxWidget, DropdownWidget, | |
|
18 | 18 | ContainerWidget) |
|
19 | 19 | from IPython.display import display, clear_output |
|
20 | 20 | from IPython.utils.py3compat import string_types, unicode_type |
|
21 | 21 | |
|
22 | 22 | #----------------------------------------------------------------------------- |
|
23 | 23 | # Classes and Functions |
|
24 | 24 | #----------------------------------------------------------------------------- |
|
25 | 25 | |
|
26 | 26 | |
|
27 | 27 | def _matches(o, pattern): |
|
28 | 28 | if not len(o) == len(pattern): |
|
29 | 29 | return False |
|
30 | 30 | comps = zip(o,pattern) |
|
31 | 31 | return all(isinstance(obj,kind) for obj,kind in comps) |
|
32 | 32 | |
|
33 | 33 | |
|
34 |
def _min_max_value( |
|
|
35 | min = o[0] | |
|
36 | max = o[1] | |
|
37 | if not max > min: | |
|
38 | raise ValueError('max must be greater than min: (min={0}, max={1})'.format(min, max)) | |
|
39 |
value = min + abs( |
|
|
34 | def _get_min_max_value(min, max, value): | |
|
35 | """Return min, max, value given input values with possible None.""" | |
|
36 | if value is None: | |
|
37 | if not max > min: | |
|
38 | raise ValueError('max must be greater than min: (min={0}, max={1})'.format(min, max)) | |
|
39 | value = min + abs(min-max)/2 | |
|
40 | value = type(min)(value) | |
|
41 | elif min is None and max is None: | |
|
42 | if value == 0.0: | |
|
43 | min, max, value = 0.0, 1.0, 0.5 | |
|
44 | elif value == 0: | |
|
45 | min, max, value = 0, 1, 0 | |
|
46 | elif isinstance(value, float): | |
|
47 | min, max = -value, 3.0*value | |
|
48 | elif isinstance(value, int): | |
|
49 | min, max = -value, 3*value | |
|
50 | else: | |
|
51 | raise TypeError('expected a number, got: %r' % number) | |
|
52 | else: | |
|
53 | raise ValueError('unable to infer range, value from: ({0}, {1}, {2})'.format(min, max, value)) | |
|
40 | 54 | return min, max, value |
|
41 | 55 | |
|
56 | ||
|
42 | 57 | def _widget_abbrev(o): |
|
43 | 58 | if isinstance(o, string_types): |
|
44 |
return Text |
|
|
59 | return TextWidget(value=unicode_type(o)) | |
|
45 | 60 | elif isinstance(o, dict): |
|
46 |
|
|
|
47 | w = DropdownWidget(value=values[0], values=values) | |
|
48 | w.actual_values = o | |
|
61 | labels = [unicode_type(k) for k in o] | |
|
62 | values = o.values() | |
|
63 | w = DropdownWidget(value=values[0], values=values, labels=labels) | |
|
49 | 64 | return w |
|
50 | 65 | # Special case float and int == 0.0 |
|
51 | 66 | # get_range(value): |
|
52 | 67 | elif isinstance(o, bool): |
|
53 |
return Check |
|
|
68 | return CheckboxWidget(value=o) | |
|
54 | 69 | elif isinstance(o, float): |
|
55 | return FloatSliderWidget(value=o, min=-o, max=3.0*o) | |
|
70 | min, max, value = _get_min_max_value(None, None, o) | |
|
71 | return FloatSliderWidget(value=o, min=min, max=max) | |
|
56 | 72 | elif isinstance(o, int): |
|
57 | return IntSliderWidget(value=o, min=-o, max=3*o) | |
|
73 | min, max, value = _get_min_max_value(None, None, o) | |
|
74 | return IntSliderWidget(value=o, min=min, max=max) | |
|
58 | 75 | if isinstance(o, (list, tuple)): |
|
59 | 76 | if _matches(o, (int, int)): |
|
60 | min, max, value = _min_max_value(o) | |
|
61 |
return IntSliderWidget(value= |
|
|
77 | min, max, value = _get_min_max_value(o[0], o[1], None) | |
|
78 | return IntSliderWidget(value=value, min=min, max=max) | |
|
62 | 79 | elif _matches(o, (int, int, int)): |
|
63 | min, max, value = _min_max_value(o) | |
|
64 |
return IntSliderWidget(value= |
|
|
80 | min, max, value = _get_min_max_value(o[0], o[1], None) | |
|
81 | return IntSliderWidget(value=value, min=min, max=max, step=o[2]) | |
|
65 | 82 | elif _matches(o, (float, float)): |
|
66 | min, max, value = _min_max_value(o) | |
|
83 | min, max, value = _get_min_max_value(o[0], o[1], None) | |
|
67 | 84 | return FloatSliderWidget(value=value, min=min, max=max) |
|
68 | 85 | elif _matches(o, (float, float, float)): |
|
69 | min, max, value = _min_max_value(o) | |
|
86 | min, max, value = _get_min_max_value(o[0], o[1], None) | |
|
70 | 87 | return FloatSliderWidget(value=value, min=min, max=max, step=o[2]) |
|
71 | 88 | elif _matches(o, (float, float, int)): |
|
72 | min, max, value = _min_max_value(o) | |
|
89 | min, max, value = _get_min_max_value(o[0], o[1], None) | |
|
73 | 90 | return FloatSliderWidget(value=value, min=min, max=max, step=float(o[2])) |
|
74 | 91 | elif all(isinstance(x, string_types) for x in o): |
|
75 | 92 | return DropdownWidget(value=unicode_type(o[0]), |
|
76 | 93 | values=[unicode_type(k) for k in o]) |
|
77 | 94 | |
|
78 | 95 | |
|
79 | 96 | def interactive(f, **kwargs): |
|
80 | 97 | """Interact with a function using widgets.""" |
|
81 | 98 | |
|
82 | 99 | co = kwargs.pop('clear_output', True) |
|
83 | 100 | # First convert all args to Widget instances |
|
84 | 101 | widgets = [] |
|
85 | 102 | container = ContainerWidget() |
|
86 | 103 | container.result = None |
|
87 |
container.arg |
|
|
104 | container.kwargs = dict() | |
|
88 | 105 | for key, value in kwargs.items(): |
|
89 | 106 | if isinstance(value, Widget): |
|
90 | 107 | widget = value |
|
91 | 108 | else: |
|
92 | 109 | widget = _widget_abbrev(value) |
|
93 | 110 | if widget is None: |
|
94 | 111 | raise ValueError("Object cannot be transformed to a Widget") |
|
112 | widget.description = key | |
|
95 | 113 | widgets.append((key,widget)) |
|
96 | 114 | widgets.sort(key=lambda e: e[1].__class__.__name__) |
|
97 | 115 | container.children = [e[1] for e in widgets] |
|
98 | 116 | |
|
99 | 117 | # Build the callback |
|
100 | 118 | def call_f(name, old, new): |
|
101 | 119 | actual_kwargs = {} |
|
102 | 120 | for key, widget in widgets: |
|
103 | 121 | value = widget.value |
|
104 | if hasattr(widget, 'actual_values'): | |
|
105 | value = widget.actual_values[value] | |
|
106 | container.arguments[key] = value | |
|
122 | container.kwargs[key] = value | |
|
107 | 123 | actual_kwargs[key] = value |
|
108 | 124 | if co: |
|
109 | 125 | clear_output(wait=True) |
|
110 | 126 | container.result = f(**actual_kwargs) |
|
111 | 127 | |
|
112 | 128 | # Wire up the widgets |
|
113 | 129 | for key, widget in widgets: |
|
114 | 130 | widget.on_trait_change(call_f, 'value') |
|
115 | widget.description = key | |
|
116 | 131 | |
|
117 | container.on_displayed(lambda : call_f(None, None, None)) | |
|
132 | container.on_displayed(lambda _: call_f(None, None, None)) | |
|
118 | 133 | |
|
119 | 134 | return container |
|
120 | 135 | |
|
121 | 136 | def interact(f, **kwargs): |
|
122 | 137 | w = interactive(f, **kwargs) |
|
123 | 138 | display(w) |
|
1 | NO CONTENT: modified file | |
The requested commit or file is too big and content was truncated. Show full diff |
General Comments 0
You need to be logged in to leave comments.
Login now