figrecipe 0.5.0__py3-none-any.whl → 0.7.4__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (189) hide show
  1. figrecipe/__init__.py +220 -819
  2. figrecipe/_api/__init__.py +48 -0
  3. figrecipe/_api/_extract.py +108 -0
  4. figrecipe/_api/_notebook.py +61 -0
  5. figrecipe/_api/_panel.py +46 -0
  6. figrecipe/_api/_save.py +191 -0
  7. figrecipe/_api/_seaborn_proxy.py +34 -0
  8. figrecipe/_api/_style_manager.py +153 -0
  9. figrecipe/_api/_subplots.py +333 -0
  10. figrecipe/_api/_validate.py +82 -0
  11. figrecipe/_dev/__init__.py +29 -0
  12. figrecipe/_dev/_plotters.py +76 -0
  13. figrecipe/_dev/_run_demos.py +56 -0
  14. figrecipe/_dev/demo_plotters/__init__.py +64 -0
  15. figrecipe/_dev/demo_plotters/_categories.py +81 -0
  16. figrecipe/_dev/demo_plotters/_figure_creators.py +119 -0
  17. figrecipe/_dev/demo_plotters/_helpers.py +31 -0
  18. figrecipe/_dev/demo_plotters/_registry.py +50 -0
  19. figrecipe/_dev/demo_plotters/bar_categorical/__init__.py +4 -0
  20. figrecipe/_dev/demo_plotters/bar_categorical/plot_bar.py +25 -0
  21. figrecipe/_dev/demo_plotters/bar_categorical/plot_barh.py +25 -0
  22. figrecipe/_dev/demo_plotters/contour_surface/__init__.py +4 -0
  23. figrecipe/_dev/demo_plotters/contour_surface/plot_contour.py +30 -0
  24. figrecipe/_dev/demo_plotters/contour_surface/plot_contourf.py +29 -0
  25. figrecipe/_dev/demo_plotters/contour_surface/plot_tricontour.py +28 -0
  26. figrecipe/_dev/demo_plotters/contour_surface/plot_tricontourf.py +28 -0
  27. figrecipe/_dev/demo_plotters/contour_surface/plot_tripcolor.py +29 -0
  28. figrecipe/_dev/demo_plotters/contour_surface/plot_triplot.py +25 -0
  29. figrecipe/_dev/demo_plotters/distribution/__init__.py +4 -0
  30. figrecipe/_dev/demo_plotters/distribution/plot_boxplot.py +24 -0
  31. figrecipe/_dev/demo_plotters/distribution/plot_ecdf.py +24 -0
  32. figrecipe/_dev/demo_plotters/distribution/plot_hist.py +24 -0
  33. figrecipe/_dev/demo_plotters/distribution/plot_hist2d.py +25 -0
  34. figrecipe/_dev/demo_plotters/distribution/plot_violinplot.py +25 -0
  35. figrecipe/_dev/demo_plotters/image_matrix/__init__.py +4 -0
  36. figrecipe/_dev/demo_plotters/image_matrix/plot_hexbin.py +25 -0
  37. figrecipe/_dev/demo_plotters/image_matrix/plot_imshow.py +23 -0
  38. figrecipe/_dev/demo_plotters/image_matrix/plot_matshow.py +23 -0
  39. figrecipe/_dev/demo_plotters/image_matrix/plot_pcolor.py +29 -0
  40. figrecipe/_dev/demo_plotters/image_matrix/plot_pcolormesh.py +29 -0
  41. figrecipe/_dev/demo_plotters/image_matrix/plot_spy.py +29 -0
  42. figrecipe/_dev/demo_plotters/line_curve/__init__.py +4 -0
  43. figrecipe/_dev/demo_plotters/line_curve/plot_errorbar.py +28 -0
  44. figrecipe/_dev/demo_plotters/line_curve/plot_fill.py +29 -0
  45. figrecipe/_dev/demo_plotters/line_curve/plot_fill_between.py +30 -0
  46. figrecipe/_dev/demo_plotters/line_curve/plot_fill_betweenx.py +28 -0
  47. figrecipe/_dev/demo_plotters/line_curve/plot_plot.py +28 -0
  48. figrecipe/_dev/demo_plotters/line_curve/plot_stackplot.py +29 -0
  49. figrecipe/_dev/demo_plotters/line_curve/plot_stairs.py +27 -0
  50. figrecipe/_dev/demo_plotters/line_curve/plot_step.py +27 -0
  51. figrecipe/_dev/demo_plotters/scatter_points/__init__.py +4 -0
  52. figrecipe/_dev/demo_plotters/scatter_points/plot_scatter.py +24 -0
  53. figrecipe/_dev/demo_plotters/special/__init__.py +4 -0
  54. figrecipe/_dev/demo_plotters/special/plot_eventplot.py +25 -0
  55. figrecipe/_dev/demo_plotters/special/plot_loglog.py +27 -0
  56. figrecipe/_dev/demo_plotters/special/plot_pie.py +27 -0
  57. figrecipe/_dev/demo_plotters/special/plot_semilogx.py +27 -0
  58. figrecipe/_dev/demo_plotters/special/plot_semilogy.py +27 -0
  59. figrecipe/_dev/demo_plotters/special/plot_stem.py +27 -0
  60. figrecipe/_dev/demo_plotters/spectral_signal/__init__.py +4 -0
  61. figrecipe/_dev/demo_plotters/spectral_signal/plot_acorr.py +24 -0
  62. figrecipe/_dev/demo_plotters/spectral_signal/plot_angle_spectrum.py +28 -0
  63. figrecipe/_dev/demo_plotters/spectral_signal/plot_cohere.py +29 -0
  64. figrecipe/_dev/demo_plotters/spectral_signal/plot_csd.py +29 -0
  65. figrecipe/_dev/demo_plotters/spectral_signal/plot_magnitude_spectrum.py +28 -0
  66. figrecipe/_dev/demo_plotters/spectral_signal/plot_phase_spectrum.py +28 -0
  67. figrecipe/_dev/demo_plotters/spectral_signal/plot_psd.py +29 -0
  68. figrecipe/_dev/demo_plotters/spectral_signal/plot_specgram.py +30 -0
  69. figrecipe/_dev/demo_plotters/spectral_signal/plot_xcorr.py +25 -0
  70. figrecipe/_dev/demo_plotters/vector_flow/__init__.py +4 -0
  71. figrecipe/_dev/demo_plotters/vector_flow/plot_barbs.py +30 -0
  72. figrecipe/_dev/demo_plotters/vector_flow/plot_quiver.py +30 -0
  73. figrecipe/_dev/demo_plotters/vector_flow/plot_streamplot.py +30 -0
  74. figrecipe/_editor/__init__.py +278 -0
  75. figrecipe/_editor/_bbox/__init__.py +43 -0
  76. figrecipe/_editor/_bbox/_collections.py +177 -0
  77. figrecipe/_editor/_bbox/_elements.py +159 -0
  78. figrecipe/_editor/_bbox/_extract.py +256 -0
  79. figrecipe/_editor/_bbox/_extract_axes.py +370 -0
  80. figrecipe/_editor/_bbox/_extract_text.py +342 -0
  81. figrecipe/_editor/_bbox/_lines.py +173 -0
  82. figrecipe/_editor/_bbox/_transforms.py +146 -0
  83. figrecipe/_editor/_flask_app.py +258 -0
  84. figrecipe/_editor/_helpers.py +242 -0
  85. figrecipe/_editor/_hitmap/__init__.py +76 -0
  86. figrecipe/_editor/_hitmap/_artists/__init__.py +21 -0
  87. figrecipe/_editor/_hitmap/_artists/_collections.py +345 -0
  88. figrecipe/_editor/_hitmap/_artists/_images.py +68 -0
  89. figrecipe/_editor/_hitmap/_artists/_lines.py +107 -0
  90. figrecipe/_editor/_hitmap/_artists/_patches.py +163 -0
  91. figrecipe/_editor/_hitmap/_artists/_text.py +190 -0
  92. figrecipe/_editor/_hitmap/_colors.py +181 -0
  93. figrecipe/_editor/_hitmap/_detect.py +137 -0
  94. figrecipe/_editor/_hitmap/_restore.py +154 -0
  95. figrecipe/_editor/_hitmap_main.py +182 -0
  96. figrecipe/_editor/_overrides.py +318 -0
  97. figrecipe/_editor/_preferences.py +135 -0
  98. figrecipe/_editor/_render_overrides.py +480 -0
  99. figrecipe/_editor/_renderer.py +199 -0
  100. figrecipe/_editor/_routes_axis.py +453 -0
  101. figrecipe/_editor/_routes_core.py +284 -0
  102. figrecipe/_editor/_routes_element.py +317 -0
  103. figrecipe/_editor/_routes_style.py +223 -0
  104. figrecipe/_editor/_templates/__init__.py +152 -0
  105. figrecipe/_editor/_templates/_html.py +502 -0
  106. figrecipe/_editor/_templates/_scripts/__init__.py +120 -0
  107. figrecipe/_editor/_templates/_scripts/_api.py +228 -0
  108. figrecipe/_editor/_templates/_scripts/_colors.py +485 -0
  109. figrecipe/_editor/_templates/_scripts/_core.py +436 -0
  110. figrecipe/_editor/_templates/_scripts/_debug_snapshot.py +186 -0
  111. figrecipe/_editor/_templates/_scripts/_element_editor.py +310 -0
  112. figrecipe/_editor/_templates/_scripts/_files.py +195 -0
  113. figrecipe/_editor/_templates/_scripts/_hitmap.py +509 -0
  114. figrecipe/_editor/_templates/_scripts/_inspector.py +315 -0
  115. figrecipe/_editor/_templates/_scripts/_labels.py +464 -0
  116. figrecipe/_editor/_templates/_scripts/_legend_drag.py +265 -0
  117. figrecipe/_editor/_templates/_scripts/_modals.py +226 -0
  118. figrecipe/_editor/_templates/_scripts/_overlays.py +292 -0
  119. figrecipe/_editor/_templates/_scripts/_panel_drag.py +334 -0
  120. figrecipe/_editor/_templates/_scripts/_panel_position.py +279 -0
  121. figrecipe/_editor/_templates/_scripts/_selection.py +237 -0
  122. figrecipe/_editor/_templates/_scripts/_tabs.py +89 -0
  123. figrecipe/_editor/_templates/_scripts/_view_mode.py +107 -0
  124. figrecipe/_editor/_templates/_scripts/_zoom.py +179 -0
  125. figrecipe/_editor/_templates/_styles/__init__.py +69 -0
  126. figrecipe/_editor/_templates/_styles/_base.py +64 -0
  127. figrecipe/_editor/_templates/_styles/_buttons.py +206 -0
  128. figrecipe/_editor/_templates/_styles/_color_input.py +123 -0
  129. figrecipe/_editor/_templates/_styles/_controls.py +265 -0
  130. figrecipe/_editor/_templates/_styles/_dynamic_props.py +144 -0
  131. figrecipe/_editor/_templates/_styles/_forms.py +126 -0
  132. figrecipe/_editor/_templates/_styles/_hitmap.py +184 -0
  133. figrecipe/_editor/_templates/_styles/_inspector.py +90 -0
  134. figrecipe/_editor/_templates/_styles/_labels.py +118 -0
  135. figrecipe/_editor/_templates/_styles/_modals.py +98 -0
  136. figrecipe/_editor/_templates/_styles/_overlays.py +130 -0
  137. figrecipe/_editor/_templates/_styles/_preview.py +225 -0
  138. figrecipe/_editor/_templates/_styles/_selection.py +73 -0
  139. figrecipe/_params/_DECORATION_METHODS.py +33 -0
  140. figrecipe/_params/_PLOTTING_METHODS.py +58 -0
  141. figrecipe/_params/__init__.py +9 -0
  142. figrecipe/_recorder.py +92 -110
  143. figrecipe/_recorder_utils.py +124 -0
  144. figrecipe/_reproducer/__init__.py +18 -0
  145. figrecipe/_reproducer/_core.py +498 -0
  146. figrecipe/_reproducer/_custom_plots.py +279 -0
  147. figrecipe/_reproducer/_seaborn.py +100 -0
  148. figrecipe/_reproducer/_violin.py +186 -0
  149. figrecipe/_seaborn.py +14 -9
  150. figrecipe/_serializer.py +2 -2
  151. figrecipe/_signatures/README.md +68 -0
  152. figrecipe/_signatures/__init__.py +12 -2
  153. figrecipe/_signatures/_kwargs.py +273 -0
  154. figrecipe/_signatures/_loader.py +114 -57
  155. figrecipe/_signatures/_parsing.py +147 -0
  156. figrecipe/_utils/__init__.py +6 -4
  157. figrecipe/_utils/_crop.py +10 -4
  158. figrecipe/_utils/_image_diff.py +37 -33
  159. figrecipe/_utils/_numpy_io.py +0 -1
  160. figrecipe/_utils/_units.py +11 -3
  161. figrecipe/_validator.py +12 -3
  162. figrecipe/_wrappers/_axes.py +193 -170
  163. figrecipe/_wrappers/_axes_helpers.py +136 -0
  164. figrecipe/_wrappers/_axes_plots.py +418 -0
  165. figrecipe/_wrappers/_axes_seaborn.py +157 -0
  166. figrecipe/_wrappers/_figure.py +277 -18
  167. figrecipe/_wrappers/_panel_labels.py +127 -0
  168. figrecipe/_wrappers/_plot_helpers.py +143 -0
  169. figrecipe/_wrappers/_violin_helpers.py +180 -0
  170. figrecipe/plt.py +0 -1
  171. figrecipe/pyplot.py +2 -1
  172. figrecipe/styles/__init__.py +12 -11
  173. figrecipe/styles/_dotdict.py +72 -0
  174. figrecipe/styles/_finalize.py +134 -0
  175. figrecipe/styles/_fonts.py +77 -0
  176. figrecipe/styles/_kwargs_converter.py +178 -0
  177. figrecipe/styles/_plot_styles.py +209 -0
  178. figrecipe/styles/_style_applier.py +60 -202
  179. figrecipe/styles/_style_loader.py +73 -121
  180. figrecipe/styles/_themes.py +151 -0
  181. figrecipe/styles/presets/MATPLOTLIB.yaml +95 -0
  182. figrecipe/styles/presets/SCITEX.yaml +181 -0
  183. figrecipe-0.7.4.dist-info/METADATA +429 -0
  184. figrecipe-0.7.4.dist-info/RECORD +188 -0
  185. figrecipe/_reproducer.py +0 -358
  186. figrecipe-0.5.0.dist-info/METADATA +0 -336
  187. figrecipe-0.5.0.dist-info/RECORD +0 -26
  188. {figrecipe-0.5.0.dist-info → figrecipe-0.7.4.dist-info}/WHEEL +0 -0
  189. {figrecipe-0.5.0.dist-info → figrecipe-0.7.4.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,370 @@
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ """Axes element bbox extraction (lines, collections, patches, images)."""
4
+
5
+ from matplotlib.collections import PathCollection, PolyCollection
6
+ from matplotlib.patches import Rectangle
7
+
8
+ from ._collections import get_collection_bbox, get_patch_bbox
9
+ from ._elements import get_element_bbox
10
+ from ._lines import get_line_bbox, get_quiver_bbox
11
+
12
+
13
+ def extract_lines(
14
+ ax,
15
+ ax_idx,
16
+ fig,
17
+ renderer,
18
+ tight_bbox,
19
+ img_width,
20
+ img_height,
21
+ scale_x,
22
+ scale_y,
23
+ pad_inches,
24
+ saved_height_inches,
25
+ include_points,
26
+ bboxes,
27
+ ):
28
+ """Extract bboxes for line elements."""
29
+ for i, line in enumerate(ax.get_lines()):
30
+ if not line.get_visible():
31
+ continue
32
+ xdata = line.get_xdata()
33
+ ydata = line.get_ydata()
34
+ if len(xdata) == 0 or len(ydata) == 0:
35
+ continue
36
+ if line.axes != ax:
37
+ continue
38
+
39
+ bbox = get_line_bbox(
40
+ line,
41
+ ax,
42
+ fig,
43
+ renderer,
44
+ tight_bbox,
45
+ img_width,
46
+ img_height,
47
+ scale_x,
48
+ scale_y,
49
+ pad_inches,
50
+ saved_height_inches,
51
+ include_points,
52
+ )
53
+ if bbox:
54
+ bboxes[f"ax{ax_idx}_line{i}"] = {
55
+ **bbox,
56
+ "type": "line",
57
+ "label": line.get_label() or f"line_{i}",
58
+ "ax_index": ax_idx,
59
+ }
60
+
61
+
62
+ def extract_collections(
63
+ ax,
64
+ ax_idx,
65
+ fig,
66
+ renderer,
67
+ tight_bbox,
68
+ img_width,
69
+ img_height,
70
+ scale_x,
71
+ scale_y,
72
+ pad_inches,
73
+ saved_height_inches,
74
+ include_points,
75
+ bboxes,
76
+ ):
77
+ """Extract bboxes for collection elements (scatter, fill, quiver, etc.)."""
78
+ from matplotlib.collections import LineCollection, QuadMesh
79
+ from matplotlib.contour import QuadContourSet
80
+ from matplotlib.quiver import Quiver
81
+
82
+ scatter_idx = 0
83
+ for i, coll in enumerate(ax.collections):
84
+ if isinstance(coll, Quiver):
85
+ if not coll.get_visible():
86
+ continue
87
+ bbox = get_quiver_bbox(
88
+ coll,
89
+ ax,
90
+ fig,
91
+ renderer,
92
+ tight_bbox,
93
+ img_width,
94
+ img_height,
95
+ scale_x,
96
+ scale_y,
97
+ pad_inches,
98
+ saved_height_inches,
99
+ )
100
+ if bbox:
101
+ bboxes[f"ax{ax_idx}_quiver{i}"] = {
102
+ **bbox,
103
+ "type": "quiver",
104
+ "label": coll.get_label() or f"quiver_{i}",
105
+ "ax_index": ax_idx,
106
+ }
107
+
108
+ elif isinstance(coll, PathCollection):
109
+ if not coll.get_visible():
110
+ continue
111
+ bbox = get_collection_bbox(
112
+ coll,
113
+ ax,
114
+ fig,
115
+ renderer,
116
+ tight_bbox,
117
+ img_width,
118
+ img_height,
119
+ scale_x,
120
+ scale_y,
121
+ pad_inches,
122
+ saved_height_inches,
123
+ include_points,
124
+ )
125
+ if bbox:
126
+ bboxes[f"ax{ax_idx}_scatter{scatter_idx}"] = {
127
+ **bbox,
128
+ "type": "scatter",
129
+ "label": coll.get_label() or f"scatter_{scatter_idx}",
130
+ "ax_index": ax_idx,
131
+ }
132
+ scatter_idx += 1
133
+
134
+ elif isinstance(coll, PolyCollection):
135
+ if not coll.get_visible():
136
+ continue
137
+ bbox = get_collection_bbox(
138
+ coll,
139
+ ax,
140
+ fig,
141
+ renderer,
142
+ tight_bbox,
143
+ img_width,
144
+ img_height,
145
+ scale_x,
146
+ scale_y,
147
+ pad_inches,
148
+ saved_height_inches,
149
+ False,
150
+ )
151
+ if bbox:
152
+ bboxes[f"ax{ax_idx}_fill{i}"] = {
153
+ **bbox,
154
+ "type": "fill",
155
+ "label": coll.get_label() or f"fill_{i}",
156
+ "ax_index": ax_idx,
157
+ }
158
+
159
+ elif isinstance(coll, LineCollection):
160
+ if not coll.get_visible():
161
+ continue
162
+ bbox = get_collection_bbox(
163
+ coll,
164
+ ax,
165
+ fig,
166
+ renderer,
167
+ tight_bbox,
168
+ img_width,
169
+ img_height,
170
+ scale_x,
171
+ scale_y,
172
+ pad_inches,
173
+ saved_height_inches,
174
+ False,
175
+ )
176
+ if bbox:
177
+ bboxes[f"ax{ax_idx}_linecoll{i}"] = {
178
+ **bbox,
179
+ "type": "linecollection",
180
+ "label": coll.get_label() or f"linecoll_{i}",
181
+ "ax_index": ax_idx,
182
+ }
183
+
184
+ elif isinstance(coll, QuadMesh):
185
+ if not coll.get_visible():
186
+ continue
187
+ bbox = get_collection_bbox(
188
+ coll,
189
+ ax,
190
+ fig,
191
+ renderer,
192
+ tight_bbox,
193
+ img_width,
194
+ img_height,
195
+ scale_x,
196
+ scale_y,
197
+ pad_inches,
198
+ saved_height_inches,
199
+ False,
200
+ )
201
+ if bbox:
202
+ bboxes[f"ax{ax_idx}_quadmesh{i}"] = {
203
+ **bbox,
204
+ "type": "quadmesh",
205
+ "label": f"quadmesh_{i}",
206
+ "ax_index": ax_idx,
207
+ }
208
+
209
+ elif isinstance(coll, QuadContourSet):
210
+ bbox = get_element_bbox(
211
+ ax,
212
+ fig,
213
+ renderer,
214
+ tight_bbox,
215
+ img_width,
216
+ img_height,
217
+ scale_x,
218
+ scale_y,
219
+ pad_inches,
220
+ saved_height_inches,
221
+ )
222
+ if bbox:
223
+ bboxes[f"ax{ax_idx}_contour{i}"] = {
224
+ **bbox,
225
+ "type": "contour",
226
+ "label": f"contour_{i}",
227
+ "ax_index": ax_idx,
228
+ }
229
+
230
+
231
+ def extract_patches(
232
+ ax,
233
+ ax_idx,
234
+ fig,
235
+ renderer,
236
+ tight_bbox,
237
+ img_width,
238
+ img_height,
239
+ scale_x,
240
+ scale_y,
241
+ pad_inches,
242
+ saved_height_inches,
243
+ bboxes,
244
+ ):
245
+ """Extract bboxes for patch elements (bars, wedges, polygons)."""
246
+ from matplotlib.patches import Polygon, Wedge
247
+
248
+ bar_idx = 0
249
+ for i, patch in enumerate(ax.patches):
250
+ if isinstance(patch, Rectangle):
251
+ if not patch.get_visible():
252
+ continue
253
+ if patch.get_width() == 1.0 and patch.get_height() == 1.0:
254
+ continue
255
+ bbox = get_patch_bbox(
256
+ patch,
257
+ ax,
258
+ fig,
259
+ renderer,
260
+ tight_bbox,
261
+ img_width,
262
+ img_height,
263
+ scale_x,
264
+ scale_y,
265
+ pad_inches,
266
+ saved_height_inches,
267
+ )
268
+ if bbox:
269
+ bboxes[f"ax{ax_idx}_bar{bar_idx}"] = {
270
+ **bbox,
271
+ "type": "bar",
272
+ "label": patch.get_label() or f"bar_{bar_idx}",
273
+ "ax_index": ax_idx,
274
+ }
275
+ bar_idx += 1
276
+
277
+ elif isinstance(patch, Wedge):
278
+ if not patch.get_visible():
279
+ continue
280
+ bbox = get_patch_bbox(
281
+ patch,
282
+ ax,
283
+ fig,
284
+ renderer,
285
+ tight_bbox,
286
+ img_width,
287
+ img_height,
288
+ scale_x,
289
+ scale_y,
290
+ pad_inches,
291
+ saved_height_inches,
292
+ )
293
+ if bbox:
294
+ bboxes[f"ax{ax_idx}_wedge{i}"] = {
295
+ **bbox,
296
+ "type": "pie",
297
+ "label": patch.get_label() or f"wedge_{i}",
298
+ "ax_index": ax_idx,
299
+ }
300
+
301
+ elif isinstance(patch, Polygon):
302
+ if not patch.get_visible():
303
+ continue
304
+ bbox = get_patch_bbox(
305
+ patch,
306
+ ax,
307
+ fig,
308
+ renderer,
309
+ tight_bbox,
310
+ img_width,
311
+ img_height,
312
+ scale_x,
313
+ scale_y,
314
+ pad_inches,
315
+ saved_height_inches,
316
+ )
317
+ if bbox:
318
+ bboxes[f"ax{ax_idx}_polygon{i}"] = {
319
+ **bbox,
320
+ "type": "fill",
321
+ "label": patch.get_label() or f"fill_{i}",
322
+ "ax_index": ax_idx,
323
+ }
324
+
325
+
326
+ def extract_images(
327
+ ax,
328
+ ax_idx,
329
+ fig,
330
+ renderer,
331
+ tight_bbox,
332
+ img_width,
333
+ img_height,
334
+ scale_x,
335
+ scale_y,
336
+ pad_inches,
337
+ saved_height_inches,
338
+ bboxes,
339
+ ):
340
+ """Extract bboxes for image elements."""
341
+ from matplotlib.image import AxesImage
342
+
343
+ for i, img in enumerate(ax.images):
344
+ if isinstance(img, AxesImage):
345
+ if not img.get_visible():
346
+ continue
347
+ bbox = get_element_bbox(
348
+ img,
349
+ fig,
350
+ renderer,
351
+ tight_bbox,
352
+ img_width,
353
+ img_height,
354
+ scale_x,
355
+ scale_y,
356
+ pad_inches,
357
+ saved_height_inches,
358
+ )
359
+ if bbox:
360
+ bboxes[f"ax{ax_idx}_image{i}"] = {
361
+ **bbox,
362
+ "type": "image",
363
+ "label": img.get_label() or f"image_{i}",
364
+ "ax_index": ax_idx,
365
+ }
366
+
367
+
368
+ __all__ = ["extract_lines", "extract_collections", "extract_patches", "extract_images"]
369
+
370
+ # EOF
@@ -0,0 +1,342 @@
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ """Text, legend, spine, and figure text bbox extraction."""
4
+
5
+ from ._elements import get_text_bbox, get_tick_labels_bbox
6
+ from ._transforms import transform_bbox
7
+
8
+
9
+ def extract_text_elements(
10
+ ax,
11
+ ax_idx,
12
+ fig,
13
+ renderer,
14
+ tight_bbox,
15
+ img_width,
16
+ img_height,
17
+ scale_x,
18
+ scale_y,
19
+ pad_inches,
20
+ saved_height_inches,
21
+ bboxes,
22
+ ):
23
+ """Extract bboxes for text elements (title, labels, ticks)."""
24
+ # Title
25
+ title = ax.get_title()
26
+ if title:
27
+ bbox = get_text_bbox(
28
+ ax.title,
29
+ fig,
30
+ renderer,
31
+ tight_bbox,
32
+ img_width,
33
+ img_height,
34
+ scale_x,
35
+ scale_y,
36
+ pad_inches,
37
+ saved_height_inches,
38
+ )
39
+ if bbox:
40
+ bboxes[f"ax{ax_idx}_title"] = {
41
+ **bbox,
42
+ "type": "title",
43
+ "label": "title",
44
+ "ax_index": ax_idx,
45
+ "text": title,
46
+ }
47
+
48
+ # X label
49
+ xlabel = ax.get_xlabel()
50
+ if xlabel:
51
+ bbox = get_text_bbox(
52
+ ax.xaxis.label,
53
+ fig,
54
+ renderer,
55
+ tight_bbox,
56
+ img_width,
57
+ img_height,
58
+ scale_x,
59
+ scale_y,
60
+ pad_inches,
61
+ saved_height_inches,
62
+ )
63
+ if bbox:
64
+ bboxes[f"ax{ax_idx}_xlabel"] = {
65
+ **bbox,
66
+ "type": "xlabel",
67
+ "label": "xlabel",
68
+ "ax_index": ax_idx,
69
+ "text": xlabel,
70
+ }
71
+
72
+ # X tick labels
73
+ xtick_bbox = get_tick_labels_bbox(
74
+ ax.xaxis,
75
+ "x",
76
+ fig,
77
+ renderer,
78
+ tight_bbox,
79
+ img_width,
80
+ img_height,
81
+ scale_x,
82
+ scale_y,
83
+ pad_inches,
84
+ saved_height_inches,
85
+ )
86
+ if xtick_bbox:
87
+ bboxes[f"ax{ax_idx}_xticks"] = {
88
+ **xtick_bbox,
89
+ "type": "xticks",
90
+ "label": "x tick labels",
91
+ "ax_index": ax_idx,
92
+ }
93
+
94
+ # Y label
95
+ ylabel = ax.get_ylabel()
96
+ if ylabel:
97
+ bbox = get_text_bbox(
98
+ ax.yaxis.label,
99
+ fig,
100
+ renderer,
101
+ tight_bbox,
102
+ img_width,
103
+ img_height,
104
+ scale_x,
105
+ scale_y,
106
+ pad_inches,
107
+ saved_height_inches,
108
+ )
109
+ if bbox:
110
+ bboxes[f"ax{ax_idx}_ylabel"] = {
111
+ **bbox,
112
+ "type": "ylabel",
113
+ "label": "ylabel",
114
+ "ax_index": ax_idx,
115
+ "text": ylabel,
116
+ }
117
+
118
+ # Y tick labels
119
+ ytick_bbox = get_tick_labels_bbox(
120
+ ax.yaxis,
121
+ "y",
122
+ fig,
123
+ renderer,
124
+ tight_bbox,
125
+ img_width,
126
+ img_height,
127
+ scale_x,
128
+ scale_y,
129
+ pad_inches,
130
+ saved_height_inches,
131
+ )
132
+ if ytick_bbox:
133
+ bboxes[f"ax{ax_idx}_yticks"] = {
134
+ **ytick_bbox,
135
+ "type": "yticks",
136
+ "label": "y tick labels",
137
+ "ax_index": ax_idx,
138
+ }
139
+
140
+
141
+ def extract_legend(
142
+ ax,
143
+ ax_idx,
144
+ fig,
145
+ renderer,
146
+ tight_bbox,
147
+ img_width,
148
+ img_height,
149
+ scale_x,
150
+ scale_y,
151
+ pad_inches,
152
+ saved_height_inches,
153
+ bboxes,
154
+ ):
155
+ """Extract bbox for legend."""
156
+ legend = ax.get_legend()
157
+ if legend is not None and legend.get_visible():
158
+ try:
159
+ legend_bbox = legend.get_window_extent(renderer)
160
+ if legend_bbox is not None:
161
+ bbox = transform_bbox(
162
+ legend_bbox,
163
+ fig,
164
+ tight_bbox,
165
+ img_width,
166
+ img_height,
167
+ scale_x,
168
+ scale_y,
169
+ pad_inches,
170
+ saved_height_inches,
171
+ )
172
+ if bbox:
173
+ bboxes[f"ax{ax_idx}_legend"] = {
174
+ **bbox,
175
+ "type": "legend",
176
+ "label": "legend",
177
+ "ax_index": ax_idx,
178
+ }
179
+ except Exception:
180
+ pass
181
+
182
+
183
+ def extract_spines(
184
+ ax,
185
+ ax_idx,
186
+ fig,
187
+ renderer,
188
+ tight_bbox,
189
+ img_width,
190
+ img_height,
191
+ scale_x,
192
+ scale_y,
193
+ pad_inches,
194
+ saved_height_inches,
195
+ bboxes,
196
+ ):
197
+ """Extract bboxes for spines with padding for easier clicking."""
198
+ spine_min_size = 8
199
+ for spine_name, spine in ax.spines.items():
200
+ if spine.get_visible():
201
+ try:
202
+ spine_bbox = spine.get_window_extent(renderer)
203
+ if spine_bbox is not None:
204
+ bbox = transform_bbox(
205
+ spine_bbox,
206
+ fig,
207
+ tight_bbox,
208
+ img_width,
209
+ img_height,
210
+ scale_x,
211
+ scale_y,
212
+ pad_inches,
213
+ saved_height_inches,
214
+ )
215
+ if bbox:
216
+ if bbox["width"] < spine_min_size:
217
+ expand = (spine_min_size - bbox["width"]) / 2
218
+ bbox["x"] -= expand
219
+ bbox["width"] = spine_min_size
220
+ if bbox["height"] < spine_min_size:
221
+ expand = (spine_min_size - bbox["height"]) / 2
222
+ bbox["y"] -= expand
223
+ bbox["height"] = spine_min_size
224
+ bboxes[f"ax{ax_idx}_spine_{spine_name}"] = {
225
+ **bbox,
226
+ "type": "spine",
227
+ "label": spine_name,
228
+ "ax_index": ax_idx,
229
+ }
230
+ except Exception:
231
+ pass
232
+
233
+
234
+ def extract_figure_text(
235
+ fig,
236
+ renderer,
237
+ tight_bbox,
238
+ img_width,
239
+ img_height,
240
+ scale_x,
241
+ scale_y,
242
+ pad_inches,
243
+ saved_height_inches,
244
+ bboxes,
245
+ ):
246
+ """Extract bboxes for figure-level text (suptitle, supxlabel, supylabel)."""
247
+ # Suptitle
248
+ if hasattr(fig, "_suptitle") and fig._suptitle is not None:
249
+ suptitle_obj = fig._suptitle
250
+ if suptitle_obj.get_text():
251
+ try:
252
+ suptitle_extent = suptitle_obj.get_window_extent(renderer)
253
+ if suptitle_extent is not None:
254
+ bbox = transform_bbox(
255
+ suptitle_extent,
256
+ fig,
257
+ tight_bbox,
258
+ img_width,
259
+ img_height,
260
+ scale_x,
261
+ scale_y,
262
+ pad_inches,
263
+ saved_height_inches,
264
+ )
265
+ if bbox:
266
+ bboxes["fig_suptitle"] = {
267
+ **bbox,
268
+ "type": "suptitle",
269
+ "label": "suptitle",
270
+ "ax_index": -1,
271
+ "text": suptitle_obj.get_text(),
272
+ }
273
+ except Exception:
274
+ pass
275
+
276
+ # Supxlabel
277
+ if hasattr(fig, "_supxlabel") and fig._supxlabel is not None:
278
+ supxlabel_obj = fig._supxlabel
279
+ if supxlabel_obj.get_text():
280
+ try:
281
+ supxlabel_extent = supxlabel_obj.get_window_extent(renderer)
282
+ if supxlabel_extent is not None:
283
+ bbox = transform_bbox(
284
+ supxlabel_extent,
285
+ fig,
286
+ tight_bbox,
287
+ img_width,
288
+ img_height,
289
+ scale_x,
290
+ scale_y,
291
+ pad_inches,
292
+ saved_height_inches,
293
+ )
294
+ if bbox:
295
+ bboxes["fig_supxlabel"] = {
296
+ **bbox,
297
+ "type": "supxlabel",
298
+ "label": "supxlabel",
299
+ "ax_index": -1,
300
+ "text": supxlabel_obj.get_text(),
301
+ }
302
+ except Exception:
303
+ pass
304
+
305
+ # Supylabel
306
+ if hasattr(fig, "_supylabel") and fig._supylabel is not None:
307
+ supylabel_obj = fig._supylabel
308
+ if supylabel_obj.get_text():
309
+ try:
310
+ supylabel_extent = supylabel_obj.get_window_extent(renderer)
311
+ if supylabel_extent is not None:
312
+ bbox = transform_bbox(
313
+ supylabel_extent,
314
+ fig,
315
+ tight_bbox,
316
+ img_width,
317
+ img_height,
318
+ scale_x,
319
+ scale_y,
320
+ pad_inches,
321
+ saved_height_inches,
322
+ )
323
+ if bbox:
324
+ bboxes["fig_supylabel"] = {
325
+ **bbox,
326
+ "type": "supylabel",
327
+ "label": "supylabel",
328
+ "ax_index": -1,
329
+ "text": supylabel_obj.get_text(),
330
+ }
331
+ except Exception:
332
+ pass
333
+
334
+
335
+ __all__ = [
336
+ "extract_text_elements",
337
+ "extract_legend",
338
+ "extract_spines",
339
+ "extract_figure_text",
340
+ ]
341
+
342
+ # EOF