scitex 2.7.3__py3-none-any.whl → 2.10.0__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 (563) hide show
  1. scitex/__init__.py +15 -7
  2. scitex/__version__.py +1 -2
  3. scitex/_install_guide.py +250 -0
  4. scitex/_optional_deps.py +206 -39
  5. scitex/ai/_gen_ai/_Groq.py +2 -4
  6. scitex/ai/_gen_ai/_OpenAI.py +5 -2
  7. scitex/ai/_gen_ai/_Perplexity.py +20 -6
  8. scitex/audio/__init__.py +24 -15
  9. scitex/audio/_cross_process_lock.py +139 -0
  10. scitex/audio/_mcp_handlers.py +256 -0
  11. scitex/audio/_mcp_tool_schemas.py +203 -0
  12. scitex/audio/engines/elevenlabs_engine.py +5 -2
  13. scitex/audio/mcp_server.py +98 -457
  14. scitex/bridge/__init__.py +30 -19
  15. scitex/bridge/_figrecipe.py +245 -0
  16. scitex/bridge/_helpers.py +2 -1
  17. scitex/bridge/_plt_vis.py +23 -10
  18. scitex/bridge/_stats_plt.py +18 -5
  19. scitex/bridge/_stats_vis.py +16 -2
  20. scitex/browser/__init__.py +84 -44
  21. scitex/browser/automation/__init__.py +5 -1
  22. scitex/browser/core/BrowserMixin.py +17 -4
  23. scitex/browser/core/__init__.py +11 -2
  24. scitex/browser/remote/CaptchaHandler.py +1 -1
  25. scitex/browser/remote/ZenRowsAPIClient.py +1 -1
  26. scitex/capture/grid.py +487 -0
  27. scitex/capture/mcp_handlers.py +401 -0
  28. scitex/capture/mcp_tool_defs.py +192 -0
  29. scitex/capture/mcp_tools.py +241 -0
  30. scitex/capture/mcp_utils.py +30 -0
  31. scitex/cli/convert.py +421 -0
  32. scitex/cli/main.py +6 -4
  33. scitex/datetime/__init__.py +46 -0
  34. scitex/datetime/_linspace.py +100 -0
  35. scitex/datetime/_normalize_timestamp.py +306 -0
  36. scitex/db/_delete_duplicates.py +4 -4
  37. scitex/db/_sqlite3/_delete_duplicates.py +11 -2
  38. scitex/dev/plt/__init__.py +61 -62
  39. scitex/dev/plt/demo_plotters/__init__.py +0 -0
  40. scitex/dev/plt/demo_plotters/plot_mpl_axhline.py +28 -0
  41. scitex/dev/plt/demo_plotters/plot_mpl_axhspan.py +28 -0
  42. scitex/dev/plt/demo_plotters/plot_mpl_axvline.py +28 -0
  43. scitex/dev/plt/demo_plotters/plot_mpl_axvspan.py +28 -0
  44. scitex/dev/plt/demo_plotters/plot_mpl_bar.py +29 -0
  45. scitex/dev/plt/demo_plotters/plot_mpl_barh.py +29 -0
  46. scitex/dev/plt/demo_plotters/plot_mpl_boxplot.py +28 -0
  47. scitex/dev/plt/demo_plotters/plot_mpl_contour.py +31 -0
  48. scitex/dev/plt/demo_plotters/plot_mpl_contourf.py +31 -0
  49. scitex/dev/plt/demo_plotters/plot_mpl_errorbar.py +30 -0
  50. scitex/dev/plt/demo_plotters/plot_mpl_eventplot.py +28 -0
  51. scitex/dev/plt/demo_plotters/plot_mpl_fill.py +30 -0
  52. scitex/dev/plt/demo_plotters/plot_mpl_fill_between.py +31 -0
  53. scitex/dev/plt/demo_plotters/plot_mpl_hexbin.py +28 -0
  54. scitex/dev/plt/demo_plotters/plot_mpl_hist.py +28 -0
  55. scitex/dev/plt/demo_plotters/plot_mpl_hist2d.py +28 -0
  56. scitex/dev/plt/demo_plotters/plot_mpl_imshow.py +29 -0
  57. scitex/dev/plt/demo_plotters/plot_mpl_pcolormesh.py +31 -0
  58. scitex/dev/plt/demo_plotters/plot_mpl_pie.py +29 -0
  59. scitex/dev/plt/demo_plotters/plot_mpl_plot.py +29 -0
  60. scitex/dev/plt/demo_plotters/plot_mpl_quiver.py +31 -0
  61. scitex/dev/plt/demo_plotters/plot_mpl_scatter.py +28 -0
  62. scitex/dev/plt/demo_plotters/plot_mpl_stackplot.py +31 -0
  63. scitex/dev/plt/demo_plotters/plot_mpl_stem.py +29 -0
  64. scitex/dev/plt/demo_plotters/plot_mpl_step.py +29 -0
  65. scitex/dev/plt/demo_plotters/plot_mpl_violinplot.py +28 -0
  66. scitex/dev/plt/demo_plotters/plot_sns_barplot.py +29 -0
  67. scitex/dev/plt/demo_plotters/plot_sns_boxplot.py +29 -0
  68. scitex/dev/plt/demo_plotters/plot_sns_heatmap.py +28 -0
  69. scitex/dev/plt/demo_plotters/plot_sns_histplot.py +29 -0
  70. scitex/dev/plt/demo_plotters/plot_sns_kdeplot.py +29 -0
  71. scitex/dev/plt/demo_plotters/plot_sns_lineplot.py +31 -0
  72. scitex/dev/plt/demo_plotters/plot_sns_scatterplot.py +29 -0
  73. scitex/dev/plt/demo_plotters/plot_sns_stripplot.py +29 -0
  74. scitex/dev/plt/demo_plotters/plot_sns_swarmplot.py +29 -0
  75. scitex/dev/plt/demo_plotters/plot_sns_violinplot.py +29 -0
  76. scitex/dev/plt/demo_plotters/plot_stx_bar.py +29 -0
  77. scitex/dev/plt/demo_plotters/plot_stx_barh.py +29 -0
  78. scitex/dev/plt/demo_plotters/plot_stx_box.py +28 -0
  79. scitex/dev/plt/demo_plotters/plot_stx_boxplot.py +28 -0
  80. scitex/dev/plt/demo_plotters/plot_stx_conf_mat.py +28 -0
  81. scitex/dev/plt/demo_plotters/plot_stx_contour.py +31 -0
  82. scitex/dev/plt/demo_plotters/plot_stx_ecdf.py +28 -0
  83. scitex/dev/plt/demo_plotters/plot_stx_errorbar.py +30 -0
  84. scitex/dev/plt/demo_plotters/plot_stx_fill_between.py +31 -0
  85. scitex/dev/plt/demo_plotters/plot_stx_fillv.py +28 -0
  86. scitex/dev/plt/demo_plotters/plot_stx_heatmap.py +28 -0
  87. scitex/dev/plt/demo_plotters/plot_stx_image.py +28 -0
  88. scitex/dev/plt/demo_plotters/plot_stx_imshow.py +28 -0
  89. scitex/dev/plt/demo_plotters/plot_stx_joyplot.py +28 -0
  90. scitex/dev/plt/demo_plotters/plot_stx_kde.py +28 -0
  91. scitex/dev/plt/demo_plotters/plot_stx_line.py +28 -0
  92. scitex/dev/plt/demo_plotters/plot_stx_mean_ci.py +28 -0
  93. scitex/dev/plt/demo_plotters/plot_stx_mean_std.py +28 -0
  94. scitex/dev/plt/demo_plotters/plot_stx_median_iqr.py +28 -0
  95. scitex/dev/plt/demo_plotters/plot_stx_raster.py +28 -0
  96. scitex/dev/plt/demo_plotters/plot_stx_rectangle.py +28 -0
  97. scitex/dev/plt/demo_plotters/plot_stx_scatter.py +29 -0
  98. scitex/dev/plt/demo_plotters/plot_stx_shaded_line.py +29 -0
  99. scitex/dev/plt/demo_plotters/plot_stx_violin.py +28 -0
  100. scitex/dev/plt/demo_plotters/plot_stx_violinplot.py +28 -0
  101. scitex/dev/plt/mpl/get_dir_ax.py +46 -0
  102. scitex/dev/plt/mpl/get_signatures.py +176 -0
  103. scitex/dev/plt/mpl/get_signatures_details.py +522 -0
  104. scitex/dev/plt/plot_mpl_axhline.py +0 -0
  105. scitex/dev/plt/plot_mpl_axhspan.py +0 -0
  106. scitex/dev/plt/plot_mpl_axvline.py +0 -0
  107. scitex/dev/plt/plot_mpl_axvspan.py +0 -0
  108. scitex/dev/plt/plot_mpl_bar.py +0 -0
  109. scitex/dev/plt/plot_mpl_barh.py +0 -0
  110. scitex/dev/plt/plot_mpl_boxplot.py +0 -0
  111. scitex/dev/plt/plot_mpl_contour.py +0 -0
  112. scitex/dev/plt/plot_mpl_contourf.py +0 -0
  113. scitex/dev/plt/plot_mpl_errorbar.py +0 -0
  114. scitex/dev/plt/plot_mpl_eventplot.py +0 -0
  115. scitex/dev/plt/plot_mpl_fill.py +0 -0
  116. scitex/dev/plt/plot_mpl_fill_between.py +0 -0
  117. scitex/dev/plt/plot_mpl_hexbin.py +0 -0
  118. scitex/dev/plt/plot_mpl_hist.py +0 -0
  119. scitex/dev/plt/plot_mpl_hist2d.py +0 -0
  120. scitex/dev/plt/plot_mpl_imshow.py +0 -0
  121. scitex/dev/plt/plot_mpl_pcolormesh.py +0 -0
  122. scitex/dev/plt/plot_mpl_pie.py +0 -0
  123. scitex/dev/plt/plot_mpl_plot.py +0 -0
  124. scitex/dev/plt/plot_mpl_quiver.py +0 -0
  125. scitex/dev/plt/plot_mpl_scatter.py +0 -0
  126. scitex/dev/plt/plot_mpl_stackplot.py +0 -0
  127. scitex/dev/plt/plot_mpl_stem.py +0 -0
  128. scitex/dev/plt/plot_mpl_step.py +0 -0
  129. scitex/dev/plt/plot_mpl_violinplot.py +0 -0
  130. scitex/dev/plt/plot_sns_barplot.py +0 -0
  131. scitex/dev/plt/plot_sns_boxplot.py +0 -0
  132. scitex/dev/plt/plot_sns_heatmap.py +0 -0
  133. scitex/dev/plt/plot_sns_histplot.py +0 -0
  134. scitex/dev/plt/plot_sns_kdeplot.py +0 -0
  135. scitex/dev/plt/plot_sns_lineplot.py +0 -0
  136. scitex/dev/plt/plot_sns_scatterplot.py +0 -0
  137. scitex/dev/plt/plot_sns_stripplot.py +0 -0
  138. scitex/dev/plt/plot_sns_swarmplot.py +0 -0
  139. scitex/dev/plt/plot_sns_violinplot.py +0 -0
  140. scitex/dev/plt/plot_stx_bar.py +0 -0
  141. scitex/dev/plt/plot_stx_barh.py +0 -0
  142. scitex/dev/plt/plot_stx_box.py +0 -0
  143. scitex/dev/plt/plot_stx_boxplot.py +0 -0
  144. scitex/dev/plt/plot_stx_conf_mat.py +0 -0
  145. scitex/dev/plt/plot_stx_contour.py +0 -0
  146. scitex/dev/plt/plot_stx_ecdf.py +0 -0
  147. scitex/dev/plt/plot_stx_errorbar.py +0 -0
  148. scitex/dev/plt/plot_stx_fill_between.py +0 -0
  149. scitex/dev/plt/plot_stx_fillv.py +0 -0
  150. scitex/dev/plt/plot_stx_heatmap.py +0 -0
  151. scitex/dev/plt/plot_stx_image.py +0 -0
  152. scitex/dev/plt/plot_stx_imshow.py +0 -0
  153. scitex/dev/plt/plot_stx_joyplot.py +0 -0
  154. scitex/dev/plt/plot_stx_kde.py +0 -0
  155. scitex/dev/plt/plot_stx_line.py +0 -0
  156. scitex/dev/plt/plot_stx_mean_ci.py +0 -0
  157. scitex/dev/plt/plot_stx_mean_std.py +0 -0
  158. scitex/dev/plt/plot_stx_median_iqr.py +0 -0
  159. scitex/dev/plt/plot_stx_raster.py +0 -0
  160. scitex/dev/plt/plot_stx_rectangle.py +0 -0
  161. scitex/dev/plt/plot_stx_scatter.py +0 -0
  162. scitex/dev/plt/plot_stx_shaded_line.py +0 -0
  163. scitex/dev/plt/plot_stx_violin.py +0 -0
  164. scitex/dev/plt/plot_stx_violinplot.py +0 -0
  165. scitex/diagram/README.md +197 -0
  166. scitex/diagram/__init__.py +48 -0
  167. scitex/diagram/_compile.py +312 -0
  168. scitex/diagram/_diagram.py +355 -0
  169. scitex/diagram/_presets.py +173 -0
  170. scitex/diagram/_schema.py +182 -0
  171. scitex/diagram/_split.py +278 -0
  172. scitex/dict/_pop_keys.py +1 -7
  173. scitex/dsp/__init__.py +15 -10
  174. scitex/dsp/add_noise.py +5 -2
  175. scitex/dsp/example.py +35 -22
  176. scitex/dsp/filt.py +8 -3
  177. scitex/dsp/reference.py +3 -2
  178. scitex/dsp/utils/__init__.py +2 -1
  179. scitex/dsp/utils/_differential_bandpass_filters.py +14 -4
  180. scitex/dt/__init__.py +39 -2
  181. scitex/errors.py +82 -521
  182. scitex/fig/__init__.py +4 -4
  183. scitex/fig/editor/__init__.py +5 -2
  184. scitex/fig/editor/_dearpygui_editor.py +1 -1
  185. scitex/fig/editor/_mpl_editor.py +1 -1
  186. scitex/fig/editor/_qt_editor.py +1 -1
  187. scitex/fig/editor/_tkinter_editor.py +1 -1
  188. scitex/fig/editor/edit/__init__.py +50 -0
  189. scitex/fig/editor/edit/backend_detector.py +109 -0
  190. scitex/fig/editor/edit/bundle_resolver.py +240 -0
  191. scitex/fig/editor/edit/editor_launcher.py +239 -0
  192. scitex/fig/editor/edit/manual_handler.py +53 -0
  193. scitex/fig/editor/edit/panel_loader.py +232 -0
  194. scitex/fig/editor/edit/path_resolver.py +67 -0
  195. scitex/fig/editor/flask_editor/_bbox.py +23 -0
  196. scitex/fig/editor/flask_editor/_core.py +908 -103
  197. scitex/fig/editor/flask_editor/_renderer.py +74 -0
  198. scitex/fig/editor/flask_editor/static/css/base/reset.css +41 -0
  199. scitex/fig/editor/flask_editor/static/css/base/typography.css +16 -0
  200. scitex/fig/editor/flask_editor/static/css/base/variables.css +85 -0
  201. scitex/fig/editor/flask_editor/static/css/components/buttons.css +217 -0
  202. scitex/fig/editor/flask_editor/static/css/components/context-menu.css +93 -0
  203. scitex/fig/editor/flask_editor/static/css/components/dropdown.css +57 -0
  204. scitex/fig/editor/flask_editor/static/css/components/forms.css +112 -0
  205. scitex/fig/editor/flask_editor/static/css/components/modal.css +59 -0
  206. scitex/fig/editor/flask_editor/static/css/components/sections.css +212 -0
  207. scitex/fig/editor/flask_editor/static/css/features/canvas.css +176 -0
  208. scitex/fig/editor/flask_editor/static/css/features/element-inspector.css +190 -0
  209. scitex/fig/editor/flask_editor/static/css/features/loading.css +59 -0
  210. scitex/fig/editor/flask_editor/static/css/features/overlay.css +45 -0
  211. scitex/fig/editor/flask_editor/static/css/features/panel-grid.css +95 -0
  212. scitex/fig/editor/flask_editor/static/css/features/selection.css +101 -0
  213. scitex/fig/editor/flask_editor/static/css/features/statistics.css +138 -0
  214. scitex/fig/editor/flask_editor/static/css/index.css +31 -0
  215. scitex/fig/editor/flask_editor/static/css/layout/container.css +7 -0
  216. scitex/fig/editor/flask_editor/static/css/layout/controls.css +56 -0
  217. scitex/fig/editor/flask_editor/static/css/layout/preview.css +78 -0
  218. scitex/fig/editor/flask_editor/static/js/alignment/axis.js +314 -0
  219. scitex/fig/editor/flask_editor/static/js/alignment/basic.js +107 -0
  220. scitex/fig/editor/flask_editor/static/js/alignment/distribute.js +54 -0
  221. scitex/fig/editor/flask_editor/static/js/canvas/canvas.js +172 -0
  222. scitex/fig/editor/flask_editor/static/js/canvas/dragging.js +258 -0
  223. scitex/fig/editor/flask_editor/static/js/canvas/resize.js +48 -0
  224. scitex/fig/editor/flask_editor/static/js/canvas/selection.js +71 -0
  225. scitex/fig/editor/flask_editor/static/js/core/api.js +288 -0
  226. scitex/fig/editor/flask_editor/static/js/core/state.js +143 -0
  227. scitex/fig/editor/flask_editor/static/js/core/utils.js +245 -0
  228. scitex/fig/editor/flask_editor/static/js/dev/element-inspector.js +992 -0
  229. scitex/fig/editor/flask_editor/static/js/editor/bbox.js +339 -0
  230. scitex/fig/editor/flask_editor/static/js/editor/element-drag.js +286 -0
  231. scitex/fig/editor/flask_editor/static/js/editor/overlay.js +371 -0
  232. scitex/fig/editor/flask_editor/static/js/editor/preview.js +293 -0
  233. scitex/fig/editor/flask_editor/static/js/main.js +426 -0
  234. scitex/fig/editor/flask_editor/static/js/shortcuts/context-menu.js +152 -0
  235. scitex/fig/editor/flask_editor/static/js/shortcuts/keyboard.js +265 -0
  236. scitex/fig/editor/flask_editor/static/js/ui/controls.js +184 -0
  237. scitex/fig/editor/flask_editor/static/js/ui/download.js +57 -0
  238. scitex/fig/editor/flask_editor/static/js/ui/help.js +100 -0
  239. scitex/fig/editor/flask_editor/static/js/ui/theme.js +34 -0
  240. scitex/fig/editor/flask_editor/templates/__init__.py +95 -5
  241. scitex/fig/editor/flask_editor/templates/_html.py +27 -9
  242. scitex/fig/editor/flask_editor/templates/_scripts.py +1928 -131
  243. scitex/fig/editor/flask_editor/templates/_styles.py +363 -51
  244. scitex/fig/io/_bundle.py +104 -19
  245. scitex/fts/README.md +262 -0
  246. scitex/fts/TODO.md +66 -0
  247. scitex/fts/__init__.py +90 -0
  248. scitex/fts/_bundle/README_IN_BUNDLE.md +102 -0
  249. scitex/fts/_bundle/_FTS.py +657 -0
  250. scitex/fts/_bundle/__init__.py +38 -0
  251. scitex/fts/_bundle/_children.py +216 -0
  252. scitex/fts/_bundle/_conversion/__init__.py +15 -0
  253. scitex/fts/_bundle/_conversion/_bundle2dict.py +44 -0
  254. scitex/fts/_bundle/_conversion/_dict2bundle.py +50 -0
  255. scitex/fts/_bundle/_dataclasses/_Axes.py +57 -0
  256. scitex/fts/_bundle/_dataclasses/_BBox.py +54 -0
  257. scitex/fts/_bundle/_dataclasses/_ColumnDef.py +72 -0
  258. scitex/fts/_bundle/_dataclasses/_DataFormat.py +40 -0
  259. scitex/fts/_bundle/_dataclasses/_DataInfo.py +135 -0
  260. scitex/fts/_bundle/_dataclasses/_DataSource.py +44 -0
  261. scitex/fts/_bundle/_dataclasses/_Node.py +319 -0
  262. scitex/fts/_bundle/_dataclasses/_NodeRefs.py +45 -0
  263. scitex/fts/_bundle/_dataclasses/_SizeMM.py +38 -0
  264. scitex/fts/_bundle/_dataclasses/__init__.py +35 -0
  265. scitex/fts/_bundle/_extractors/__init__.py +32 -0
  266. scitex/fts/_bundle/_extractors/_extract_bar.py +131 -0
  267. scitex/fts/_bundle/_extractors/_extract_line.py +71 -0
  268. scitex/fts/_bundle/_extractors/_extract_scatter.py +79 -0
  269. scitex/fts/_bundle/_loader.py +134 -0
  270. scitex/fts/_bundle/_mpl_helpers.py +389 -0
  271. scitex/fts/_bundle/_saver.py +269 -0
  272. scitex/fts/_bundle/_storage.py +200 -0
  273. scitex/fts/_bundle/_utils/__init__.py +55 -0
  274. scitex/fts/_bundle/_utils/_const.py +26 -0
  275. scitex/fts/_bundle/_utils/_errors.py +73 -0
  276. scitex/fts/_bundle/_utils/_generate.py +21 -0
  277. scitex/fts/_bundle/_utils/_types.py +76 -0
  278. scitex/fts/_bundle/_validation.py +434 -0
  279. scitex/fts/_bundle/_zipbundle.py +165 -0
  280. scitex/fts/_fig/__init__.py +22 -0
  281. scitex/fts/_fig/_backend/__init__.py +53 -0
  282. scitex/fts/_fig/_backend/_export.py +165 -0
  283. scitex/fts/_fig/_backend/_parser.py +188 -0
  284. scitex/fts/_fig/_backend/_render.py +538 -0
  285. scitex/fts/_fig/_composite.py +345 -0
  286. scitex/fts/_fig/_dataclasses/_ChannelEncoding.py +46 -0
  287. scitex/fts/_fig/_dataclasses/_Encoding.py +82 -0
  288. scitex/fts/_fig/_dataclasses/_Theme.py +441 -0
  289. scitex/fts/_fig/_dataclasses/_TraceEncoding.py +52 -0
  290. scitex/fts/_fig/_dataclasses/__init__.py +47 -0
  291. scitex/fts/_fig/_editor/__init__.py +14 -0
  292. scitex/fts/_fig/_editor/_cui/__init__.py +33 -0
  293. scitex/fts/_fig/_editor/_cui/_backend_detector.py +39 -0
  294. scitex/fts/_fig/_editor/_cui/_bundle_resolver.py +366 -0
  295. scitex/fts/_fig/_editor/_cui/_editor_launcher.py +175 -0
  296. scitex/fts/_fig/_editor/_cui/_manual_handler.py +52 -0
  297. scitex/fts/_fig/_editor/_cui/_panel_loader.py +246 -0
  298. scitex/fts/_fig/_editor/_cui/_path_resolver.py +66 -0
  299. scitex/fts/_fig/_editor/_defaults.py +300 -0
  300. scitex/fts/_fig/_editor/_gui/__init__.py +11 -0
  301. scitex/fts/_fig/_editor/_gui/_flask_editor/__init__.py +20 -0
  302. scitex/fts/_fig/_editor/_gui/_flask_editor/_bbox.py +1339 -0
  303. scitex/fts/_fig/_editor/_gui/_flask_editor/_core.py +1688 -0
  304. scitex/fts/_fig/_editor/_gui/_flask_editor/_plotter.py +664 -0
  305. scitex/fts/_fig/_editor/_gui/_flask_editor/_renderer.py +853 -0
  306. scitex/fts/_fig/_editor/_gui/_flask_editor/_utils.py +79 -0
  307. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/base/reset.css +41 -0
  308. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/base/typography.css +16 -0
  309. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/base/variables.css +85 -0
  310. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/buttons.css +217 -0
  311. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/context-menu.css +93 -0
  312. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/dropdown.css +57 -0
  313. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/forms.css +112 -0
  314. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/modal.css +59 -0
  315. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/sections.css +212 -0
  316. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/canvas.css +176 -0
  317. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/element-inspector.css +190 -0
  318. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/loading.css +59 -0
  319. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/overlay.css +45 -0
  320. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/panel-grid.css +95 -0
  321. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/selection.css +101 -0
  322. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/statistics.css +138 -0
  323. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/index.css +31 -0
  324. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/layout/container.css +7 -0
  325. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/layout/controls.css +56 -0
  326. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/layout/preview.css +78 -0
  327. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/alignment/axis.js +314 -0
  328. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/alignment/basic.js +107 -0
  329. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/alignment/distribute.js +54 -0
  330. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/canvas.js +172 -0
  331. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/dragging.js +258 -0
  332. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/resize.js +48 -0
  333. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/selection.js +71 -0
  334. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/core/api.js +288 -0
  335. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/core/state.js +143 -0
  336. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/core/utils.js +245 -0
  337. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/dev/element-inspector.js +992 -0
  338. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/bbox.js +339 -0
  339. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/element-drag.js +286 -0
  340. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/overlay.js +371 -0
  341. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/preview.js +293 -0
  342. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/main.js +426 -0
  343. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/shortcuts/context-menu.js +152 -0
  344. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/shortcuts/keyboard.js +265 -0
  345. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/controls.js +184 -0
  346. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/download.js +57 -0
  347. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/help.js +100 -0
  348. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/theme.js +34 -0
  349. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/__init__.py +124 -0
  350. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/_html.py +851 -0
  351. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/_scripts.py +4932 -0
  352. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/_styles.py +1657 -0
  353. scitex/fts/_fig/_editor/_gui/_flask_editor.py +36 -0
  354. scitex/fts/_fig/_models/_Annotations.py +115 -0
  355. scitex/fts/_fig/_models/_Axes.py +152 -0
  356. scitex/fts/_fig/_models/_Figure.py +138 -0
  357. scitex/fts/_fig/_models/_Guides.py +104 -0
  358. scitex/fts/_fig/_models/_Plot.py +123 -0
  359. scitex/fts/_fig/_models/_Styles.py +245 -0
  360. scitex/fts/_fig/_models/__init__.py +80 -0
  361. scitex/fts/_fig/_models/_plot_types/__init__.py +156 -0
  362. scitex/fts/_fig/_models/_plot_types/_bar.py +43 -0
  363. scitex/fts/_fig/_models/_plot_types/_box.py +38 -0
  364. scitex/fts/_fig/_models/_plot_types/_distribution.py +36 -0
  365. scitex/fts/_fig/_models/_plot_types/_errorbar.py +60 -0
  366. scitex/fts/_fig/_models/_plot_types/_histogram.py +30 -0
  367. scitex/fts/_fig/_models/_plot_types/_image.py +61 -0
  368. scitex/fts/_fig/_models/_plot_types/_line.py +57 -0
  369. scitex/fts/_fig/_models/_plot_types/_scatter.py +30 -0
  370. scitex/fts/_fig/_models/_plot_types/_seaborn.py +121 -0
  371. scitex/fts/_fig/_models/_plot_types/_violin.py +36 -0
  372. scitex/fts/_fig/_utils/__init__.py +129 -0
  373. scitex/fts/_fig/_utils/_auto_layout.py +127 -0
  374. scitex/fts/_fig/_utils/_calc_bounds.py +111 -0
  375. scitex/fts/_fig/_utils/_const_sizes.py +48 -0
  376. scitex/fts/_fig/_utils/_convert_coords.py +77 -0
  377. scitex/fts/_fig/_utils/_get_template.py +178 -0
  378. scitex/fts/_fig/_utils/_normalize.py +73 -0
  379. scitex/fts/_fig/_utils/_plot_layout.py +397 -0
  380. scitex/fts/_fig/_utils/_validate.py +197 -0
  381. scitex/fts/_kinds/__init__.py +45 -0
  382. scitex/fts/_kinds/_figure/__init__.py +19 -0
  383. scitex/fts/_kinds/_figure/_composite.py +345 -0
  384. scitex/fts/_kinds/_plot/__init__.py +25 -0
  385. scitex/fts/_kinds/_plot/_backend/__init__.py +53 -0
  386. scitex/fts/_kinds/_plot/_backend/_export.py +165 -0
  387. scitex/fts/_kinds/_plot/_backend/_parser.py +188 -0
  388. scitex/fts/_kinds/_plot/_backend/_render.py +538 -0
  389. scitex/fts/_kinds/_plot/_dataclasses/_ChannelEncoding.py +46 -0
  390. scitex/fts/_kinds/_plot/_dataclasses/_Encoding.py +82 -0
  391. scitex/fts/_kinds/_plot/_dataclasses/_Theme.py +441 -0
  392. scitex/fts/_kinds/_plot/_dataclasses/_TraceEncoding.py +52 -0
  393. scitex/fts/_kinds/_plot/_dataclasses/__init__.py +47 -0
  394. scitex/fts/_kinds/_plot/_models/_Annotations.py +115 -0
  395. scitex/fts/_kinds/_plot/_models/_Axes.py +152 -0
  396. scitex/fts/_kinds/_plot/_models/_Figure.py +138 -0
  397. scitex/fts/_kinds/_plot/_models/_Guides.py +104 -0
  398. scitex/fts/_kinds/_plot/_models/_Plot.py +123 -0
  399. scitex/fts/_kinds/_plot/_models/_Styles.py +245 -0
  400. scitex/fts/_kinds/_plot/_models/__init__.py +80 -0
  401. scitex/fts/_kinds/_plot/_models/_plot_types/__init__.py +156 -0
  402. scitex/fts/_kinds/_plot/_models/_plot_types/_bar.py +43 -0
  403. scitex/fts/_kinds/_plot/_models/_plot_types/_box.py +38 -0
  404. scitex/fts/_kinds/_plot/_models/_plot_types/_distribution.py +36 -0
  405. scitex/fts/_kinds/_plot/_models/_plot_types/_errorbar.py +60 -0
  406. scitex/fts/_kinds/_plot/_models/_plot_types/_histogram.py +30 -0
  407. scitex/fts/_kinds/_plot/_models/_plot_types/_image.py +61 -0
  408. scitex/fts/_kinds/_plot/_models/_plot_types/_line.py +57 -0
  409. scitex/fts/_kinds/_plot/_models/_plot_types/_scatter.py +30 -0
  410. scitex/fts/_kinds/_plot/_models/_plot_types/_seaborn.py +121 -0
  411. scitex/fts/_kinds/_plot/_models/_plot_types/_violin.py +36 -0
  412. scitex/fts/_kinds/_plot/_utils/__init__.py +129 -0
  413. scitex/fts/_kinds/_plot/_utils/_auto_layout.py +127 -0
  414. scitex/fts/_kinds/_plot/_utils/_calc_bounds.py +111 -0
  415. scitex/fts/_kinds/_plot/_utils/_const_sizes.py +48 -0
  416. scitex/fts/_kinds/_plot/_utils/_convert_coords.py +77 -0
  417. scitex/fts/_kinds/_plot/_utils/_get_template.py +178 -0
  418. scitex/fts/_kinds/_plot/_utils/_normalize.py +73 -0
  419. scitex/fts/_kinds/_plot/_utils/_plot_layout.py +397 -0
  420. scitex/fts/_kinds/_plot/_utils/_validate.py +197 -0
  421. scitex/fts/_kinds/_shape/__init__.py +141 -0
  422. scitex/fts/_kinds/_stats/__init__.py +56 -0
  423. scitex/fts/_kinds/_stats/_dataclasses/_Stats.py +423 -0
  424. scitex/fts/_kinds/_stats/_dataclasses/__init__.py +48 -0
  425. scitex/fts/_kinds/_table/__init__.py +72 -0
  426. scitex/fts/_kinds/_table/_latex/__init__.py +93 -0
  427. scitex/fts/_kinds/_table/_latex/_editor/__init__.py +11 -0
  428. scitex/fts/_kinds/_table/_latex/_editor/_app.py +725 -0
  429. scitex/fts/_kinds/_table/_latex/_export.py +279 -0
  430. scitex/fts/_kinds/_table/_latex/_figure_exporter.py +153 -0
  431. scitex/fts/_kinds/_table/_latex/_stats_formatter.py +274 -0
  432. scitex/fts/_kinds/_table/_latex/_table_exporter.py +362 -0
  433. scitex/fts/_kinds/_table/_latex/_utils.py +369 -0
  434. scitex/fts/_kinds/_table/_latex/_validator.py +445 -0
  435. scitex/fts/_kinds/_text/__init__.py +77 -0
  436. scitex/fts/_schemas/data_info.schema.json +75 -0
  437. scitex/fts/_schemas/encoding.schema.json +90 -0
  438. scitex/fts/_schemas/node.schema.json +145 -0
  439. scitex/fts/_schemas/render_manifest.schema.json +62 -0
  440. scitex/fts/_schemas/stats.schema.json +132 -0
  441. scitex/fts/_schemas/theme.schema.json +141 -0
  442. scitex/fts/_stats/__init__.py +48 -0
  443. scitex/fts/_stats/_dataclasses/_Stats.py +423 -0
  444. scitex/fts/_stats/_dataclasses/__init__.py +48 -0
  445. scitex/fts/_tables/__init__.py +65 -0
  446. scitex/fts/_tables/_latex/__init__.py +93 -0
  447. scitex/fts/_tables/_latex/_editor/__init__.py +11 -0
  448. scitex/fts/_tables/_latex/_editor/_app.py +725 -0
  449. scitex/fts/_tables/_latex/_export.py +279 -0
  450. scitex/fts/_tables/_latex/_figure_exporter.py +153 -0
  451. scitex/fts/_tables/_latex/_stats_formatter.py +274 -0
  452. scitex/fts/_tables/_latex/_table_exporter.py +362 -0
  453. scitex/fts/_tables/_latex/_utils.py +369 -0
  454. scitex/fts/_tables/_latex/_validator.py +445 -0
  455. scitex/gen/__init__.py +66 -25
  456. scitex/gen/misc.py +28 -0
  457. scitex/io/__init__.py +47 -20
  458. scitex/io/_load.py +87 -36
  459. scitex/io/_load_modules/__init__.py +10 -7
  460. scitex/io/_load_modules/_pandas.py +6 -1
  461. scitex/io/_save.py +299 -1556
  462. scitex/io/_save_modules/__init__.py +76 -19
  463. scitex/io/_save_modules/_figure_utils.py +90 -0
  464. scitex/io/_save_modules/_image_csv.py +497 -0
  465. scitex/io/_save_modules/_legends.py +91 -0
  466. scitex/io/_save_modules/_pltz_bundle.py +356 -0
  467. scitex/io/_save_modules/_pltz_stx.py +536 -0
  468. scitex/io/_save_modules/_stx_bundle.py +104 -0
  469. scitex/io/_save_modules/_symlink.py +96 -0
  470. scitex/io/_save_modules/_yaml.py +1 -1
  471. scitex/io/_save_modules/_zarr.py +64 -18
  472. scitex/io/bundle/README.md +212 -0
  473. scitex/io/bundle/__init__.py +110 -0
  474. scitex/io/{_bundle.py → bundle/_core.py} +219 -89
  475. scitex/io/bundle/_nested.py +713 -0
  476. scitex/io/bundle/_types.py +74 -0
  477. scitex/io/bundle/_zip.py +487 -0
  478. scitex/io/utils/h5_to_zarr.py +1 -1
  479. scitex/logging/__init__.py +108 -13
  480. scitex/logging/_errors.py +508 -0
  481. scitex/logging/_formatters.py +30 -6
  482. scitex/logging/_warnings.py +261 -0
  483. scitex/plt/__init__.py +4 -1
  484. scitex/plt/_figrecipe.py +236 -0
  485. scitex/plt/_subplots/_AxisWrapper.py +6 -0
  486. scitex/plt/_subplots/_AxisWrapperMixins/_AdjustmentMixin/__init__.py +0 -0
  487. scitex/plt/_subplots/_AxisWrapperMixins/_AdjustmentMixin/_labels.py +0 -0
  488. scitex/plt/_subplots/_AxisWrapperMixins/_AdjustmentMixin/_metadata.py +0 -0
  489. scitex/plt/_subplots/_AxisWrapperMixins/_AdjustmentMixin/_visual.py +0 -0
  490. scitex/plt/_subplots/_AxisWrapperMixins/_MatplotlibPlotMixin/__init__.py +0 -0
  491. scitex/plt/_subplots/_AxisWrapperMixins/_MatplotlibPlotMixin/_base.py +0 -0
  492. scitex/plt/_subplots/_AxisWrapperMixins/_MatplotlibPlotMixin/_scientific.py +0 -0
  493. scitex/plt/_subplots/_AxisWrapperMixins/_MatplotlibPlotMixin/_statistical.py +0 -0
  494. scitex/plt/_subplots/_AxisWrapperMixins/_MatplotlibPlotMixin/_stx_aliases.py +0 -0
  495. scitex/plt/_subplots/_AxisWrapperMixins/_RawMatplotlibMixin.py +0 -0
  496. scitex/plt/_subplots/_AxisWrapperMixins/_SeabornMixin/__init__.py +0 -0
  497. scitex/plt/_subplots/_AxisWrapperMixins/_SeabornMixin/_base.py +0 -0
  498. scitex/plt/_subplots/_AxisWrapperMixins/_SeabornMixin/_wrappers.py +0 -0
  499. scitex/plt/_subplots/_AxisWrapperMixins/_UnitAwareMixin.py +112 -1
  500. scitex/plt/_subplots/_FigWrapper.py +15 -0
  501. scitex/plt/_subplots/_SubplotsWrapper.py +125 -489
  502. scitex/plt/_subplots/_export_as_csv.py +11 -0
  503. scitex/plt/_subplots/_export_as_csv_formatters/__init__.py +2 -0
  504. scitex/plt/_subplots/_export_as_csv_formatters/_format_pcolormesh.py +66 -0
  505. scitex/plt/_subplots/_export_as_csv_formatters/_format_stackplot.py +62 -0
  506. scitex/plt/_subplots/_export_as_csv_formatters/_format_stx_bar.py +0 -0
  507. scitex/plt/_subplots/_export_as_csv_formatters/_format_stx_barh.py +0 -0
  508. scitex/plt/_subplots/_export_as_csv_formatters/_format_stx_errorbar.py +0 -0
  509. scitex/plt/_subplots/_export_as_csv_formatters/_format_stx_scatter.py +0 -0
  510. scitex/plt/_subplots/_export_as_csv_formatters/test_formatters.py +208 -0
  511. scitex/plt/_subplots/_fonts.py +71 -0
  512. scitex/plt/_subplots/_mm_layout.py +282 -0
  513. scitex/plt/gallery/__init__.py +99 -2
  514. scitex/plt/io/_layered_bundle.py +0 -0
  515. scitex/plt/styles/_plot_postprocess.py +3 -1
  516. scitex/plt/utils/_configure_mpl.py +16 -19
  517. scitex/repro/_RandomStateManager.py +13 -8
  518. scitex/resource/__init__.py +19 -1
  519. scitex/resource/_utils/_get_env_info.py +13 -25
  520. scitex/schema/__init__.py +149 -160
  521. scitex/schema/_encoding.py +273 -0
  522. scitex/schema/_figure_elements.py +406 -0
  523. scitex/schema/_plot.py +0 -0
  524. scitex/schema/_theme.py +360 -0
  525. scitex/schema/_validation.py +0 -98
  526. scitex/scholar/__init__.py +56 -14
  527. scitex/scholar/auth/ScholarAuthManager.py +1 -1
  528. scitex/scholar/auth/__init__.py +11 -2
  529. scitex/scholar/auth/providers/BaseAuthenticator.py +1 -1
  530. scitex/scholar/auth/providers/EZProxyAuthenticator.py +1 -1
  531. scitex/scholar/auth/providers/OpenAthensAuthenticator.py +1 -1
  532. scitex/scholar/auth/providers/ShibbolethAuthenticator.py +1 -1
  533. scitex/scholar/config/ScholarConfig.py +1 -1
  534. scitex/scholar/core/Scholar.py +1 -1
  535. scitex/session/_decorator.py +18 -16
  536. scitex/session/_lifecycle.py +9 -11
  537. scitex/session/template.py +9 -8
  538. scitex/sh/test_sh.py +72 -0
  539. scitex/sh/test_sh_simple.py +61 -0
  540. scitex/stats/__init__.py +221 -97
  541. scitex/stats/_schema.py +21 -22
  542. scitex/stats/descriptive/_circular.py +212 -351
  543. scitex/stats/descriptive/_describe.py +81 -132
  544. scitex/stats/descriptive/_nan.py +205 -433
  545. scitex/stats/descriptive/_real.py +127 -141
  546. scitex/str/_format_plot_text.py +5 -5
  547. scitex/str/_latex.py +26 -84
  548. scitex/str/_latex_fallback.py +53 -47
  549. scitex/web/_search_pubmed.py +5 -4
  550. scitex/writer/tests/test_diff_between.py +451 -0
  551. scitex/writer/tests/test_document_section.py +311 -0
  552. scitex/writer/tests/test_document_workflow.py +393 -0
  553. scitex/writer/tests/test_writer.py +361 -0
  554. scitex/writer/tests/test_writer_integration.py +303 -0
  555. {scitex-2.7.3.dist-info → scitex-2.10.0.dist-info}/METADATA +364 -181
  556. {scitex-2.7.3.dist-info → scitex-2.10.0.dist-info}/RECORD +479 -108
  557. scitex/fig/editor/_edit.py +0 -751
  558. scitex/scholar/docs/to_claude/guidelines/examples/mgmt/ARCHITECTURE_EXAMPLE.md +0 -905
  559. scitex/scholar/docs/to_claude/guidelines/examples/mgmt/BULLETIN_BOARD_EXAMPLE.md +0 -99
  560. scitex/scholar/docs/to_claude/guidelines/examples/mgmt/PROJECT_DESCRIPTION_EXAMPLE.md +0 -96
  561. {scitex-2.7.3.dist-info → scitex-2.10.0.dist-info}/WHEEL +0 -0
  562. {scitex-2.7.3.dist-info → scitex-2.10.0.dist-info}/entry_points.txt +0 -0
  563. {scitex-2.7.3.dist-info → scitex-2.10.0.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,992 @@
1
+ /**
2
+ * Element Inspector - Visual Debugging Tool
3
+ * Shows all HTML elements with colored rectangles and labels.
4
+ *
5
+ * Shortcuts:
6
+ * Alt+I: Toggle inspector overlay
7
+ * Ctrl+Alt+I: Rectangle selection mode
8
+ * Ctrl+Shift+I: Debug snapshot (console logs + page info)
9
+ * Escape: Deactivate inspector / Cancel selection
10
+ */
11
+
12
+ (function() {
13
+ 'use strict';
14
+
15
+ // ============================================================================
16
+ // State
17
+ // ============================================================================
18
+ var isActive = false;
19
+ var overlayContainer = null;
20
+ var elementBoxMap = new Map();
21
+ var currentlyHoveredBox = null;
22
+ var currentlyHoveredElement = null;
23
+ var consoleLogs = [];
24
+ var maxLogs = 500;
25
+ var onCopyCallback = null;
26
+
27
+ // Selection mode state
28
+ var selectionMode = false;
29
+ var selectionStart = null;
30
+ var selectionRect = null;
31
+ var selectionOverlay = null;
32
+ var currentlySelectedElements = new Set();
33
+
34
+ // ============================================================================
35
+ // Console Capture (start immediately)
36
+ // ============================================================================
37
+ var originalConsole = {
38
+ log: console.log.bind(console),
39
+ warn: console.warn.bind(console),
40
+ error: console.error.bind(console),
41
+ info: console.info.bind(console)
42
+ };
43
+
44
+ function captureLog(type, args) {
45
+ var entry = {
46
+ type: type,
47
+ timestamp: new Date().toISOString(),
48
+ args: args.map(function(arg) { return stringify(arg); }),
49
+ source: getCallSource()
50
+ };
51
+ consoleLogs.push(entry);
52
+ if (consoleLogs.length > maxLogs) {
53
+ consoleLogs.shift();
54
+ }
55
+ }
56
+
57
+ function getCallSource() {
58
+ try {
59
+ var stack = new Error().stack;
60
+ if (!stack) return '';
61
+ var lines = stack.split('\n');
62
+ for (var i = 4; i < lines.length; i++) {
63
+ var line = lines[i];
64
+ var match = line.match(/(?:at\s+)?(?:.*?\s+\()?([^\s()]+):(\d+):(\d+)\)?$/);
65
+ if (match) {
66
+ var file = match[1];
67
+ var lineNum = match[2];
68
+ var fileName = file.split('/').pop() || file;
69
+ if (fileName.includes('element-inspector')) continue;
70
+ return fileName + ':' + lineNum;
71
+ }
72
+ }
73
+ } catch (e) {}
74
+ return '';
75
+ }
76
+
77
+ function stringify(obj) {
78
+ if (obj === null) return 'null';
79
+ if (obj === undefined) return 'undefined';
80
+ if (typeof obj === 'string') return obj;
81
+ if (typeof obj === 'number' || typeof obj === 'boolean') return String(obj);
82
+ if (obj instanceof Error) {
83
+ return obj.name + ': ' + obj.message + '\n' + (obj.stack || '');
84
+ }
85
+ try {
86
+ return JSON.stringify(obj, null, 2);
87
+ } catch (e) {
88
+ return String(obj);
89
+ }
90
+ }
91
+
92
+ // Override console methods
93
+ console.log = function() {
94
+ captureLog('log', Array.from(arguments));
95
+ originalConsole.log.apply(console, arguments);
96
+ };
97
+ console.warn = function() {
98
+ captureLog('warn', Array.from(arguments));
99
+ originalConsole.warn.apply(console, arguments);
100
+ };
101
+ console.error = function() {
102
+ captureLog('error', Array.from(arguments));
103
+ originalConsole.error.apply(console, arguments);
104
+ };
105
+ console.info = function() {
106
+ captureLog('info', Array.from(arguments));
107
+ originalConsole.info.apply(console, arguments);
108
+ };
109
+
110
+ // ============================================================================
111
+ // Notification Manager
112
+ // ============================================================================
113
+ function showNotification(message, type) {
114
+ var notification = document.createElement('div');
115
+ notification.className = 'element-inspector-notification ' + type;
116
+ notification.textContent = message;
117
+ document.body.appendChild(notification);
118
+
119
+ requestAnimationFrame(function() {
120
+ notification.classList.add('visible');
121
+ });
122
+
123
+ setTimeout(function() {
124
+ notification.classList.remove('visible');
125
+ setTimeout(function() { notification.remove(); }, 200);
126
+ }, 1000);
127
+ }
128
+
129
+ function showCameraFlash() {
130
+ var flash = document.createElement('div');
131
+ flash.className = 'camera-flash';
132
+ document.body.appendChild(flash);
133
+ setTimeout(function() { flash.remove(); }, 300);
134
+ }
135
+
136
+ function triggerCopyCallback() {
137
+ if (onCopyCallback) {
138
+ setTimeout(function() {
139
+ onCopyCallback();
140
+ }, 400);
141
+ }
142
+ }
143
+
144
+ // ============================================================================
145
+ // Debug Info Collector
146
+ // ============================================================================
147
+ function buildCSSSelector(element) {
148
+ var tag = element.tagName.toLowerCase();
149
+ var id = element.id;
150
+ var classes = element.className;
151
+
152
+ var selector = tag;
153
+ if (id) selector += '#' + id;
154
+ if (classes && typeof classes === 'string') {
155
+ var classList = classes.split(/\s+/).filter(function(c) { return c; });
156
+ if (classList.length > 0) {
157
+ selector += '.' + classList.join('.');
158
+ }
159
+ }
160
+ return selector;
161
+ }
162
+
163
+ function getXPath(element) {
164
+ if (element.id) {
165
+ return '//*[@id="' + element.id + '"]';
166
+ }
167
+
168
+ var parts = [];
169
+ var current = element;
170
+
171
+ while (current && current.nodeType === Node.ELEMENT_NODE) {
172
+ var index = 0;
173
+ var sibling = current.previousSibling;
174
+
175
+ while (sibling) {
176
+ if (sibling.nodeType === Node.ELEMENT_NODE && sibling.nodeName === current.nodeName) {
177
+ index++;
178
+ }
179
+ sibling = sibling.previousSibling;
180
+ }
181
+
182
+ var tagName = current.nodeName.toLowerCase();
183
+ var pathIndex = index > 0 ? '[' + (index + 1) + ']' : '';
184
+ parts.unshift(tagName + pathIndex);
185
+ current = current.parentElement;
186
+ }
187
+
188
+ return '/' + parts.join('/');
189
+ }
190
+
191
+ function getParentChain(element) {
192
+ var chain = [];
193
+ var current = element.parentElement;
194
+ var depth = 0;
195
+
196
+ while (current && depth < 5) {
197
+ chain.push(buildCSSSelector(current));
198
+ current = current.parentElement;
199
+ depth++;
200
+ }
201
+
202
+ return chain;
203
+ }
204
+
205
+ function gatherElementDebugInfo(element) {
206
+ var info = {};
207
+
208
+ info.url = window.location.href;
209
+ info.timestamp = new Date().toISOString();
210
+
211
+ var className = typeof element.className === 'string' ? element.className : '';
212
+ info.element = {
213
+ tag: element.tagName.toLowerCase(),
214
+ id: element.id || null,
215
+ classes: className ? className.split(/\s+/).filter(function(c) { return c; }) : [],
216
+ selector: buildCSSSelector(element),
217
+ xpath: getXPath(element)
218
+ };
219
+
220
+ info.attributes = {};
221
+ for (var i = 0; i < element.attributes.length; i++) {
222
+ var attr = element.attributes[i];
223
+ info.attributes[attr.name] = attr.value;
224
+ }
225
+
226
+ if (element instanceof HTMLElement) {
227
+ var computed = window.getComputedStyle(element);
228
+ info.styles = {
229
+ display: computed.display,
230
+ position: computed.position,
231
+ width: computed.width,
232
+ height: computed.height,
233
+ backgroundColor: computed.backgroundColor,
234
+ color: computed.color,
235
+ fontSize: computed.fontSize
236
+ };
237
+
238
+ var rect = element.getBoundingClientRect();
239
+ info.dimensions = {
240
+ width: rect.width,
241
+ height: rect.height,
242
+ top: rect.top,
243
+ left: rect.left
244
+ };
245
+
246
+ info.content = {
247
+ textContent: (element.textContent || '').substring(0, 200)
248
+ };
249
+ }
250
+
251
+ info.parentChain = getParentChain(element);
252
+
253
+ return formatDebugInfoForAI(info);
254
+ }
255
+
256
+ function formatDebugInfoForAI(info) {
257
+ return '# Element Debug Information\n\n' +
258
+ '## Page Context\n' +
259
+ '- URL: ' + info.url + '\n' +
260
+ '- Timestamp: ' + info.timestamp + '\n\n' +
261
+ '## Element Identification\n' +
262
+ '- Tag: <' + info.element.tag + '>\n' +
263
+ '- ID: ' + (info.element.id || 'none') + '\n' +
264
+ '- Classes: ' + (info.element.classes.join(', ') || 'none') + '\n' +
265
+ '- CSS Selector: ' + info.element.selector + '\n' +
266
+ '- XPath: ' + info.element.xpath + '\n\n' +
267
+ '## Attributes\n' +
268
+ Object.entries(info.attributes).map(function(entry) {
269
+ return '- ' + entry[0] + ': ' + entry[1];
270
+ }).join('\n') + '\n\n' +
271
+ '## Computed Styles\n' +
272
+ Object.entries(info.styles || {}).map(function(entry) {
273
+ return '- ' + entry[0] + ': ' + entry[1];
274
+ }).join('\n') + '\n\n' +
275
+ '## Dimensions & Position\n' +
276
+ '- Width: ' + (info.dimensions ? info.dimensions.width : 0) + 'px\n' +
277
+ '- Height: ' + (info.dimensions ? info.dimensions.height : 0) + 'px\n' +
278
+ '- Top: ' + (info.dimensions ? info.dimensions.top : 0) + 'px\n' +
279
+ '- Left: ' + (info.dimensions ? info.dimensions.left : 0) + 'px\n\n' +
280
+ '## Content (truncated)\n' +
281
+ (info.content ? info.content.textContent : 'none') + '\n\n' +
282
+ '## Parent Chain\n' +
283
+ info.parentChain.map(function(p, i) { return (i + 1) + '. ' + p; }).join('\n') + '\n\n' +
284
+ '---\nGenerated by Element Inspector';
285
+ }
286
+
287
+ // ============================================================================
288
+ // Element Scanner
289
+ // ============================================================================
290
+ function getDepth(element) {
291
+ var depth = 0;
292
+ var current = element;
293
+
294
+ while (current && current !== document.body) {
295
+ depth++;
296
+ current = current.parentElement;
297
+ }
298
+
299
+ return depth;
300
+ }
301
+
302
+ function getColorForDepth(depth) {
303
+ var colors = [
304
+ '#3B82F6', // Blue (depth 0-2)
305
+ '#10B981', // Green (depth 3-5)
306
+ '#F59E0B', // Yellow (depth 6-8)
307
+ '#EF4444', // Red (depth 9-11)
308
+ '#EC4899' // Pink (depth 12+)
309
+ ];
310
+
311
+ var index = Math.min(Math.floor(depth / 3), colors.length - 1);
312
+ return colors[index];
313
+ }
314
+
315
+ function shouldShowLabel(element, rect, depth) {
316
+ if (element.id) {
317
+ return rect.width > 20 && rect.height > 20;
318
+ }
319
+ if (rect.width > 100 || rect.height > 100) {
320
+ return true;
321
+ }
322
+ var importantTags = ['header', 'nav', 'main', 'section', 'article', 'aside', 'footer', 'form', 'table'];
323
+ if (importantTags.indexOf(element.tagName.toLowerCase()) !== -1 && (rect.width > 50 || rect.height > 50)) {
324
+ return true;
325
+ }
326
+ var interactiveTags = ['button', 'a', 'input', 'select', 'textarea'];
327
+ if (interactiveTags.indexOf(element.tagName.toLowerCase()) !== -1 && (rect.width > 30 || rect.height > 30)) {
328
+ return true;
329
+ }
330
+ if (depth > 8 && rect.width < 100 && rect.height < 100) {
331
+ return false;
332
+ }
333
+ return false;
334
+ }
335
+
336
+ function scanElements(container) {
337
+ var allElements = document.querySelectorAll('*');
338
+ var count = 0;
339
+ var occupiedPositions = [];
340
+
341
+ allElements.forEach(function(element) {
342
+ if (element.closest('#element-inspector-overlay')) return;
343
+
344
+ if (element instanceof HTMLElement) {
345
+ var computed = window.getComputedStyle(element);
346
+ if (computed.display === 'none' || computed.visibility === 'hidden') return;
347
+ }
348
+
349
+ var depth = getDepth(element);
350
+ var color = getColorForDepth(depth);
351
+
352
+ var rect = element.getBoundingClientRect();
353
+ if (rect.width === 0 || rect.height === 0) return;
354
+
355
+ var box = document.createElement('div');
356
+ box.className = 'element-inspector-box';
357
+ box.style.cssText =
358
+ 'top: ' + (rect.top + window.scrollY) + 'px;' +
359
+ 'left: ' + (rect.left + window.scrollX) + 'px;' +
360
+ 'width: ' + rect.width + 'px;' +
361
+ 'height: ' + rect.height + 'px;' +
362
+ 'border-color: ' + color + ';';
363
+
364
+ var tag = element.tagName.toLowerCase();
365
+ var id = element.id ? '#' + element.id : '';
366
+ box.title = 'Click to copy debug info: ' + tag + id;
367
+
368
+ elementBoxMap.set(box, element);
369
+
370
+ box.addEventListener('mouseenter', function() {
371
+ currentlyHoveredBox = box;
372
+ currentlyHoveredElement = element;
373
+ });
374
+
375
+ box.addEventListener('mouseleave', function() {
376
+ if (currentlyHoveredBox === box) {
377
+ currentlyHoveredBox = null;
378
+ currentlyHoveredElement = null;
379
+ }
380
+ });
381
+
382
+ box.addEventListener('click', function(e) {
383
+ e.preventDefault();
384
+ e.stopPropagation();
385
+
386
+ var selectedElement = currentlyHoveredElement || element;
387
+ var selectedBox = currentlyHoveredBox || box;
388
+
389
+ selectedBox.classList.add('highlighted');
390
+
391
+ var debugInfo = gatherElementDebugInfo(selectedElement);
392
+ navigator.clipboard.writeText(debugInfo).then(function() {
393
+ showNotification('✓ Copied!', 'success');
394
+ console.log('[ElementInspector] Copied:', debugInfo);
395
+ triggerCopyCallback();
396
+ }).catch(function(err) {
397
+ console.error('[ElementInspector] Copy failed:', err);
398
+ showNotification('✗ Copy Failed', 'error');
399
+ selectedBox.classList.remove('highlighted');
400
+ });
401
+ });
402
+
403
+ if (shouldShowLabel(element, rect, depth)) {
404
+ var label = createLabel(element, depth);
405
+ if (label) {
406
+ var labelPos = findLabelPosition(rect, occupiedPositions);
407
+ if (labelPos.isValid) {
408
+ label.style.top = labelPos.top + 'px';
409
+ label.style.left = labelPos.left + 'px';
410
+ addCopyToClipboard(label, element);
411
+ addHoverHighlight(label, box, element);
412
+
413
+ occupiedPositions.push({
414
+ top: labelPos.top - 8,
415
+ left: labelPos.left - 8,
416
+ bottom: labelPos.top + 20 + 8,
417
+ right: labelPos.left + 250 + 8
418
+ });
419
+
420
+ container.appendChild(label);
421
+ }
422
+ }
423
+ }
424
+
425
+ container.appendChild(box);
426
+ count++;
427
+ });
428
+
429
+ console.log('[ElementInspector] Visualized ' + count + ' elements');
430
+ }
431
+
432
+ function createLabel(element, depth) {
433
+ var tag = element.tagName.toLowerCase();
434
+ var id = element.id;
435
+ var classes = element.className;
436
+
437
+ var labelText = '<span class="element-inspector-label-tag">' + tag + '</span>';
438
+
439
+ if (id) {
440
+ labelText += ' <span class="element-inspector-label-id">#' + id + '</span>';
441
+ }
442
+
443
+ if (classes && typeof classes === 'string') {
444
+ var classList = classes.split(/\s+/).filter(function(c) { return c.length > 0; });
445
+ if (classList.length > 0) {
446
+ var classPreview = classList.slice(0, 2).join('.');
447
+ labelText += ' <span class="element-inspector-label-class">.' + classPreview + '</span>';
448
+ if (classList.length > 2) {
449
+ labelText += '<span class="element-inspector-label-class">+' + (classList.length - 2) + '</span>';
450
+ }
451
+ }
452
+ }
453
+
454
+ if (depth > 5) {
455
+ labelText += ' <span style="color: #999; font-size: 9px;">d' + depth + '</span>';
456
+ }
457
+
458
+ var label = document.createElement('div');
459
+ label.className = 'element-inspector-label';
460
+ label.innerHTML = labelText;
461
+ label.title = 'Click to copy comprehensive debug info for AI';
462
+
463
+ return label;
464
+ }
465
+
466
+ function findLabelPosition(rect, occupiedPositions) {
467
+ var scrollY = window.scrollY;
468
+ var scrollX = window.scrollX;
469
+
470
+ var positions = [
471
+ { top: rect.top + scrollY - 24, left: rect.left + scrollX },
472
+ { top: rect.top + scrollY - 24, left: rect.right + scrollX - 200 },
473
+ { top: rect.top + scrollY + 4, left: rect.left + scrollX + 4 },
474
+ { top: rect.bottom + scrollY + 4, left: rect.left + scrollX }
475
+ ];
476
+
477
+ for (var i = 0; i < positions.length; i++) {
478
+ if (!isPositionOccupied(positions[i], occupiedPositions)) {
479
+ return { top: positions[i].top, left: positions[i].left, isValid: true };
480
+ }
481
+ }
482
+
483
+ return { top: 0, left: 0, isValid: false };
484
+ }
485
+
486
+ function isPositionOccupied(pos, occupiedPositions) {
487
+ var labelWidth = 250;
488
+ var labelHeight = 20;
489
+
490
+ for (var i = 0; i < occupiedPositions.length; i++) {
491
+ var occupied = occupiedPositions[i];
492
+ if (!(pos.left + labelWidth < occupied.left ||
493
+ pos.left > occupied.right ||
494
+ pos.top + labelHeight < occupied.top ||
495
+ pos.top > occupied.bottom)) {
496
+ return true;
497
+ }
498
+ }
499
+ return false;
500
+ }
501
+
502
+ function addHoverHighlight(label, box, element) {
503
+ label.addEventListener('mouseenter', function() {
504
+ currentlyHoveredBox = box;
505
+ currentlyHoveredElement = element;
506
+ box.classList.add('highlighted');
507
+ if (element instanceof HTMLElement) {
508
+ element.style.outline = '3px solid rgba(59, 130, 246, 0.8)';
509
+ element.style.outlineOffset = '2px';
510
+ }
511
+ });
512
+
513
+ label.addEventListener('mouseleave', function() {
514
+ currentlyHoveredBox = null;
515
+ currentlyHoveredElement = null;
516
+ box.classList.remove('highlighted');
517
+ if (element instanceof HTMLElement) {
518
+ element.style.outline = '';
519
+ element.style.outlineOffset = '';
520
+ }
521
+ });
522
+ }
523
+
524
+ function addCopyToClipboard(label, element) {
525
+ label.addEventListener('click', function(e) {
526
+ e.stopPropagation();
527
+ e.preventDefault();
528
+
529
+ var debugInfo = gatherElementDebugInfo(element);
530
+
531
+ navigator.clipboard.writeText(debugInfo).then(function() {
532
+ showNotification('✓ Copied!', 'success');
533
+ console.log('[ElementInspector] Copied debug info to clipboard');
534
+ triggerCopyCallback();
535
+ }).catch(function(err) {
536
+ console.error('[ElementInspector] Failed to copy:', err);
537
+ showNotification('✗ Copy Failed', 'error');
538
+ });
539
+ });
540
+ }
541
+
542
+ // ============================================================================
543
+ // Selection Mode
544
+ // ============================================================================
545
+ function startSelectionMode() {
546
+ if (!isActive) {
547
+ activate();
548
+ }
549
+
550
+ selectionMode = true;
551
+ document.body.classList.add('element-inspector-selection-mode');
552
+
553
+ selectionOverlay = document.createElement('div');
554
+ selectionOverlay.className = 'selection-overlay';
555
+ document.body.appendChild(selectionOverlay);
556
+
557
+ showNotification('Drag to select area', 'success');
558
+
559
+ document.addEventListener('mousedown', onSelectionMouseDown);
560
+ document.addEventListener('mousemove', onSelectionMouseMove);
561
+ document.addEventListener('mouseup', onSelectionMouseUp);
562
+ }
563
+
564
+ function cancelSelectionMode() {
565
+ selectionMode = false;
566
+ document.body.classList.remove('element-inspector-selection-mode');
567
+
568
+ if (selectionOverlay) {
569
+ selectionOverlay.remove();
570
+ selectionOverlay = null;
571
+ }
572
+
573
+ if (selectionRect) {
574
+ selectionRect.remove();
575
+ selectionRect = null;
576
+ }
577
+
578
+ document.removeEventListener('mousedown', onSelectionMouseDown);
579
+ document.removeEventListener('mousemove', onSelectionMouseMove);
580
+ document.removeEventListener('mouseup', onSelectionMouseUp);
581
+
582
+ selectionStart = null;
583
+ currentlySelectedElements.clear();
584
+ }
585
+
586
+ function onSelectionMouseDown(e) {
587
+ if (!selectionMode) return;
588
+
589
+ e.preventDefault();
590
+ selectionStart = { x: e.clientX, y: e.clientY };
591
+
592
+ selectionRect = document.createElement('div');
593
+ selectionRect.className = 'selection-rectangle';
594
+ selectionRect.style.left = e.clientX + 'px';
595
+ selectionRect.style.top = e.clientY + 'px';
596
+ selectionRect.style.width = '0px';
597
+ selectionRect.style.height = '0px';
598
+
599
+ document.body.appendChild(selectionRect);
600
+ }
601
+
602
+ function onSelectionMouseMove(e) {
603
+ if (!selectionMode || !selectionStart || !selectionRect) return;
604
+
605
+ e.preventDefault();
606
+
607
+ var left = Math.min(selectionStart.x, e.clientX);
608
+ var top = Math.min(selectionStart.y, e.clientY);
609
+ var width = Math.abs(e.clientX - selectionStart.x);
610
+ var height = Math.abs(e.clientY - selectionStart.y);
611
+
612
+ selectionRect.style.left = left + 'px';
613
+ selectionRect.style.top = top + 'px';
614
+ selectionRect.style.width = width + 'px';
615
+ selectionRect.style.height = height + 'px';
616
+ }
617
+
618
+ function onSelectionMouseUp(e) {
619
+ if (!selectionMode || !selectionStart || !selectionRect) return;
620
+
621
+ e.preventDefault();
622
+
623
+ var left = Math.min(selectionStart.x, e.clientX);
624
+ var top = Math.min(selectionStart.y, e.clientY);
625
+ var width = Math.abs(e.clientX - selectionStart.x);
626
+ var height = Math.abs(e.clientY - selectionStart.y);
627
+
628
+ if (width < 5 || height < 5) {
629
+ cancelSelectionMode();
630
+ showNotification('Selection too small', 'error');
631
+ return;
632
+ }
633
+
634
+ var selectedElements = findElementsInRect({ left: left, top: top, width: width, height: height });
635
+
636
+ console.log('[ElementInspector] Found ' + selectedElements.length + ' elements in selection');
637
+
638
+ var selectionInfo = gatherSelectionInfo(selectedElements, { left: left, top: top, width: width, height: height });
639
+
640
+ navigator.clipboard.writeText(selectionInfo).then(function() {
641
+ showNotification('✓ ' + selectedElements.length + ' elements copied!', 'success');
642
+ triggerCopyCallback();
643
+ }).catch(function(err) {
644
+ console.error('[ElementInspector] Failed to copy:', err);
645
+ showNotification('✗ Copy Failed', 'error');
646
+ });
647
+
648
+ cancelSelectionMode();
649
+ }
650
+
651
+ function findElementsInRect(rect) {
652
+ var selectedElements = [];
653
+ var allElements = document.querySelectorAll('*');
654
+
655
+ var selectionRect = {
656
+ left: rect.left,
657
+ top: rect.top,
658
+ right: rect.left + rect.width,
659
+ bottom: rect.top + rect.height
660
+ };
661
+
662
+ allElements.forEach(function(element) {
663
+ if (element.closest('#element-inspector-overlay') ||
664
+ element.classList.contains('selection-rectangle') ||
665
+ element.classList.contains('selection-overlay')) {
666
+ return;
667
+ }
668
+
669
+ if (element instanceof HTMLElement) {
670
+ var computed = window.getComputedStyle(element);
671
+ if (computed.display === 'none' || computed.visibility === 'hidden') return;
672
+ }
673
+
674
+ var elementRect = element.getBoundingClientRect();
675
+ var intersects = !(
676
+ elementRect.right < selectionRect.left ||
677
+ elementRect.left > selectionRect.right ||
678
+ elementRect.bottom < selectionRect.top ||
679
+ elementRect.top > selectionRect.bottom
680
+ );
681
+
682
+ if (intersects) {
683
+ selectedElements.push(element);
684
+ }
685
+ });
686
+
687
+ return selectedElements;
688
+ }
689
+
690
+ function gatherSelectionInfo(elements, rect) {
691
+ var info = '# Rectangle Selection Debug Information\n\n' +
692
+ '## Selection Area\n' +
693
+ '- Position: (' + Math.round(rect.left) + ', ' + Math.round(rect.top) + ')\n' +
694
+ '- Size: ' + Math.round(rect.width) + 'x' + Math.round(rect.height) + 'px\n' +
695
+ '- URL: ' + window.location.href + '\n' +
696
+ '- Elements Found: ' + elements.length + '\n\n---\n\n';
697
+
698
+ var elementTypes = {};
699
+ elements.forEach(function(el) {
700
+ var tag = el.tagName.toLowerCase();
701
+ elementTypes[tag] = (elementTypes[tag] || 0) + 1;
702
+ });
703
+
704
+ info += '## Element Type Summary\n';
705
+ Object.entries(elementTypes)
706
+ .sort(function(a, b) { return b[1] - a[1]; })
707
+ .forEach(function(entry) {
708
+ info += '- ' + entry[0] + ': ' + entry[1] + '\n';
709
+ });
710
+
711
+ info += '\n---\n\n## Detailed Element Information (first 10 elements)\n\n';
712
+
713
+ elements.slice(0, 10).forEach(function(element, index) {
714
+ info += '### Element ' + (index + 1) + '\n';
715
+ info += gatherElementDebugInfo(element);
716
+ info += '\n---\n\n';
717
+ });
718
+
719
+ return info;
720
+ }
721
+
722
+ // ============================================================================
723
+ // Debug Snapshot (Screenshot + Console Logs)
724
+ // ============================================================================
725
+ function captureDebugSnapshot() {
726
+ showCameraFlash();
727
+
728
+ // Run screenshot and console log capture in parallel
729
+ Promise.all([
730
+ captureScreenshot(),
731
+ captureConsoleLogs()
732
+ ]).then(function(results) {
733
+ var screenshotOk = results[0];
734
+ var logsOk = results[1];
735
+
736
+ if (screenshotOk && logsOk) {
737
+ showNotification('✓ Screenshot + logs copied', 'success');
738
+ } else if (screenshotOk) {
739
+ showNotification('✓ Screenshot copied', 'success');
740
+ } else if (logsOk) {
741
+ showNotification('✓ Console logs copied', 'success');
742
+ } else {
743
+ showNotification('✗ Capture failed', 'error');
744
+ }
745
+ });
746
+ }
747
+
748
+ function captureScreenshot() {
749
+ return navigator.mediaDevices.getDisplayMedia({
750
+ video: {
751
+ displaySurface: 'browser'
752
+ },
753
+ preferCurrentTab: true,
754
+ selfBrowserSurface: 'include',
755
+ systemAudio: 'exclude'
756
+ }).then(function(stream) {
757
+ var video = document.createElement('video');
758
+ video.srcObject = stream;
759
+ video.muted = true;
760
+
761
+ return new Promise(function(resolve) {
762
+ video.onloadedmetadata = function() {
763
+ video.play().then(function() {
764
+ // Small delay to ensure frame is rendered
765
+ setTimeout(function() {
766
+ var canvas = document.createElement('canvas');
767
+ canvas.width = video.videoWidth;
768
+ canvas.height = video.videoHeight;
769
+ var ctx = canvas.getContext('2d');
770
+
771
+ if (!ctx) {
772
+ stream.getTracks().forEach(function(t) { t.stop(); });
773
+ resolve(false);
774
+ return;
775
+ }
776
+
777
+ ctx.drawImage(video, 0, 0);
778
+ stream.getTracks().forEach(function(t) { t.stop(); });
779
+
780
+ // Convert to blob and copy to clipboard
781
+ canvas.toBlob(function(blob) {
782
+ if (!blob) {
783
+ resolve(false);
784
+ return;
785
+ }
786
+
787
+ navigator.clipboard.write([
788
+ new ClipboardItem({ 'image/png': blob })
789
+ ]).then(function() {
790
+ console.log('[ElementInspector] Screenshot copied to clipboard');
791
+ resolve(true);
792
+ }).catch(function(err) {
793
+ console.error('[ElementInspector] Screenshot clipboard write failed:', err);
794
+ resolve(false);
795
+ });
796
+ }, 'image/png');
797
+ }, 100);
798
+ }).catch(function() {
799
+ stream.getTracks().forEach(function(t) { t.stop(); });
800
+ resolve(false);
801
+ });
802
+ };
803
+
804
+ video.onerror = function() {
805
+ stream.getTracks().forEach(function(t) { t.stop(); });
806
+ resolve(false);
807
+ };
808
+
809
+ // Timeout fallback
810
+ setTimeout(function() {
811
+ stream.getTracks().forEach(function(t) { t.stop(); });
812
+ resolve(false);
813
+ }, 5000);
814
+ });
815
+ }).catch(function(err) {
816
+ // User cancelled or permission denied - this is normal
817
+ if (err.name !== 'NotAllowedError') {
818
+ console.error('[ElementInspector] Screenshot capture failed:', err);
819
+ }
820
+ return false;
821
+ });
822
+ }
823
+
824
+ function captureConsoleLogs() {
825
+ var logs = getConsoleLogs();
826
+ return navigator.clipboard.writeText(logs).then(function() {
827
+ console.log('[ElementInspector] Console logs copied to clipboard');
828
+ return true;
829
+ }).catch(function(err) {
830
+ console.error('[ElementInspector] Failed to copy logs:', err);
831
+ return false;
832
+ });
833
+ }
834
+
835
+ function getConsoleLogs() {
836
+ if (consoleLogs.length === 0) {
837
+ return 'No console logs captured.';
838
+ }
839
+
840
+ var output = '# Console Logs\n\n';
841
+ output += 'URL: ' + window.location.href + '\n';
842
+ output += 'Timestamp: ' + new Date().toISOString() + '\n';
843
+ output += 'Total Logs: ' + consoleLogs.length + '\n\n---\n\n';
844
+
845
+ consoleLogs.forEach(function(entry) {
846
+ var icon = entry.type === 'error' ? '❌' : entry.type === 'warn' ? '⚠️' : '📝';
847
+ var source = entry.source ? entry.source + ' ' : '';
848
+ output += icon + ' ' + source + entry.args.join(' ') + '\n';
849
+ });
850
+
851
+ return output;
852
+ }
853
+
854
+ // ============================================================================
855
+ // Main Functions
856
+ // ============================================================================
857
+ function activate() {
858
+ console.log('[ElementInspector] Activating...');
859
+ isActive = true;
860
+
861
+ // Create overlay container
862
+ overlayContainer = document.createElement('div');
863
+ overlayContainer.id = 'element-inspector-overlay';
864
+
865
+ var docHeight = Math.max(
866
+ document.body.scrollHeight,
867
+ document.documentElement.scrollHeight,
868
+ document.body.offsetHeight,
869
+ document.documentElement.offsetHeight
870
+ );
871
+
872
+ overlayContainer.style.cssText =
873
+ 'position: absolute;' +
874
+ 'top: 0;' +
875
+ 'left: 0;' +
876
+ 'width: 100%;' +
877
+ 'height: ' + docHeight + 'px;' +
878
+ 'pointer-events: none;' +
879
+ 'z-index: 999999;';
880
+
881
+ document.body.appendChild(overlayContainer);
882
+
883
+ // Scan all elements
884
+ scanElements(overlayContainer);
885
+
886
+ console.log('[ElementInspector] Active - Press Alt+I to deactivate');
887
+ }
888
+
889
+ function deactivate() {
890
+ console.log('[ElementInspector] Deactivating...');
891
+ isActive = false;
892
+
893
+ elementBoxMap.clear();
894
+ currentlyHoveredBox = null;
895
+ currentlyHoveredElement = null;
896
+
897
+ if (overlayContainer) {
898
+ overlayContainer.remove();
899
+ overlayContainer = null;
900
+ }
901
+ }
902
+
903
+ function toggle() {
904
+ if (isActive) {
905
+ deactivate();
906
+ } else {
907
+ activate();
908
+ }
909
+ }
910
+
911
+ function refresh() {
912
+ if (isActive) {
913
+ deactivate();
914
+ activate();
915
+ }
916
+ }
917
+
918
+ // ============================================================================
919
+ // Keyboard Shortcuts
920
+ // ============================================================================
921
+ document.addEventListener('keydown', function(e) {
922
+ var key = e.key.toLowerCase();
923
+
924
+ // Ctrl+Shift+I: Debug snapshot
925
+ if (e.ctrlKey && e.shiftKey && !e.altKey && key === 'i') {
926
+ e.preventDefault();
927
+ e.stopPropagation();
928
+ console.log('[ElementInspector] Ctrl+Shift+I pressed - capturing debug snapshot');
929
+ captureDebugSnapshot();
930
+ return;
931
+ }
932
+
933
+ // Ctrl+Alt+I: Start rectangle selection mode
934
+ if (e.ctrlKey && e.altKey && !e.shiftKey && key === 'i') {
935
+ e.preventDefault();
936
+ startSelectionMode();
937
+ return;
938
+ }
939
+
940
+ // Alt+I: Toggle inspector
941
+ if (e.altKey && !e.shiftKey && !e.ctrlKey && key === 'i') {
942
+ e.preventDefault();
943
+ toggle();
944
+ return;
945
+ }
946
+
947
+ // Escape: Deactivate
948
+ if (e.key === 'Escape') {
949
+ e.preventDefault();
950
+ if (selectionMode) {
951
+ cancelSelectionMode();
952
+ deactivate();
953
+ } else if (isActive) {
954
+ deactivate();
955
+ }
956
+ return;
957
+ }
958
+ });
959
+
960
+ // Set up auto-dismiss callback
961
+ onCopyCallback = function() {
962
+ deactivate();
963
+ };
964
+
965
+ // Auto-refresh on window resize
966
+ var resizeTimeout;
967
+ window.addEventListener('resize', function() {
968
+ clearTimeout(resizeTimeout);
969
+ resizeTimeout = setTimeout(function() {
970
+ if (isActive) {
971
+ refresh();
972
+ }
973
+ }, 500);
974
+ });
975
+
976
+ // Export to window for manual control
977
+ window.elementInspector = {
978
+ toggle: toggle,
979
+ activate: activate,
980
+ deactivate: deactivate,
981
+ refresh: refresh,
982
+ captureDebugSnapshot: captureDebugSnapshot,
983
+ getConsoleLogs: getConsoleLogs
984
+ };
985
+
986
+ console.log('[ElementInspector] Initialized');
987
+ console.log(' Alt+I: Toggle inspector overlay');
988
+ console.log(' Ctrl+Alt+I: Rectangle selection mode');
989
+ console.log(' Ctrl+Shift+I: Debug snapshot (console logs)');
990
+ console.log(' Escape: Deactivate inspector');
991
+
992
+ })();