|
33 | 33 | 'ErrorZ': {'object_name': 'error_z', 'base_type': dict},
|
34 | 34 | 'Figure': {'object_name': 'figure', 'base_type': dict},
|
35 | 35 | 'Font': {'object_name': 'font', 'base_type': dict},
|
36 |
| - 'Frames': {'object_name': 'frames', 'base_type': dict}, |
| 36 | + 'Frames': {'object_name': 'frames', 'base_type': list}, |
37 | 37 | 'Heatmap': {'object_name': 'heatmap', 'base_type': dict},
|
38 | 38 | 'Histogram': {'object_name': 'histogram', 'base_type': dict},
|
39 | 39 | 'Histogram2d': {'object_name': 'histogram2d', 'base_type': dict},
|
@@ -68,9 +68,62 @@ def get_graph_reference():
|
68 | 68 | """
|
69 | 69 | path = os.path.join('package_data', 'default-schema.json')
|
70 | 70 | s = resource_string('plotly', path).decode('utf-8')
|
71 |
| - graph_reference = _json.loads(s) |
| 71 | + graph_reference = utils.decode_unicode(_json.loads(s)) |
| 72 | + |
| 73 | + # TODO: Patch in frames info until it hits streambed. See #659 |
| 74 | + graph_reference['frames'] = { |
| 75 | + "items": { |
| 76 | + "frames_entry": { |
| 77 | + "baseframe": { |
| 78 | + "description": "The name of the frame into which this " |
| 79 | + "frame's properties are merged before " |
| 80 | + "applying. This is used to unify " |
| 81 | + "properties and avoid needing to specify " |
| 82 | + "the same values for the same properties " |
| 83 | + "in multiple frames.", |
| 84 | + "role": "info", |
| 85 | + "valType": "string" |
| 86 | + }, |
| 87 | + "data": { |
| 88 | + "description": "A list of traces this frame modifies. " |
| 89 | + "The format is identical to the normal " |
| 90 | + "trace definition.", |
| 91 | + "role": "object", |
| 92 | + "valType": "any" |
| 93 | + }, |
| 94 | + "group": { |
| 95 | + "description": "An identifier that specifies the group " |
| 96 | + "to which the frame belongs, used by " |
| 97 | + "animate to select a subset of frames.", |
| 98 | + "role": "info", |
| 99 | + "valType": "string" |
| 100 | + }, |
| 101 | + "layout": { |
| 102 | + "role": "object", |
| 103 | + "description": "Layout properties which this frame " |
| 104 | + "modifies. The format is identical to " |
| 105 | + "the normal layout definition.", |
| 106 | + "valType": "any" |
| 107 | + }, |
| 108 | + "name": { |
| 109 | + "description": "A label by which to identify the frame", |
| 110 | + "role": "info", |
| 111 | + "valType": "string" |
| 112 | + }, |
| 113 | + "role": "object", |
| 114 | + "traces": { |
| 115 | + "description": "A list of trace indices that identify " |
| 116 | + "the respective traces in the data " |
| 117 | + "attribute", |
| 118 | + "role": "info", |
| 119 | + "valType": "info_array" |
| 120 | + } |
| 121 | + } |
| 122 | + }, |
| 123 | + "role": "object" |
| 124 | + } |
72 | 125 |
|
73 |
| - return utils.decode_unicode(graph_reference) |
| 126 | + return graph_reference |
74 | 127 |
|
75 | 128 |
|
76 | 129 | def string_to_class_name(string):
|
@@ -136,6 +189,27 @@ def get_attributes_dicts(object_name, parent_object_names=()):
|
136 | 189 | # We should also one or more paths where attributes are defined.
|
137 | 190 | attribute_paths = list(object_dict['attribute_paths']) # shallow copy
|
138 | 191 |
|
| 192 | + # Map frame 'data' and 'layout' to previously-defined figure attributes. |
| 193 | + # Examples of parent_object_names changes: |
| 194 | + # ['figure', 'frames'] --> ['figure', 'frames'] |
| 195 | + # ['figure', 'frames', FRAME_NAME] --> ['figure'] |
| 196 | + # ['figure', 'frames', FRAME_NAME, 'data'] --> ['figure', 'data'] |
| 197 | + # ['figure', 'frames', FRAME_NAME, 'layout'] --> ['figure', 'layout'] |
| 198 | + # ['figure', 'frames', FRAME_NAME, 'foo'] --> |
| 199 | + # ['figure', 'frames', FRAME_NAME, 'foo'] |
| 200 | + # [FRAME_NAME, 'layout'] --> ['figure', 'layout'] |
| 201 | + if FRAME_NAME in parent_object_names: |
| 202 | + len_parent_object_names = len(parent_object_names) |
| 203 | + index = parent_object_names.index(FRAME_NAME) |
| 204 | + if len_parent_object_names == index + 1: |
| 205 | + if object_name in ('data', 'layout'): |
| 206 | + parent_object_names = ['figure', object_name] |
| 207 | + elif len_parent_object_names > index + 1: |
| 208 | + if parent_object_names[index + 1] in ('data', 'layout'): |
| 209 | + parent_object_names = ( |
| 210 | + ['figure'] + list(parent_object_names)[index + 1:] |
| 211 | + ) |
| 212 | + |
139 | 213 | # If we have parent_names, some of these attribute paths may be invalid.
|
140 | 214 | for parent_object_name in reversed(parent_object_names):
|
141 | 215 | if parent_object_name in ARRAYS:
|
@@ -410,8 +484,11 @@ def _patch_objects():
|
410 | 484 | 'attribute_paths': layout_attribute_paths,
|
411 | 485 | 'additional_attributes': {}}
|
412 | 486 |
|
413 |
| - figure_attributes = {'layout': {'role': 'object'}, |
414 |
| - 'data': {'role': 'object', '_isLinkedToArray': True}} |
| 487 | + figure_attributes = { |
| 488 | + 'layout': {'role': 'object'}, |
| 489 | + 'data': {'role': 'object', '_isLinkedToArray': True}, |
| 490 | + 'frames': {'role': 'object', '_isLinkedToArray': True} |
| 491 | + } |
415 | 492 | OBJECTS['figure'] = {'meta_paths': [],
|
416 | 493 | 'attribute_paths': [],
|
417 | 494 | 'additional_attributes': figure_attributes}
|
@@ -479,6 +556,8 @@ def _get_classes():
|
479 | 556 | # The ordering here is important.
|
480 | 557 | GRAPH_REFERENCE = get_graph_reference()
|
481 | 558 |
|
| 559 | +FRAME_NAME = list(GRAPH_REFERENCE['frames']['items'].keys())[0] |
| 560 | + |
482 | 561 | # See http://blog.labix.org/2008/06/27/watch-out-for-listdictkeys-in-python-3
|
483 | 562 | TRACE_NAMES = list(GRAPH_REFERENCE['traces'].keys())
|
484 | 563 |
|
|
0 commit comments