cubevis 1.0.21__tar.gz → 1.0.28__tar.gz

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 (164) hide show
  1. {cubevis-1.0.21 → cubevis-1.0.28}/PKG-INFO +1 -1
  2. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__js__/bokeh-3.8/cubevisjs.min.js +3 -3
  3. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/__init__.py +1 -0
  4. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/models/_showable.py +78 -13
  5. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/_initialize.py +7 -0
  6. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/toolbox/_cube.py +1 -1
  7. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/__init__.py +1 -1
  8. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_jupyter.py +35 -12
  9. {cubevis-1.0.21 → cubevis-1.0.28}/pyproject.toml +1 -1
  10. {cubevis-1.0.21 → cubevis-1.0.28}/LICENSE +0 -0
  11. {cubevis-1.0.21 → cubevis-1.0.28}/README.rst +0 -0
  12. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/LICENSE.rst +0 -0
  13. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/20px/fast-backward.svg +0 -0
  14. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/20px/fast-forward.svg +0 -0
  15. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/20px/step-backward.svg +0 -0
  16. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/20px/step-forward.svg +0 -0
  17. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/add-chan.png +0 -0
  18. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/add-chan.svg +0 -0
  19. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/add-cube.png +0 -0
  20. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/add-cube.svg +0 -0
  21. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/drag.png +0 -0
  22. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/drag.svg +0 -0
  23. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/mask-selected.png +0 -0
  24. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/mask.png +0 -0
  25. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/mask.svg +0 -0
  26. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/new-layer-sm-selected.png +0 -0
  27. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/new-layer-sm-selected.svg +0 -0
  28. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/new-layer-sm.png +0 -0
  29. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/new-layer-sm.svg +0 -0
  30. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/reset.png +0 -0
  31. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/reset.svg +0 -0
  32. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/sub-chan.png +0 -0
  33. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/sub-chan.svg +0 -0
  34. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/sub-cube.png +0 -0
  35. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/sub-cube.svg +0 -0
  36. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/trash.png +0 -0
  37. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/trash.svg +0 -0
  38. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/trash_full.png +0 -0
  39. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/trash_full.svg +0 -0
  40. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/trash_full_raw.png +0 -0
  41. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/zoom-to-fit.png +0 -0
  42. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__icons__/zoom-to-fit.svg +0 -0
  43. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__init__.py +0 -0
  44. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__js__/bokeh-3.6/cubevisjs.min.js +0 -0
  45. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__js__/bokeh-3.7/cubevisjs.min.js +0 -0
  46. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/__js__/casalib.min.js +0 -0
  47. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/annotations/__init__.py +0 -0
  48. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/annotations/_ev_poly_annotation.py +0 -0
  49. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/components/__init__.py +0 -0
  50. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/format/__init__.py +0 -0
  51. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/format/_time_ticks.py +0 -0
  52. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/format/_wcs_ticks.py +0 -0
  53. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/models/__init__.py +0 -0
  54. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/models/_bokeh_app_context.py +0 -0
  55. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/models/_edit_span.py +0 -0
  56. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/models/_ev_text_input.py +0 -0
  57. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/models/_shared_dict.py +0 -0
  58. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/models/_tip.py +0 -0
  59. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/models/_tip_button.py +0 -0
  60. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/sources/__init__.py +0 -0
  61. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/sources/_data_pipe.py +0 -0
  62. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/sources/_image_data_source.py +0 -0
  63. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/sources/_image_pipe.py +0 -0
  64. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/sources/_spectra_data_source.py +0 -0
  65. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/sources/_updatable_data_source.py +0 -0
  66. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/__init__.py +0 -0
  67. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/_current.py +0 -0
  68. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/_javascript.py +0 -0
  69. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/_palette.py +0 -0
  70. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/_session.py +0 -0
  71. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/js/bokeh-2.4.1.min.js +0 -0
  72. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/js/bokeh-gl-2.4.1.min.js +0 -0
  73. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/js/bokeh-tables-2.4.1.min.js +0 -0
  74. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/js/bokeh-widgets-2.4.1.min.js +0 -0
  75. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/js/casaguijs-v0.0.4.0-b2.4.min.js +0 -0
  76. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/js/casaguijs-v0.0.5.0-b2.4.min.js +0 -0
  77. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/js/casaguijs-v0.0.6.0-b2.4.min.js +0 -0
  78. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/state/js/casalib-v0.0.1.min.js +0 -0
  79. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/tools/__init__.py +0 -0
  80. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/tools/_cbreset_tool.py +0 -0
  81. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/tools/_drag_tool.py +0 -0
  82. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/utils/__init__.py +0 -0
  83. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/utils/_axes_labels.py +0 -0
  84. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/bokeh/utils/_svg_icon.py +0 -0
  85. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/data/__init__.py +0 -0
  86. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/data/casaimage/__init__.py +0 -0
  87. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/exe/__init__.py +0 -0
  88. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/exe/_context.py +0 -0
  89. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/exe/_mode.py +0 -0
  90. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/exe/_setting.py +0 -0
  91. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/exe/_task.py +0 -0
  92. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/_gclean.py +0 -0
  93. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/apps/__init__.py +0 -0
  94. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/apps/_createmask.py +0 -0
  95. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/apps/_createregion.py +0 -0
  96. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/apps/_interactiveclean.mustache +0 -0
  97. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/apps/_interactiveclean.py +0 -0
  98. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/apps/_interactivecleannotebook.mustache +0 -0
  99. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/apps/_interactivecleannotebook.py +0 -0
  100. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/apps/_plotants.py +0 -0
  101. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/apps/_plotbandpass.py +0 -0
  102. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/casashell/createmask.py +0 -0
  103. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/casashell/iclean.py +0 -0
  104. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/casatasks/__init__.py +0 -0
  105. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/casatasks/createmask.py +0 -0
  106. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/casatasks/createregion.py +0 -0
  107. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/casatasks/iclean.py +0 -0
  108. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/private/casatasks/iclean_notebook.py +0 -0
  109. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/readme.rst +0 -0
  110. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/remote/__init__.py +0 -0
  111. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/remote/_gclean.py +0 -0
  112. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/remote/_local.py +0 -0
  113. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/remote/_remote_kernel.py +0 -0
  114. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/toolbox/__init__.py +0 -0
  115. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/toolbox/_app_context.py +0 -0
  116. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/toolbox/_interactive_clean_ui.mustache +0 -0
  117. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/toolbox/_interactive_clean_ui.py +0 -0
  118. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/toolbox/_interactiveclean_wrappers.py +0 -0
  119. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/toolbox/_region_list.py +0 -0
  120. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_ResourceManager.py +0 -0
  121. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_browser.py +0 -0
  122. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_contextmgrchain.py +0 -0
  123. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_conversion.py +0 -0
  124. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_copydoc.py +0 -0
  125. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_docenum.py +0 -0
  126. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_git.py +0 -0
  127. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_import_protected_module.py +0 -0
  128. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_logging.py +0 -0
  129. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_mutual_exclusion.py +0 -0
  130. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_pkgs.py +0 -0
  131. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_regions.py +0 -0
  132. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_static.py +0 -0
  133. {cubevis-1.0.21 → cubevis-1.0.28}/cubevis/utils/_tiles.py +0 -0
  134. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/alma-many-chan/alma-many-chan.py +0 -0
  135. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/basic-websockets-demo/client.html +0 -0
  136. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/basic-websockets-demo/client.py +0 -0
  137. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/basic-websockets-demo/server.py +0 -0
  138. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/createmask-demo/run-createmask.py +0 -0
  139. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/createregion-demo/run-createregion.py +0 -0
  140. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/cubemask-demo/image-slider-spectra-done-stats.py +0 -0
  141. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/cubemask-demo/image-slider-spectra-done.py +0 -0
  142. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/cubemask-demo/image-slider-spectra.py +0 -0
  143. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/cubemask-demo/image-slider.py +0 -0
  144. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/cubemask-demo/image.py +0 -0
  145. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-demo/iclean-demo.ipynb +0 -0
  146. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-demo/m100_interactive.py +0 -0
  147. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-demo/mask0-iclean.py +0 -0
  148. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-demo/run-gclean.py +0 -0
  149. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-demo/run-iclean-obj.py +0 -0
  150. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-demo/run-iclean.py +0 -0
  151. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-demo/vla-sim-jet-iclean.py +0 -0
  152. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-first-look/run-fl-cont.py +0 -0
  153. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-first-look/run-fl-line.py +0 -0
  154. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-outlier/run-iclean.py +0 -0
  155. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-outlier/test_outlier.txt +0 -0
  156. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/iclean-remote/iclean_remote_webserver.py +0 -0
  157. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/large-cube/run-largecube.py +0 -0
  158. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/svg-test.py +0 -0
  159. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/updatable-data-source/direct-plot.py +0 -0
  160. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/updatable-data-source/simple-update.py +0 -0
  161. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/updatable-data-source/updated-plot.py +0 -0
  162. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/uranus-demo/uranus-iclean.py +0 -0
  163. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/websocket-reconnect/client.html +0 -0
  164. {cubevis-1.0.21 → cubevis-1.0.28}/tests/manual/websocket-reconnect/server.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: cubevis
3
- Version: 1.0.21
3
+ Version: 1.0.28
4
4
  Summary: visualization toolkit and apps for casa
5
5
  License: LGPL
6
6
  Author-email: Darrell Schiebel <darrell@schiebel.us>,Pam Harris <pharris@nrao.edu>
@@ -42,7 +42,7 @@
42
42
  }
43
43
  })
44
44
  ({
45
- "07c9fde72b": function _(e,a,t,o,c){o();const i=e("tslib"),n=e("f8eca7a5dd");c("DataPipe",n.DataPipe);c("activeDataPipes",e("5179d11e71").activeDataPipes);const p=e("2889e0dd45");c("ImagePipe",p.ImagePipe);const s=e("de65005924");c("ImageDataSource",s.ImageDataSource);const r=e("02e3c3e46c");c("SpectraDataSource",r.SpectraDataSource);const d=e("64b16deff9");c("UpdatableDataSource",d.UpdatableDataSource);const S=e("b6ae454f0d");c("WcsTicks",S.WcsTicks);const l=e("cb7d28d6b3");c("DragTool",l.DragTool);const D=e("9d3c34ff8e");c("CBResetTool",D.CBResetTool);const T=e("e3901fa9f2");c("serialize",T.serialize),c("deserialize",T.deserialize);const b=e("9f961622ce");c("TipButton",b.TipButton);const u=e("ca4c845905");c("Tip",u.Tip);const f=e("a6e757a69a");c("Showable",f.Showable);const P=e("467d2716b0");c("BokehAppContext",P.BokehAppContext);const g=e("50a1e32f01");c("SharedDict",g.SharedDict);const h=e("b55081402e");c("EditSpan",h.EditSpan);const B=e("9144bfc7a5");c("EvTextInput",B.EvTextInput);const E=e("74e0abef8a");c("EvPolyAnnotation",E.EvPolyAnnotation);const I=i.__importStar(e("484bb85d20"));t.find=I;(0,e("@bokehjs/base").register_models)({DataPipe:n.DataPipe,ImagePipe:p.ImagePipe,ImageDataSource:s.ImageDataSource,SpectraDataSource:r.SpectraDataSource,UpdatableDataSource:d.UpdatableDataSource,WcsTicks:S.WcsTicks,DragTool:l.DragTool,CBResetTool:D.CBResetTool,Tip:u.Tip,TipButton:b.TipButton,SharedDict:g.SharedDict,Showable:f.Showable,BokehAppContext:P.BokehAppContext,EditSpan:h.EditSpan,EvTextInput:B.EvTextInput,EvPolyAnnotation:E.EvPolyAnnotation})},
45
+ "07c9fde72b": function _(e,t,a,o,c){o();const i=e("tslib"),n=e("f8eca7a5dd");c("DataPipe",n.DataPipe);c("activeDataPipes",e("5179d11e71").activeDataPipes);const p=e("2889e0dd45");c("ImagePipe",p.ImagePipe);const s=e("de65005924");c("ImageDataSource",s.ImageDataSource);const r=e("02e3c3e46c");c("SpectraDataSource",r.SpectraDataSource);const d=e("64b16deff9");c("UpdatableDataSource",d.UpdatableDataSource);const S=e("b6ae454f0d");c("WcsTicks",S.WcsTicks);const l=e("cb7d28d6b3");c("DragTool",l.DragTool);const D=e("9d3c34ff8e");c("CBResetTool",D.CBResetTool);const T=e("e3901fa9f2");c("serialize",T.serialize),c("deserialize",T.deserialize);const b=e("9f961622ce");c("TipButton",b.TipButton);const u=e("ca4c845905");c("Tip",u.Tip);const f=e("92e97078e3");c("Showable",f.Showable);const P=e("467d2716b0");c("BokehAppContext",P.BokehAppContext);const g=e("50a1e32f01");c("SharedDict",g.SharedDict);const h=e("b55081402e");c("EditSpan",h.EditSpan);const B=e("9144bfc7a5");c("EvTextInput",B.EvTextInput);const E=e("74e0abef8a");c("EvPolyAnnotation",E.EvPolyAnnotation);const I=i.__importStar(e("484bb85d20"));a.find=I;(0,e("@bokehjs/base").register_models)({DataPipe:n.DataPipe,ImagePipe:p.ImagePipe,ImageDataSource:s.ImageDataSource,SpectraDataSource:r.SpectraDataSource,UpdatableDataSource:d.UpdatableDataSource,WcsTicks:S.WcsTicks,DragTool:l.DragTool,CBResetTool:D.CBResetTool,Tip:u.Tip,TipButton:b.TipButton,SharedDict:g.SharedDict,Showable:f.Showable,BokehAppContext:P.BokehAppContext,EditSpan:h.EditSpan,EvTextInput:B.EvTextInput,EvPolyAnnotation:E.EvPolyAnnotation})},
46
46
  "f8eca7a5dd": function _(e,s,t,i,n){var o;i();const a=e("@bokehjs/models/sources/data_source"),c=e("e3901fa9f2"),r=e("@bokehjs/core/util/callbacks"),l=e("5179d11e71");class d extends a.DataSource{constructor(e){super(e),this.send_queue={},this.connection_queue=[],this.pending={},this.incoming_callbacks={},this.session_id=casalib.object_id(this)}checkSessionConflict(){try{if("undefined"==typeof Storage)return console.warn("localStorage not available, skipping session conflict detection"),!0;const e=localStorage.getItem(this.session_storage_key);if(e){const s=JSON.parse(e);if(s.sessionId!==this.session_id&&Date.now()-s.timestamp<12e4){if(this.conflict_check){const e=`CubeVis DataPipe (${this.instance_key}) is already running in another browser window or tab.\n\nPlease close other instances and refresh this page, or\nclose this window to continue using the other instance.`;return alert(e),window.opener||1===window.history.length?window.close():window.location.href="about:blank",!1}console.group(`DataPipe ${this.instance_key} conflict detected in Jupyter context`),console.log("Current session ID:",this.session_id),console.log("Existing session ID:",s.sessionId),console.log("Existing timestamp:",new Date(s.timestamp).toISOString()),console.log("Age of existing session (ms):",Date.now()-s.timestamp),console.log("Address:",this.address),console.log("Instance key:",this.instance_key),console.log("Storage key:",this.session_storage_key),console.log("Existing data:",s),console.log("All localStorage keys:",Object.keys(localStorage).filter((e=>e.startsWith("cubevis_datapipe_")))),console.groupEnd()}}return this.updateSessionHeartbeat(),!0}catch(e){return console.warn("Session conflict detection failed:",e),!0}}updateSessionHeartbeat(){try{"undefined"!=typeof Storage&&localStorage.setItem(this.session_storage_key,JSON.stringify({sessionId:this.session_id,timestamp:Date.now(),instanceKey:this.instance_key}))}catch(e){console.warn("Session heartbeat update failed:",e)}}startHeartbeat(){this.heartbeat_interval=window.setInterval((()=>{this.updateSessionHeartbeat()}),3e4)}stopHeartbeat(){this.heartbeat_interval&&(clearInterval(this.heartbeat_interval),this.heartbeat_interval=void 0)}cleanupSession(){try{if("undefined"!=typeof Storage){const e=localStorage.getItem(this.session_storage_key);if(e){JSON.parse(e).sessionId===this.session_id&&localStorage.removeItem(this.session_storage_key)}}}catch(e){console.warn("Session cleanup failed:",e)}this.stopHeartbeat()}handleSessionConflictMessage(e){console.error("Session conflict detected by server:",e);let s="Session conflict detected by server.";"session_conflict"===e.type?s=e.error||s:"session_corruption"===e.type&&(s=`Session corruption detected.\nExpected: ${e.expected}\nReceived: ${e.received}`),alert(s+"\n\nThis window will be closed to prevent data corruption."),this.cleanupSession();const t=new CustomEvent("cubevis_session_conflict",{detail:{message:e,sessionId:this.session_id}});window.dispatchEvent(t),setTimeout((()=>{window.opener||1===window.history.length?window.close():window.location.href="about:blank"}),2e3)}generateInstanceKey(){const e=`${this.address[0]}_${this.address[1]}`;return`${this.instance_id}_${e}`}initialize(){if(super.initialize(),l.activeDataPipes.register(this),this.instance_key=this.generateInstanceKey(),this.session_storage_key=`cubevis_datapipe_${this.instance_key}`,!this.checkSessionConflict())return;let e=`ws://${this.address[0]}:${this.address[1]}`;console.log("datapipe url:",e);var s=void 0;document.shutdown_in_progress_=!1;var t=()=>{void 0!==this.websocket&&this.websocket.close(),this.websocket=new WebSocket(e),this.websocket.binaryType="arraybuffer",this.websocket.addEventListener("error",(e=>{console.log("error encountered:",e)})),this.websocket.onmessage=e=>{if("string"==typeof e.data||e.data instanceof String){let s=(0,c.deserialize)(e.data);if("id"in s&&"direction"in s&&"message"in s){let{id:e,message:t,direction:i}=s;if("error"===i&&("session_conflict"===e||e===this.session_id)&&t&&("session_conflict"===t.type||"session_corruption"===t.type||"close_duplicate"===t.action))return void this.handleSessionConflictMessage(t);if(void 0===t&&console.log("Error, event failure",s),"j2p"==i)if(e in this.pending){let{cb:i}=this.pending[e];if(delete this.pending[e],e in this.send_queue&&this.send_queue[e].length>0){let{cb:s,msg:t}=this.send_queue[e].shift();this.pending[e]={cb:s},this.websocket.send((0,c.serialize)(t))}void 0===t?console.log("DROPPING ERROR FOR NOW (maybe need error callbacks)",s):i(t)}else console.log("message received but could not find id");else if(e in this.incoming_callbacks){let s=this.incoming_callbacks[e](t);this.websocket.send((0,c.serialize)({id:e,direction:i,message:s,session:this.session_id}))}}else console.log(`datapipe received message without one of 'id', 'message' or 'direction': ${s}`)}else console.log("datapipe received binary data",e.data.byteLength,"bytes")},this.websocket.onopen=()=>{for(s?0==s.connected&&console.log(`connection reestablished at ${new Date}`):(this.websocket.send((0,c.serialize)({id:"initialize",direction:"j2p",session:this.session_id})),this.startHeartbeat()),s=new casalib.ReconnectState;this.connection_queue.length>0;){let e=this.connection_queue.shift();this.send.apply(e[0],e[1])}},this.websocket.onclose=()=>{if(s&&1==s.connected&&(console.log(`connection lost at ${new Date}`),s.connected=!1,!document.shutdown_in_progress_)){console.log(`connection lost at ${new Date}`);var e=s;function i(n){0==s.connected&&(console.log(`${n+1}\treconnection attempt ${new Date}`),t(),e.backoff(),e.retries>0?setTimeout(i,e.timeout,n+1):0==s.connected&&console.log(`aborting reconnection after ${n} attempts ${new Date}`))}i(0)}}};window.addEventListener("beforeunload",(()=>{this.cleanupSession()})),document.addEventListener("visibilitychange",(()=>{"hidden"===document.visibilityState?this.stopHeartbeat():"visible"===document.visibilityState&&(this.updateSessionHeartbeat(),this.startHeartbeat())})),t();(()=>{null!=this.init_script&&(0,r.execute)(this.init_script,this)})()}destroy(){l.activeDataPipes.unregister(this),super.destroy()}register(e,s){this.incoming_callbacks[e]=s}send(e,s,t,i=!1){let n={id:e,message:s,direction:"j2p",session:this.session_id};if(!this.websocket||e in this.pending)if(e in this.send_queue)if("boolean"==typeof i&&i&&this.send_queue[e].length>0)this.send_queue[e][0].msg=n,this.send_queue[e][0].cb=t;else if("function"==typeof i&&this.send_queue[e].length>0){let o=!1;for(const a of this.send_queue[e])i(a.msg.message)&&(a.msg=n,a.cb=t,o=!0);o||this.send_queue[e].push({cb:t,msg:n})}else this.send_queue[e].push({cb:t,msg:n});else this.send_queue[e]=[{cb:t,msg:n}];else if(this.websocket.readyState===WebSocket.CONNECTING)this.connection_queue.push([this,[e,s,t]]);else if(e in this.send_queue&&this.send_queue[e].length>0){this.send_queue[e].push({cb:t,msg:n});{let{cb:r,msg:l}=this.send_queue[e].shift();if(this.pending[e]={cb:r},this.websocket.readyState===WebSocket.OPEN)this.websocket.send((0,c.serialize)(l));else{let d=20,h=this;function u(){h.websocket.readyState===WebSocket.OPEN?h.websocket.send((0,c.serialize)(l)):(d-=1,d>0&&setTimeout(u,3e3))}setTimeout(u,3e3)}}}else if(this.websocket.readyState===WebSocket.OPEN)this.pending[e]={cb:t},this.websocket.send((0,c.serialize)(n));else{let g=20,b=this;function _(){b.websocket.readyState===WebSocket.OPEN?(b.pending[e]={cb:t},b.websocket.send((0,c.serialize)(n))):(g-=1,g>0&&setTimeout(_,3e3))}setTimeout(_,3e3)}}}t.DataPipe=d,o=d,d.__name__="DataPipe",d.__module__="cubevis.bokeh.sources._data_pipe",o.define((({Any:e,Tuple:s,String:t,Number:i,Bool:n})=>({init_script:[e,null],address:[s(t,i)],instance_id:[t],conflict_check:[n,!0]})))},
47
47
  "e3901fa9f2": function _(e,r,s,i,o){i();const l=e("@bokehjs/base"),a=e("@bokehjs/core/resolvers"),t=e("@bokehjs/core/serialization/deserializer"),n=e("@bokehjs/core/serialization/serializer"),{deserialize:c}=new class{constructor(){this.resolver=new a.ModelResolver(l.default_resolver),this.deserializer=new t.Deserializer(this.resolver),this.deserialize=e=>{try{return this.deserializer.decode(JSON.parse(e))}catch(r){return console.group("deserialize error"),console.log(e),console.log(r),console.groupEnd(),{}}}}};s.deserialize=c;const{serialize:z}=new class{constructor(){this.serializer=new n.Serializer,this.serialize=e=>JSON.stringify(this.serializer.encode(e))}};s.serialize=z},
48
48
  "5179d11e71": function _(a,e,n,c,t){c();const i=a("30b45c52a1");n.activeDataPipes=new i.ModelManager},
@@ -58,10 +58,10 @@
58
58
  "9d3c34ff8e": function _(e,o,l,s,t){var c;s();const _=e("@bokehjs/models/tools/actions/reset_tool"),i=e("@bokehjs/styles/icons.css"),a=e("@bokehjs/core/util/callbacks");class n extends _.ResetToolView{doit(){const{precallback:e,postcallback:o}=this.model;null!=e&&(0,a.execute)(e,this.model),this.plot_view.reset(),null!=o&&(0,a.execute)(o,this.model)}}l.CBResetToolView=n,n.__name__="CBResetToolView";class r extends _.ResetTool{constructor(e){super(e),this.tool_name="CBReset",this.tool_icon=i.tool_icon_reset}}l.CBResetTool=r,c=r,r.__name__="CBResetTool",r.__module__="cubevis.bokeh.tools._cbreset_tool",c.prototype.default_view=n,c.define((({Any:e,Nullable:o})=>({precallback:[o(e),null],postcallback:[o(e),null]}))),c.register_alias("cbreset",(()=>new c))},
59
59
  "9f961622ce": function _(e,t,i,o,s){var n;o();const l=e("@bokehjs/models/widgets/abstract_button"),c=e("@bokehjs/models/ui/tooltip"),d=e("@bokehjs/models/ui/icons/builtin_icon"),u=e("@bokehjs/core/build_views"),h=e("@bokehjs/core/bokeh_events"),r=e("@bokehjs/core/util/object");class a extends l.AbstractButtonView{constructor(){super(...arguments),this.isMouseInside=!1}click(){this.model.trigger_event(new h.ButtonClick),super.click()}*children(){yield*super.children(),yield this.tooltip}Show(){this.tooltip.model.setv({visible:!0,closable:!1})}unShow(){this.tooltip.model.setv({visible:!1,closable:!1})}async lazy_initialize(){const{hover_wait:e}=this.model;this.debouncedShow=casalib.debounce((()=>{this.isMouseInside&&this.Show()}),1e3*e),await super.lazy_initialize();const{tooltip:t}=this.model;this.tooltip=await(0,u.build_view)(t,{parent:this})}remove(){this.tooltip.remove(),super.remove()}render(){super.render(),this.el.addEventListener("mouseenter",(()=>{this.isMouseInside=!0,this.debouncedShow()})),this.el.addEventListener("mouseleave",(()=>{this.isMouseInside=!1,this.debouncedShow.cancel(),this.unShow()})),document.addEventListener("mousedown",(e=>{if(void 0===(0,r.dict)(this.model.js_event_callbacks).get(h.ButtonClick.prototype.event_name)){if(e.composedPath().includes(this.tooltip.el))return;this.debouncedShow.cancel(),this.unShow()}else this.isMouseInside=!1,this.debouncedShow.cancel(),this.unShow()})),window.addEventListener("blur",(()=>{this.debouncedShow.cancel(),this.unShow()}))}}i.TipButtonView=a,a.__name__="TipButtonView";class b extends l.AbstractButton{constructor(e){super(e)}on_click(e){this.on_event(h.ButtonClick,e)}}i.TipButton=b,n=b,b.__name__="TipButton",b.__module__="cubevis.bokeh.models._tip_button",n.prototype.default_view=a,n.define((({Ref:e,Number:t})=>({tooltip:[e(c.Tooltip)],hover_wait:[t,1.5]}))),n.override({label:"",icon:new d.BuiltinIcon({icon_name:"help",size:18}),button_type:"default"})},
60
60
  "ca4c845905": function _(e,i,s,t,o){var l;t();const d=e("@bokehjs/models/layouts/layout_dom"),n=e("@bokehjs/models/ui/ui_element"),h=e("@bokehjs/models/ui/tooltip"),r=e("@bokehjs/core/dom"),a=e("@bokehjs/core/build_views");class c extends d.LayoutDOMView{constructor(){super(...arguments),this.isMouseInside=!1}stylesheets(){return[...super.stylesheets(),"*, *:before, *:after { box-sizing: border-box; } fieldset { border: 0px; margin: 0px; padding: 0px; }"]}Show(){this.tooltip.model.setv({visible:!0,closable:!1})}unShow(){this.tooltip.model.setv({visible:!1,closable:!1})}async lazy_initialize(){const{hover_wait:e}=this.model;this.debouncedShow=casalib.debounce((()=>{this.isMouseInside&&this.Show()}),1e3*e),await super.lazy_initialize(),await this.build_child_views();const{tooltip:i}=this.model;this.tooltip=await(0,a.build_view)(i,{parent:this})}remove(){this.tooltip.remove(),super.remove()}connect_signals(){super.connect_signals();const{child:e}=this.model.properties;this.on_change(e,(()=>this.update_children())),this.el.addEventListener("mouseenter",(e=>{this.mouseenter.emit(e)})),this.el.addEventListener("mouseleave",(e=>{this.mouseleave.emit(e)}))}*children(){yield*super.children(),yield this.tooltip}get child_models(){return[this.model.child]}render(){super.render(),this.el.addEventListener("mouseenter",(()=>{this.isMouseInside=!0,this.debouncedShow()})),this.el.addEventListener("mouseleave",(()=>{this.isMouseInside=!1,this.debouncedShow.cancel(),this.unShow()})),window.addEventListener("blur",(()=>{this.debouncedShow.cancel(),this.unShow()})),this.el.addEventListener("click",(()=>{this.debouncedShow.cancel(),this.unShow()}));const e=this.child_views.map((e=>e.el));this.fieldset_el=(0,r.fieldset)({},...e),this.shadow_el.appendChild(this.fieldset_el)}_update_children(){const e=this.child_views.map((e=>e.el));this.fieldset_el.append(...e)}}s.TipView=c,c.__name__="TipView";class u extends d.LayoutDOM{constructor(e){super(e)}}s.Tip=u,l=u,u.__name__="Tip",u.__module__="cubevis.bokeh.models._tip",l.prototype.default_view=c,l.define((({Ref:e,Number:i})=>({child:[e(n.UIElement)],tooltip:[e(h.Tooltip)],hover_wait:[i,1.5]})))},
61
- "a6e757a69a": function _(e,t,l,o,s){var i;o();const a=e("@bokehjs/models/layouts/layout_dom"),r=e("@bokehjs/models/ui/ui_element");class n extends a.LayoutDOMView{constructor(){super(...arguments),this._overlay_el=null}get child_models(){return null!=this.model.ui?[this.model.ui]:[]}initialize(){super.initialize(),this.el.setAttribute("data-lm-suppress-shortcuts","true")}async lazy_initialize(){await super.lazy_initialize()}connect_signals(){super.connect_signals(),this.connect(this.model.properties.disabled.change,(()=>{this._update_disabled_state()})),this.connect(this.model.properties.disabled_message.change,(()=>{this.model.disabled&&null!=this._overlay_el&&this._update_overlay_message()}))}_update_layout(){super._update_layout()}render(){super.render(),console.log("Showable render() - disabled:",this.model.disabled,"shadow_el:",null!=this.shadow_el),0===this.child_views.length&&null==this.model.ui&&(this.el.innerHTML='<div style="color: gray; padding: 10px; border: 1px dashed gray;">\n Showable: No UI element set\n </div>'),this._update_disabled_state()}after_layout(){super.after_layout(),console.log("Showable after_layout() - disabled:",this.model.disabled,"shadow_el:",null!=this.shadow_el),this.model.disabled&&this._update_disabled_state()}_intrinsic_display(){return super._intrinsic_display()}_update_disabled_state(){console.log("_update_disabled_state called, disabled:",this.model.disabled),this.model.disabled?(this._show_disabled_overlay(),this.el.style.filter="grayscale(50%)"):(this._hide_disabled_overlay(),this.el.style.filter="")}_disable_interactive_elements(){this.el.querySelectorAll(".bk-toolbar, .bk-toolbar-button").forEach((e=>{e.style.pointerEvents="none",e.style.opacity="0.5"}));this.el.querySelectorAll("canvas").forEach((e=>{e.style.pointerEvents="none"}));this.el.querySelectorAll("button, .bk-btn").forEach((e=>{e.disabled=!0}))}_enable_interactive_elements(){const e=this.shadow_el;if(!e)return;e.querySelectorAll(".bk-toolbar, .bk-toolbar-button").forEach((e=>{e.style.pointerEvents="",e.style.opacity=""}));e.querySelectorAll("canvas").forEach((e=>{e.style.pointerEvents=""}));e.querySelectorAll("button, .bk-btn").forEach((e=>{e.disabled=!1})),console.log("Interactive elements re-enabled")}_show_disabled_overlay(){if(console.log("_show_disabled_overlay called, _overlay_el exists:",null!=this._overlay_el),console.log("disabled_message:",this.model.disabled_message),null==this._overlay_el){console.log("Creating new overlay element");const e=this.shadow_el;if(console.log("Shadow root:",e),!e)return void console.error("No shadow root found!");const t=document.createElement("div");t.className="showable-disabled-message",t.style.cssText="\n background: white;\n padding: 30px 40px;\n border-radius: 10px;\n box-shadow: 0 4px 20px rgba(0, 0, 0, 0.3);\n text-align: center;\n border: 2px solid #4CAF50;\n max-width: 90%;\n word-wrap: break-word;\n pointer-events: auto;\n cursor: default;\n ",t.innerHTML=`\n <div style="font-size: 24px; font-weight: bold; color: #4CAF50; margin-bottom: 10px;">\n ${this.model.disabled_message}\n </div>\n `,console.log("Message box created"),this._overlay_el=document.createElement("div"),this._overlay_el.appendChild(t),this._overlay_el.style.setProperty("position","absolute","important"),this._overlay_el.style.setProperty("top","0","important"),this._overlay_el.style.setProperty("left","0","important"),this._overlay_el.style.setProperty("right","0","important"),this._overlay_el.style.setProperty("bottom","0","important"),this._overlay_el.style.setProperty("width","100%","important"),this._overlay_el.style.setProperty("height","100%","important"),this._overlay_el.style.setProperty("background-color","rgba(220, 220, 220, 0.85)","important"),this._overlay_el.style.setProperty("display","flex","important"),this._overlay_el.style.setProperty("justify-content","center","important"),this._overlay_el.style.setProperty("align-items","center","important"),this._overlay_el.style.setProperty("z-index","2147483647","important"),this._overlay_el.style.setProperty("pointer-events","auto","important"),this._overlay_el.style.setProperty("cursor","not-allowed","important"),console.log("Overlay element styles set");const l=e.firstElementChild;l&&(l.style.position="relative",console.log("Set content container to relative positioning")),e.appendChild(this._overlay_el),console.log("Overlay appended to shadow root"),setTimeout((()=>{const e=this._overlay_el.getBoundingClientRect(),l=t.getBoundingClientRect();console.log("Overlay bounding rect:",{width:e.width,height:e.height,top:e.top,left:e.left}),console.log("Message box bounding rect:",{width:l.width,height:l.height})}),100),this._disable_interactive_elements();["mousedown","mouseup","click","dblclick","contextmenu","touchstart","touchmove","touchend","keydown","keyup","keypress","pointerdown","pointerup","pointermove","dragstart","drag","dragend"].forEach((e=>{this._overlay_el.addEventListener(e,(e=>{if(e.target!==t&&!t.contains(e.target))return e.stopPropagation(),e.preventDefault(),e.stopImmediatePropagation(),!1}),{capture:!0,passive:!1})})),this._overlay_el.addEventListener("wheel",(t=>{let l=e.host.parentElement;for(;l;){const e=window.getComputedStyle(l).overflow,t=window.getComputedStyle(l).overflowY;if("auto"===e||"scroll"===e||"auto"===t||"scroll"===t)break;l=l.parentElement}l&&(l.scrollTop+=t.deltaY,l.scrollLeft+=t.deltaX),t.stopPropagation(),t.preventDefault()}),{capture:!0,passive:!1})}else console.log("Overlay already exists, showing it"),this._overlay_el.style.display="flex";console.log("_show_disabled_overlay complete")}_update_overlay_message(){if(null!=this._overlay_el){const e=this._overlay_el.querySelector(".showable-disabled-message");e&&(e.innerHTML=`\n <div style="font-size: 24px; font-weight: bold; color: #4CAF50; margin-bottom: 10px;">\n ${this.model.disabled_message}\n </div>\n <div style="font-size: 14px; color: #666;">\n You can now close this GUI or continue working in your notebook\n </div>\n `)}}_hide_disabled_overlay(){null!=this._overlay_el&&(this._overlay_el.style.display="none",this._enable_interactive_elements())}remove(){null!=this._overlay_el&&(this._overlay_el.remove(),this._overlay_el=null),super.remove()}}l.ShowableView=n,n.__name__="ShowableView";class d extends a.LayoutDOM{constructor(e){super(e)}}l.Showable=d,i=d,d.__name__="Showable",d.__module__="cubevis.bokeh.models._showable",i.prototype.default_view=n,i.define((({Ref:e,String:t})=>({ui:[e(r.UIElement)],disabled_message:[t,"Interaction Complete \u2713"]})))},
61
+ "92e97078e3": function _(e,t,o,l,s){var i;l();const a=e("@bokehjs/models/layouts/layout_dom"),n=e("@bokehjs/models/ui/ui_element");class r extends a.LayoutDOMView{constructor(){super(...arguments),this._overlay_el=null}get child_models(){return null!=this.model.ui?[this.model.ui]:[]}initialize(){super.initialize(),this.el.setAttribute("data-lm-suppress-shortcuts","true")}async lazy_initialize(){await super.lazy_initialize()}connect_signals(){super.connect_signals(),this.connect(this.model.properties.disabled.change,(()=>{this._update_disabled_state()})),this.connect(this.model.properties.disabled_message.change,(()=>{this.model.disabled&&null!=this._overlay_el&&this._update_overlay_message()}))}_update_layout(){super._update_layout()}render(){super.render(),console.log("Showable render() - disabled:",this.model.disabled,"shadow_el:",null!=this.shadow_el);try{console.log("Colab environment check:"),console.log("window.location:",window.location),console.log("window.google:",window.google),console.log("Parent frame:",window.parent!==window),window.parent!==window&&(console.log("Running in iframe"),console.log("Parent location:",document.referrer))}catch(e){console.warn("An error occurred during Colab diagnostics")}console.log("Past Colab diagnostics..."),0===this.child_views.length&&null==this.model.ui&&(this.el.innerHTML='<div style="color: gray; padding: 10px; border: 1px dashed gray;">\n Showable: No UI element set\n </div>'),this._update_disabled_state()}after_layout(){super.after_layout(),console.log("Showable after_layout() - disabled:",this.model.disabled,"shadow_el:",null!=this.shadow_el),this.model.disabled&&this._update_disabled_state()}_intrinsic_display(){return super._intrinsic_display()}_update_disabled_state(){console.log("_update_disabled_state called, disabled:",this.model.disabled),this.model.disabled?(this._show_disabled_overlay(),this.el.style.filter="grayscale(50%)"):(this._hide_disabled_overlay(),this.el.style.filter="")}_disable_interactive_elements(){this.el.querySelectorAll(".bk-toolbar, .bk-toolbar-button").forEach((e=>{e.style.pointerEvents="none",e.style.opacity="0.5"}));this.el.querySelectorAll("canvas").forEach((e=>{e.style.pointerEvents="none"}));this.el.querySelectorAll("button, .bk-btn").forEach((e=>{e.disabled=!0}))}_enable_interactive_elements(){const e=this.shadow_el;if(!e)return;e.querySelectorAll(".bk-toolbar, .bk-toolbar-button").forEach((e=>{e.style.pointerEvents="",e.style.opacity=""}));e.querySelectorAll("canvas").forEach((e=>{e.style.pointerEvents=""}));e.querySelectorAll("button, .bk-btn").forEach((e=>{e.disabled=!1})),console.log("Interactive elements re-enabled")}_show_disabled_overlay(){if(console.log("_show_disabled_overlay called, _overlay_el exists:",null!=this._overlay_el),console.log("disabled_message:",this.model.disabled_message),null==this._overlay_el){console.log("Creating new overlay element");const e=this.shadow_el;if(console.log("Shadow root:",e),!e)return void console.error("No shadow root found!");const t=document.createElement("div");t.className="showable-disabled-message",t.style.cssText="\n background: white;\n padding: 30px 40px;\n border-radius: 10px;\n box-shadow: 0 4px 20px rgba(0, 0, 0, 0.3);\n text-align: center;\n border: 2px solid #4CAF50;\n max-width: 90%;\n word-wrap: break-word;\n pointer-events: auto;\n cursor: default;\n ",t.innerHTML=`\n <div style="font-size: 24px; font-weight: bold; color: #4CAF50; margin-bottom: 10px;">\n ${this.model.disabled_message}\n </div>\n `,console.log("Message box created"),this._overlay_el=document.createElement("div"),this._overlay_el.appendChild(t),this._overlay_el.style.setProperty("position","absolute","important"),this._overlay_el.style.setProperty("top","0","important"),this._overlay_el.style.setProperty("left","0","important"),this._overlay_el.style.setProperty("right","0","important"),this._overlay_el.style.setProperty("bottom","0","important"),this._overlay_el.style.setProperty("width","100%","important"),this._overlay_el.style.setProperty("height","100%","important"),this._overlay_el.style.setProperty("background-color","rgba(220, 220, 220, 0.85)","important"),this._overlay_el.style.setProperty("display","flex","important"),this._overlay_el.style.setProperty("justify-content","center","important"),this._overlay_el.style.setProperty("align-items","center","important"),this._overlay_el.style.setProperty("z-index","2147483647","important"),this._overlay_el.style.setProperty("pointer-events","auto","important"),this._overlay_el.style.setProperty("cursor","not-allowed","important"),console.log("Overlay element styles set");const o=e.firstElementChild;o&&(o.style.position="relative",console.log("Set content container to relative positioning")),e.appendChild(this._overlay_el),console.log("Overlay appended to shadow root"),setTimeout((()=>{const e=this._overlay_el.getBoundingClientRect(),o=t.getBoundingClientRect();console.log("Overlay bounding rect:",{width:e.width,height:e.height,top:e.top,left:e.left}),console.log("Message box bounding rect:",{width:o.width,height:o.height})}),100),this._disable_interactive_elements();["mousedown","mouseup","click","dblclick","contextmenu","touchstart","touchmove","touchend","keydown","keyup","keypress","pointerdown","pointerup","pointermove","dragstart","drag","dragend"].forEach((e=>{this._overlay_el.addEventListener(e,(e=>{if(e.target!==t&&!t.contains(e.target))return e.stopPropagation(),e.preventDefault(),e.stopImmediatePropagation(),!1}),{capture:!0,passive:!1})})),this._overlay_el.addEventListener("wheel",(t=>{let o=e.host.parentElement;for(;o;){const e=window.getComputedStyle(o).overflow,t=window.getComputedStyle(o).overflowY;if("auto"===e||"scroll"===e||"auto"===t||"scroll"===t)break;o=o.parentElement}o&&(o.scrollTop+=t.deltaY,o.scrollLeft+=t.deltaX),t.stopPropagation(),t.preventDefault()}),{capture:!0,passive:!1})}else console.log("Overlay already exists, showing it"),this._overlay_el.style.display="flex";console.log("_show_disabled_overlay complete")}_update_overlay_message(){if(null!=this._overlay_el){const e=this._overlay_el.querySelector(".showable-disabled-message");e&&(e.innerHTML=`\n <div style="font-size: 24px; font-weight: bold; color: #4CAF50; margin-bottom: 10px;">\n ${this.model.disabled_message}\n </div>\n <div style="font-size: 14px; color: #666;">\n You can now close this GUI or continue working in your notebook\n </div>\n `)}}_hide_disabled_overlay(){null!=this._overlay_el&&(this._overlay_el.style.display="none",this._enable_interactive_elements())}remove(){null!=this._overlay_el&&(this._overlay_el.remove(),this._overlay_el=null),super.remove()}}o.ShowableView=r,r.__name__="ShowableView";class d extends a.LayoutDOM{constructor(e){super(e)}}o.Showable=d,i=d,d.__name__="Showable",d.__module__="cubevis.bokeh.models._showable",i.prototype.default_view=r,i.define((({Ref:e,String:t})=>({ui:[e(n.UIElement)],disabled_message:[t,"Interaction Complete \u2713"]})))},
62
62
  "467d2716b0": function _(e,i,s,t,o){var n;t();const p=e("@bokehjs/models/layouts/layout_dom"),a=e("@bokehjs/models/ui/ui_element");class l extends p.LayoutDOMView{get child_models(){return null!=this.model.ui?[this.model.ui]:[]}initialize(){super.initialize(),this.el&&(this.el.style.padding="0",this.el.style.margin="0",this.el.style.border="none",this.el.style.background="transparent",this.el.style.display="contents"),this.connect(this.model.properties.app_state.change,(()=>{const e=window.cubevisAppSession;(null==e?void 0:e.applications[this.model.app_id])&&(e.applications[this.model.app_id].state=this.model.app_state)}))}}s.BokehAppContextView=l,l.__name__="BokehAppContextView";class d extends p.LayoutDOM{constructor(e){super(e)}initialize(){super.initialize(),window.cubevisAppSession||(window.cubevisAppSession={sessionId:this.session_id,applications:{}},console.log(`Initialized Bokeh session: ${this.session_id}`));const e=window.cubevisAppSession;e.applications[this.app_id]||(e.applications[this.app_id]={appId:this.app_id,state:this.app_state,createdAt:(new Date).toISOString()},console.log(`Registered application: ${this.app_id}`))}}s.BokehAppContext=d,n=d,d.__name__="BokehAppContext",d.__module__="cubevis.bokeh.models._bokeh_app_context",n.prototype.default_view=l,n.define((({Ref:e,Nullable:i,Dict:s,String:t,Unknown:o})=>({ui:[i(e(a.UIElement)),null],app_id:[t,""],session_id:[t,""],app_state:[s(o),{}]})))},
63
63
  "50a1e32f01": function _(e,s,i,n,t){var a;n();const c=e("@bokehjs/core/view"),o=e("@bokehjs/model");class _ extends c.View{initialize(){super.initialize()}connect_signals(){super.connect_signals();const{values:e}=this.model.properties;this.on_change(e,(()=>this.values_changed()))}values_changed(){}}i.SharedDictView=_,_.__name__="SharedDictView";class d extends o.Model{constructor(e){super(e)}}i.SharedDict=d,a=d,d.__name__="SharedDict",d.__module__="cubevis.bokeh.models._shared_dict",a.prototype.default_view=_,a.define((({Dict:e,Unknown:s})=>({values:[e(s),{}]})))},
64
64
  "b55081402e": function _(e,n,t,_,s){var a;_();const o=e("@bokehjs/models/annotations/span"),p=e("@bokehjs/core/bokeh_events");class r extends o.SpanView{on_pan_start(e){const n=super.on_pan_start(e);return this.model.trigger_event(new p.LODStart),n}on_pan(e){super.on_pan(e)}on_pan_end(e){super.on_pan_end(e),this.model.trigger_event(new p.LODEnd)}}t.EditSpanView=r,r.__name__="EditSpanView";class d extends o.Span{constructor(e){super(e)}}t.EditSpan=d,a=d,d.__name__="EditSpan",d.__module__="cubevis.bokeh.models._edit_span",a.prototype.default_view=r},
65
65
  "9144bfc7a5": function _(e,t,s,n,r){var i;n();const l=e("@bokehjs/models/widgets/text_input"),o=e("@bokehjs/core/dom"),u=e("@bokehjs/core/bokeh_events");class _ extends l.TextInputView{stylesheets(){return[...super.stylesheets(),new o.InlineStyleSheet(".bk-input-prefix { padding: 0 var(--padding-vertical); }")]}connect_signals(){super.connect_signals(),this.el.addEventListener("mouseenter",(e=>{this.model.trigger_event(new u.MouseEnter(e.screenX,e.screenY,e.x,e.y,{shift:e.shiftKey,ctrl:e.ctrlKey,alt:e.altKey}))})),this.el.addEventListener("mouseleave",(e=>{this.model.trigger_event(new u.MouseLeave(e.screenX,e.screenY,e.x,e.y,{shift:e.shiftKey,ctrl:e.ctrlKey,alt:e.altKey}))}))}render(){super.render()}}s.EvTextInputView=_,_.__name__="EvTextInputView";class c extends l.TextInput{constructor(e){super(e)}}s.EvTextInput=c,i=c,c.__name__="EvTextInput",c.__module__="cubevis.bokeh.models._ev_text_input",i.prototype.default_view=_},
66
66
  "74e0abef8a": function _(e,t,n,s,o){var i;s();const r=e("@bokehjs/models/annotations/poly_annotation"),a=e("@bokehjs/core/bokeh_events");class _ extends r.PolyAnnotationView{on_enter(e){const{x_scale:t,y_scale:n}=this.plot_view.frame,s=new a.MouseEnter(e.sx,e.sy,t.invert(e.sx),n.invert(e.sy),{shift:e.modifiers.shift,ctrl:e.modifiers.ctrl,alt:e.modifiers.alt}),o=super.on_enter(e);return this.model.trigger_event(s),o}on_leave(e){const{x_scale:t,y_scale:n}=this.plot_view.frame,s=new a.MouseLeave(e.sx,e.sy,t.invert(e.sx),n.invert(e.sy),{shift:e.modifiers.shift,ctrl:e.modifiers.ctrl,alt:e.modifiers.alt});super.on_leave(e),this.model.trigger_event(s)}on_pan_start(e){const{x_scale:t,y_scale:n}=this.plot_view.frame,s=new a.PanStart(e.sx,e.sy,t.invert(e.sx),n.invert(e.sy),{shift:e.modifiers.shift,ctrl:e.modifiers.ctrl,alt:e.modifiers.alt}),o=super.on_pan_start(e);return this.model.trigger_event(s),o}on_pan_end(e){const{x_scale:t,y_scale:n}=this.plot_view.frame,s=new a.PanEnd(e.sx,e.sy,t.invert(e.sx),n.invert(e.sy),{shift:e.modifiers.shift,ctrl:e.modifiers.ctrl,alt:e.modifiers.alt});super.on_pan_end(e),this.model.trigger_event(s)}on_pan(e){super.on_pan(e);const t=new a.RangesUpdate(e.sx,e.sx+e.dx,e.sy,e.sy+e.dy);this.model.trigger_event(t)}}n.EvPolyAnnotationView=_,_.__name__="EvPolyAnnotationView";class l extends r.PolyAnnotation{constructor(e){super(e)}}n.EvPolyAnnotation=l,i=l,l.__name__="EvPolyAnnotation",l.__module__="cubevis.bokeh.annotations._ev_poly_annotation",i.prototype.default_view=_},
67
- }, "07c9fde72b", {"index":"07c9fde72b","src/bokeh/sources/data_pipe":"f8eca7a5dd","src/bokeh/util/conversions":"e3901fa9f2","src/bokeh/sources/active_data_pipes":"5179d11e71","src/bokeh/util/model_manager":"30b45c52a1","src/bokeh/sources/image_pipe":"2889e0dd45","src/bokeh/sources/image_data_source":"de65005924","src/bokeh/sources/spectra_data_source":"02e3c3e46c","src/bokeh/sources/updatable_data_source":"64b16deff9","src/bokeh/format/wcs_ticks":"b6ae454f0d","src/bokeh/tools/drag_tool":"cb7d28d6b3","src/bokeh/events":"8fc7a9e935","src/bokeh/util/find":"484bb85d20","src/bokeh/tools/cbreset_tool":"9d3c34ff8e","src/bokeh/models/tip_button":"9f961622ce","src/bokeh/models/tip":"ca4c845905","src/bokeh/models/showable":"a6e757a69a","src/bokeh/models/bokeh_app_context":"467d2716b0","src/bokeh/models/shared_dict":"50a1e32f01","src/bokeh/models/edit_span":"b55081402e","src/bokeh/models/ev_text_input":"9144bfc7a5","src/bokeh/annotations/ev_poly_annotation":"74e0abef8a"}, {});});
67
+ }, "07c9fde72b", {"index":"07c9fde72b","src/bokeh/sources/data_pipe":"f8eca7a5dd","src/bokeh/util/conversions":"e3901fa9f2","src/bokeh/sources/active_data_pipes":"5179d11e71","src/bokeh/util/model_manager":"30b45c52a1","src/bokeh/sources/image_pipe":"2889e0dd45","src/bokeh/sources/image_data_source":"de65005924","src/bokeh/sources/spectra_data_source":"02e3c3e46c","src/bokeh/sources/updatable_data_source":"64b16deff9","src/bokeh/format/wcs_ticks":"b6ae454f0d","src/bokeh/tools/drag_tool":"cb7d28d6b3","src/bokeh/events":"8fc7a9e935","src/bokeh/util/find":"484bb85d20","src/bokeh/tools/cbreset_tool":"9d3c34ff8e","src/bokeh/models/tip_button":"9f961622ce","src/bokeh/models/tip":"ca4c845905","src/bokeh/models/showable":"92e97078e3","src/bokeh/models/bokeh_app_context":"467d2716b0","src/bokeh/models/shared_dict":"50a1e32f01","src/bokeh/models/edit_span":"b55081402e","src/bokeh/models/ev_text_input":"9144bfc7a5","src/bokeh/annotations/ev_poly_annotation":"74e0abef8a"}, {});});
@@ -30,6 +30,7 @@ provided by Bokeh'''
30
30
 
31
31
  from .state import order_bokeh_js as _order_bokeh_js
32
32
  from .state import register_model as _register_model
33
+ from .state._initialize import get_bokeh_js_paths
33
34
  from .state import set_cubevis_lib
34
35
 
35
36
  class BokehInit:
@@ -5,6 +5,7 @@ from bokeh.core.properties import Instance, String
5
5
 
6
6
  from bokeh.io import curdoc
7
7
  from .. import BokehInit
8
+ from ...utils import is_colab
8
9
 
9
10
  logger = logging.getLogger(__name__)
10
11
 
@@ -272,8 +273,11 @@ class Showable(LayoutDOM,BokehInit):
272
273
  Common logic for generating HTML in notebook environments.
273
274
  Returns the HTML string to display, or None if not in a notebook.
274
275
  """
275
- from bokeh.embed import components
276
+ from bokeh.embed import components, json_item
276
277
  from bokeh.io.state import curstate
278
+ from bokeh.resources import CDN
279
+ import sys
280
+ import json as json_lib
277
281
 
278
282
  state = curstate()
279
283
 
@@ -284,31 +288,92 @@ class Showable(LayoutDOM,BokehInit):
284
288
  return '<div style="color: red; padding: 10px; border: 1px solid red;">Showable object with no UI set</div>'
285
289
 
286
290
  if self._display_context:
287
- self._display_context.on_show( )
291
+ self._display_context.on_show()
288
292
 
289
293
  if self._notebook_rendering:
290
294
  # Return a lightweight reference instead of re-rendering the full GUI
291
295
  return f'''
292
296
  <div style="padding: 10px; background: #f0f8f0; border-left: 4px solid #4CAF50; margin: 5px 0;">
293
- <strong>↑ iclean GUI active above</strong>
297
+ <strong>→ iclean GUI active above</strong>
294
298
  <small style="color: #666; display: block; margin-top: 5px;">
295
299
  Showable ID: {self.id[-8:]} | Backend: Running
296
300
  </small>
297
301
  </div>
298
302
  '''
299
303
 
300
- # Apply notebook sizing for Jupyter context
301
- ###if self._notebook_sizing == 'fixed':
302
- ### self.sizing_mode = None
303
- ### self.width = self._notebook_width
304
- ### self.height = self._notebook_height
304
+ if is_colab( ):
305
+ # Get all JS paths from the existing function
306
+ # This returns paths in the correct order:
307
+ # [casalib, bokeh-core, bokeh-widgets, bokeh-tables, cubevisjs]
308
+ from cubevis.bokeh import get_bokeh_js_paths
309
+ js_paths = get_bokeh_js_paths( )
310
+
311
+ # Build script tags for all libraries in order
312
+ all_scripts = '\n'.join([
313
+ f'<script type="text/javascript" src="{url}"></script>'
314
+ for url in js_paths
315
+ ])
316
+
317
+ # Use json_item approach which is more reliable in iframes
318
+ item = json_item(self, target=f"bokeh-{self.id}")
319
+ item_json = json_lib.dumps(item)
320
+
321
+ # Build complete HTML with proper loading sequence
322
+ # get_bokeh_js_paths() already returns libs in the correct order:
323
+ # 1. casalib (third-party libs for CustomJS)
324
+ # 2. bokeh-core
325
+ # 3. bokeh-widgets
326
+ # 4. bokeh-tables
327
+ # 5. cubevisjs (custom Bokeh models)
328
+ html = f'''
329
+ {f'<link href="{CDN.css_files[0]}" rel="stylesheet" type="text/css">' if CDN.css_files else ""}
330
+ <div id="bokeh-{self.id}" class="bk-root"></div>
331
+ {all_scripts}
332
+ <script type="text/javascript">
333
+ (function() {{
334
+ var item = {item_json};
335
+
336
+ function embedWhenReady() {{
337
+ // Check if all required libraries are loaded
338
+ if (typeof Bokeh !== 'undefined' && Bokeh.embed) {{
339
+ var target = document.getElementById("bokeh-{self.id}");
340
+ if (target) {{
341
+ try {{
342
+ Bokeh.embed.embed_item(item);
343
+ console.log("Bokeh plot embedded successfully");
344
+ }} catch(e) {{
345
+ console.error("Error embedding Bokeh plot:", e);
346
+ }}
347
+ }} else {{
348
+ console.error("Target element not found");
349
+ setTimeout(embedWhenReady, 50);
350
+ }}
351
+ }} else {{
352
+ setTimeout(embedWhenReady, 50);
353
+ }}
354
+ }}
355
+
356
+ if (document.readyState === 'loading') {{
357
+ document.addEventListener('DOMContentLoaded', embedWhenReady);
358
+ }} else {{
359
+ embedWhenReady();
360
+ }}
361
+ }})();
362
+ </script>
363
+ '''
305
364
 
306
- script, div = components(self)
307
- if start_backend:
308
- self._start_backend()
365
+ if start_backend:
366
+ self._start_backend()
309
367
 
310
- self._notebook_rendering = f'{script}\n{div}'
311
- return self._notebook_rendering
368
+ self._notebook_rendering = html
369
+ return html
370
+ else:
371
+ # In Jupyter Lab/Classic, use components() as before
372
+ script, div = components(self)
373
+ if start_backend:
374
+ self._start_backend()
375
+ self._notebook_rendering = f'{script}\n{div}'
376
+ return self._notebook_rendering
312
377
 
313
378
  def _repr_mimebundle_(self, include=None, exclude=None):
314
379
  """
@@ -321,6 +321,13 @@ def order_bokeh_js():
321
321
  resources.Resources.js_files = property(js_files)
322
322
  return
323
323
 
324
+ #def get_bokeh_js_paths( ):
325
+ # modes = ['cdn','inline','server','server-dev','relative','relative-dev','absolute','absolute-dev']
326
+ # return { 'new': { mode: resources.Resources(mode=mode).js_files for mode in modes },
327
+ # 'old': { mode: resources.Resources(mode=mode)._old_js_files for mode in modes } }
328
+ def get_bokeh_js_paths( ):
329
+ return resources.Resources(mode='cdn').js_files
330
+
324
331
  def get_jupyter_state( ):
325
332
  """Get the package-level Jupyter state"""
326
333
  return _JUPYTER_STATE
@@ -1082,7 +1082,7 @@ class CubeMask:
1082
1082
  ### Firefox incorrectly sizes the colormap histogram plot resulting in the right
1083
1083
  ### side being outside of the browser window... this may have to be tweaked again
1084
1084
  ### when Jupyter notebooks are supported...
1085
- ( [ Spacer( width=120 ) ] if have_firefox( ) else [ ] ), sizing_mode='stretch_both' ),
1085
+ ( [ Spacer( width=120 ) ] if not is_interactive_jupyter( ) and have_firefox( ) else [ ] ), sizing_mode='stretch_both' ),
1086
1086
  row( Tip( self._cm_adjust['min input'],
1087
1087
  tooltip=Tooltip( content=HTML("set minimum clip here or drag the left red line above"),
1088
1088
  position="top" ) ),
@@ -43,7 +43,7 @@ from ._regions import polygon_indexes
43
43
  from ._docenum import DocEnum
44
44
  from ._copydoc import copydoc
45
45
  from ._pkgs import find_pkg, load_pkg
46
- from ._jupyter import is_interactive_jupyter
46
+ from ._jupyter import is_interactive_jupyter, is_colab
47
47
  from ._browser import have_firefox
48
48
  from ._git import max_git_version
49
49
 
@@ -1,3 +1,5 @@
1
+ import os
2
+ import sys
1
3
  import logging
2
4
 
3
5
  logger = logging.getLogger(__name__)
@@ -17,6 +19,15 @@ logger = logging.getLogger(__name__)
17
19
  # except NameError:
18
20
  # return False
19
21
 
22
+ def is_colab():
23
+ """
24
+ Detects if the current environment is Google Colab.
25
+ Combines module check (process-level) and env-var check (VM-level).
26
+ """
27
+ # 1. Check for the Colab-specific environment variable (VM level)
28
+ # 2. Check for the google.colab module (Python process level)
29
+ return os.getenv("COLAB_RELEASE_TAG") is not None or 'google.colab' in sys.modules
30
+
20
31
  def is_interactive_jupyter( ) -> bool:
21
32
  """
22
33
  Detect if running in an interactive Jupyter notebook with frontend connection.
@@ -25,6 +36,7 @@ def is_interactive_jupyter( ) -> bool:
25
36
  - Interactive Jupyter notebook/lab with frontend (returns True)
26
37
  - Standalone Jupyter kernel without frontend (returns False)
27
38
  - Regular Python interpreter (returns False)
39
+ - Support Google Colab and legacy Jupyter versions.
28
40
 
29
41
  Returns:
30
42
  bool: True if running in interactive Jupyter with frontend, False otherwise
@@ -37,12 +49,21 @@ def is_interactive_jupyter( ) -> bool:
37
49
  logger.debug(f"\tis_interactive_jupyter<1>: False")
38
50
  return False
39
51
 
40
- # Check if we're in a ZMQ-based shell (kernel)
41
- if ipython.__class__.__name__ != 'ZMQInteractiveShell':
52
+ # Allow Colab's 'Shell' class in addition to standard 'ZMQInteractiveShell'
53
+ shell_class = ipython.__class__.__name__
54
+ is_colab = 'google.colab' in sys.modules
55
+
56
+ # Check if we're in a ZMQ-based shell (kernel) or Colab's 'Shell'
57
+ if shell_class not in ['ZMQInteractiveShell', 'Shell'] and not is_colab:
42
58
  logger.debug(f"\tis_interactive_jupyter<2>: False")
43
59
  return False
44
60
 
45
- # Check for active frontend connection
61
+ # If we are in Colab, the existence of the shell itself implies
62
+ # an interactive frontend, as Colab sessions are inherently interactive.
63
+ if is_colab:
64
+ return True
65
+
66
+ # Check for active frontend connection (Standard Jupyter/Legacy)
46
67
  if hasattr(ipython, 'kernel') and ipython.kernel is not None:
47
68
  kernel = ipython.kernel
48
69
 
@@ -51,14 +72,15 @@ def is_interactive_jupyter( ) -> bool:
51
72
  # For newer Jupyter versions, check connection count
52
73
  if hasattr(kernel, 'connection_count'):
53
74
  logger.debug(f"\tis_interactive_jupyter<3>: {kernel.connection_count > 0}")
54
- return kernel.connection_count > 0
75
+ if kernel.connection_count > 0:
76
+ return True
55
77
 
56
78
  # For older versions, check if socket is connected
57
79
  try:
58
80
  # Try to get socket state - if it fails, likely no frontend
59
- socket_state = kernel.shell_socket.closed
60
81
  logger.debug(f"\tis_interactive_jupyter<4>: {not socket_state}")
61
- return not socket_state
82
+ if not kernel.shell_socket.closed:
83
+ return True
62
84
  except AttributeError:
63
85
  pass
64
86
 
@@ -68,7 +90,8 @@ def is_interactive_jupyter( ) -> bool:
68
90
  parent = kernel.get_parent()
69
91
  # If there's a parent message, we're likely in interactive mode
70
92
  logger.debug(f"\tis_interactive_jupyter<5>: {parent is not None and len(parent) > 0}")
71
- return parent is not None and len(parent) > 0
93
+ if parent is not None and len(parent) > 0:
94
+ return True
72
95
  except Exception:
73
96
  pass
74
97
 
@@ -77,12 +100,12 @@ def is_interactive_jupyter( ) -> bool:
77
100
  logger.debug(f"\tis_interactive_jupyter<6>: True")
78
101
  return True
79
102
 
80
- # Fallback: Check for common Jupyter notebook environment indicators
81
- # This catches cases where kernel introspection doesn't work
82
- import os
103
+ # Fallback: Check for environment indicators (Legacy/Containerized Jupyter)
83
104
  jupyter_indicators = [
84
- 'JPY_PARENT_PID', # JupyterLab/Notebook sets this
105
+ 'JPY_PARENT_PID', # JupyterLab/Notebook sets this
85
106
  'JUPYTER_RUNTIME_DIR',
107
+ 'COLAB_RELEASE_DEVICE', # Additional Colab-specific env check
108
+ 'COLAB_GPU' # Additional Colab-specific env check
86
109
  ]
87
110
 
88
111
  for indicator in jupyter_indicators:
@@ -99,6 +122,6 @@ def is_interactive_jupyter( ) -> bool:
99
122
  logger.debug(f"\tis_interactive_jupyter<8>: False")
100
123
  return False
101
124
 
102
- except ImportError:
125
+ except (ImportError, Exception):
103
126
  logger.debug(f"\tis_interactive_jupyter<9>: False")
104
127
  return False
@@ -16,7 +16,7 @@ dependencies = [
16
16
  ]
17
17
  requires-python = ">=3.11"
18
18
  readme = "README.rst"
19
- version = "1.0.21"
19
+ version = "1.0.28"
20
20
 
21
21
  [project.license]
22
22
  text = "LGPL"
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes