bec-widgets 0.61.0__tar.gz → 0.62.0__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 (166) hide show
  1. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/CHANGELOG.md +11 -9
  2. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/PKG-INFO +1 -1
  3. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/cli/client_utils.py +62 -34
  4. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/examples/jupyter_console/jupyter_console_window.py +6 -5
  5. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/bec_dispatcher.py +5 -1
  6. bec_widgets-0.62.0/docs/user/customisation.md +123 -0
  7. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/pyproject.toml +1 -1
  8. bec_widgets-0.61.0/docs/user/customisation.md +0 -13
  9. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/.git_hooks/pre-commit +0 -0
  10. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/.gitignore +0 -0
  11. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/.gitlab/issue_templates/bug_report_template.md +0 -0
  12. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/.gitlab/issue_templates/documentation_update_template.md +0 -0
  13. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/.gitlab/issue_templates/feature_request_template.md +0 -0
  14. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/.gitlab/merge_request_templates/default.md +0 -0
  15. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/.gitlab-ci.yml +0 -0
  16. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/.pylintrc +0 -0
  17. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/.readthedocs.yaml +0 -0
  18. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/LICENSE +0 -0
  19. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/README.md +0 -0
  20. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/__init__.py +0 -0
  21. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/assets/bec_widgets_icon.png +0 -0
  22. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/assets/terminal_icon.png +0 -0
  23. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/cli/__init__.py +0 -0
  24. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/cli/auto_updates.py +0 -0
  25. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/cli/client.py +0 -0
  26. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/cli/generate_cli.py +0 -0
  27. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/cli/rpc_register.py +0 -0
  28. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/cli/rpc_wigdet_handler.py +0 -0
  29. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/cli/server.py +0 -0
  30. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/examples/__init__.py +0 -0
  31. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/examples/jupyter_console/__init__.py +0 -0
  32. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/examples/jupyter_console/jupyter_console_window.ui +0 -0
  33. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/examples/motor_movement/__init__.py +0 -0
  34. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/examples/motor_movement/motor_control_compilations.py +0 -0
  35. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/examples/motor_movement/motor_controller.ui +0 -0
  36. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/__init__.py +0 -0
  37. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/bec_connector.py +0 -0
  38. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/bec_table.py +0 -0
  39. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/colors.py +0 -0
  40. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/container_utils.py +0 -0
  41. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/crosshair.py +0 -0
  42. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/entry_validator.py +0 -0
  43. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/layout_manager.py +0 -0
  44. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/plugin_utils.py +0 -0
  45. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/rpc_decorator.py +0 -0
  46. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/thread_checker.py +0 -0
  47. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/ui_loader.py +0 -0
  48. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/validator_delegate.py +0 -0
  49. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/widget_io.py +0 -0
  50. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/utils/yaml_dialog.py +0 -0
  51. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/__init__.py +0 -0
  52. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/buttons/__init__.py +0 -0
  53. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/buttons/stop_button/__init__.py +0 -0
  54. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/buttons/stop_button/stop_button.py +0 -0
  55. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/dock/__init__.py +0 -0
  56. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/dock/dock.py +0 -0
  57. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/dock/dock_area.py +0 -0
  58. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/__init__.py +0 -0
  59. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/figure.py +0 -0
  60. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/__init__.py +0 -0
  61. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/image/__init__.py +0 -0
  62. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/image/image.py +0 -0
  63. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/image/image_item.py +0 -0
  64. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/image/image_processor.py +0 -0
  65. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/motor_map/__init__.py +0 -0
  66. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/motor_map/motor_map.py +0 -0
  67. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/plot_base.py +0 -0
  68. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/waveform/__init__.py +0 -0
  69. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/waveform/waveform.py +0 -0
  70. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/figure/plots/waveform/waveform_curve.py +0 -0
  71. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/jupyter_console/__init__.py +0 -0
  72. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/jupyter_console/jupyter_console.py +0 -0
  73. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/__init__.py +0 -0
  74. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/motor_control.py +0 -0
  75. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/motor_table/__init__.py +0 -0
  76. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/motor_table/motor_table.py +0 -0
  77. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/motor_table/motor_table.ui +0 -0
  78. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/movement_absolute/__init__.py +0 -0
  79. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/movement_absolute/movement_absolute.py +0 -0
  80. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/movement_absolute/movement_absolute.ui +0 -0
  81. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/movement_relative/__init__.py +0 -0
  82. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/movement_relative/movement_relative.py +0 -0
  83. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/movement_relative/movement_relative.ui +0 -0
  84. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/selection/__init__.py +0 -0
  85. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/selection/selection.py +0 -0
  86. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/motor_control/selection/selection.ui +0 -0
  87. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/scan_control/__init__.py +0 -0
  88. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/scan_control/scan_control.py +0 -0
  89. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/spiral_progress_bar/__init__.py +0 -0
  90. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/spiral_progress_bar/ring.py +0 -0
  91. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/spiral_progress_bar/spiral_progress_bar.py +0 -0
  92. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/toolbar/__init__.py +0 -0
  93. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/toolbar/toolbar.py +0 -0
  94. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/website/__init__.py +0 -0
  95. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/bec_widgets/widgets/website/website.py +0 -0
  96. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/Makefile +0 -0
  97. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/_static/custom.css +0 -0
  98. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/_templates/custom-class-template.rst +0 -0
  99. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/_templates/custom-module-template.rst +0 -0
  100. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/api_reference/api_reference.md +0 -0
  101. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/assets/apps_48dp.svg +0 -0
  102. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/assets/display_settings_48dp.svg +0 -0
  103. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/assets/index_api.svg +0 -0
  104. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/assets/index_contribute.svg +0 -0
  105. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/assets/index_getting_started.svg +0 -0
  106. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/assets/index_user_guide.svg +0 -0
  107. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/assets/rocket_launch_48dp.svg +0 -0
  108. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/conf.py +0 -0
  109. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/developer/developer.md +0 -0
  110. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/index.md +0 -0
  111. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/introduction/introduction.md +0 -0
  112. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/make.bat +0 -0
  113. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/requirements.txt +0 -0
  114. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/api_reference/api_reference.md +0 -0
  115. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/applications/applications.md +0 -0
  116. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/getting_started/BECDockArea.png +0 -0
  117. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/getting_started/auto_updates.md +0 -0
  118. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/getting_started/getting_started.md +0 -0
  119. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/getting_started/gui_complex_gui.gif +0 -0
  120. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/getting_started/installation.md +0 -0
  121. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/getting_started/quick_start.md +0 -0
  122. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/user.md +0 -0
  123. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/BECFigure.png +0 -0
  124. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/bec_figure.md +0 -0
  125. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/buttons.md +0 -0
  126. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/image_plot.gif +0 -0
  127. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/motor.gif +0 -0
  128. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/progress_bar.gif +0 -0
  129. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/scatter_2D.gif +0 -0
  130. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/spiral_progress_bar.md +0 -0
  131. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/w1D.gif +0 -0
  132. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/website.md +0 -0
  133. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/docs/user/widgets/widgets.md +0 -0
  134. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/__init__.py +0 -0
  135. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/end-2-end/__init__.py +0 -0
  136. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/end-2-end/conftest.py +0 -0
  137. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/end-2-end/test_bec_dock_rpc_e2e.py +0 -0
  138. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/end-2-end/test_bec_figure_rpc_e2e.py +0 -0
  139. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/end-2-end/test_rpc_register_e2e.py +0 -0
  140. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/__init__.py +0 -0
  141. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/client_mocks.py +0 -0
  142. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/conftest.py +0 -0
  143. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_bec_connector.py +0 -0
  144. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_bec_dispatcher.py +0 -0
  145. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_bec_dock.py +0 -0
  146. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_bec_figure.py +0 -0
  147. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_bec_motor_map.py +0 -0
  148. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_client_utils.py +0 -0
  149. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_color_validation.py +0 -0
  150. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_configs/config_device.yaml +0 -0
  151. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_configs/config_device_no_entry.yaml +0 -0
  152. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_configs/config_scan.yaml +0 -0
  153. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_crosshair.py +0 -0
  154. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_generate_cli_client.py +0 -0
  155. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_motor_control.py +0 -0
  156. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_msgs/__init__.py +0 -0
  157. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_msgs/available_scans_message.py +0 -0
  158. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_plot_base.py +0 -0
  159. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_rpc_register.py +0 -0
  160. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_scan_control.py +0 -0
  161. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_spiral_progress_bar.py +0 -0
  162. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_stop_button.py +0 -0
  163. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_waveform1d.py +0 -0
  164. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_website_widget.py +0 -0
  165. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_widget_io.py +0 -0
  166. {bec_widgets-0.61.0 → bec_widgets-0.62.0}/tests/unit_tests/test_yaml_dialog.py +0 -0
@@ -2,6 +2,17 @@
2
2
 
3
3
 
4
4
 
5
+ ## v0.62.0 (2024-06-12)
6
+
7
+ ### Feature
8
+
9
+ * feat: implement non-polling, interruptible waiting of gui instruction response with timeout ([`abc6caa`](https://gitlab.psi.ch/bec/bec_widgets/-/commit/abc6caa2d0b6141dfbe1f3d025f78ae14deddcb3))
10
+
11
+ ### Unknown
12
+
13
+ * doc: add documentation about creating custom GUI applications embedding BEC Widgets ([`17a0068`](https://gitlab.psi.ch/bec/bec_widgets/-/commit/17a00687579f5efab1990cd83862ec0e78198633))
14
+
15
+
5
16
  ## v0.61.0 (2024-06-12)
6
17
 
7
18
  ### Feature
@@ -152,12 +163,3 @@
152
163
  ### Fix
153
164
 
154
165
  * fix(docks): set_title do update dock internal _name now ([`15cbc21`](https://gitlab.psi.ch/bec/bec_widgets/-/commit/15cbc21e5bb3cf85f5822d44a2b3665b5aa2f346))
155
-
156
- * fix(docks): docks widget_list adn dockarea panels return values fixed ([`ffae5ee`](https://gitlab.psi.ch/bec/bec_widgets/-/commit/ffae5ee54e6b43da660131092452adff195ba4fb))
157
-
158
-
159
- ## v0.57.3 (2024-06-06)
160
-
161
- ### Documentation
162
-
163
- * docs(bar): docs updated ([`4be0d14`](https://gitlab.psi.ch/bec/bec_widgets/-/commit/4be0d14b7445c2322c2aef86257db168a841265c))
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.3
2
2
  Name: bec_widgets
3
- Version: 0.61.0
3
+ Version: 0.62.0
4
4
  Summary: BEC Widgets
5
5
  Project-URL: Bug Tracker, https://gitlab.psi.ch/bec/bec_widgets/issues
6
6
  Project-URL: Homepage, https://gitlab.psi.ch/bec/bec_widgets
@@ -14,7 +14,7 @@ from typing import TYPE_CHECKING
14
14
 
15
15
  from bec_lib.endpoints import MessageEndpoints
16
16
  from bec_lib.utils.import_utils import isinstance_based_on_class_name, lazy_import, lazy_import_from
17
- from qtpy.QtCore import QCoreApplication
17
+ from qtpy.QtCore import QEventLoop, QSocketNotifier, QTimer
18
18
 
19
19
  import bec_widgets.cli.client as client
20
20
  from bec_widgets.cli.auto_updates import AutoUpdates
@@ -24,6 +24,8 @@ if TYPE_CHECKING:
24
24
 
25
25
  from bec_widgets.cli.client import BECDockArea, BECFigure
26
26
 
27
+ from bec_lib.serialization import MsgpackSerialization
28
+
27
29
  messages = lazy_import("bec_lib.messages")
28
30
  # from bec_lib.connector import MessageObject
29
31
  MessageObject = lazy_import_from("bec_lib.connector", ("MessageObject",))
@@ -205,6 +207,48 @@ class RPCResponseTimeoutError(Exception):
205
207
  )
206
208
 
207
209
 
210
+ class QtRedisMessageWaiter:
211
+ def __init__(self, redis_connector, message_to_wait):
212
+ self.ev_loop = QEventLoop()
213
+ self.response = None
214
+ self.connector = redis_connector
215
+ self.message_to_wait = message_to_wait
216
+ self.pubsub = redis_connector._redis_conn.pubsub()
217
+ self.pubsub.subscribe(self.message_to_wait.endpoint)
218
+ fd = self.pubsub.connection._sock.fileno()
219
+ self.notifier = QSocketNotifier(fd, QSocketNotifier.Read)
220
+ self.notifier.activated.connect(self._pubsub_readable)
221
+
222
+ def _msg_received(self, msg_obj):
223
+ self.response = msg_obj.value
224
+ self.ev_loop.quit()
225
+
226
+ def wait(self, timeout=1):
227
+ timer = QTimer()
228
+ timer.singleShot(timeout * 1000, self.ev_loop.quit)
229
+ self.ev_loop.exec_()
230
+ timer.stop()
231
+ self.notifier.setEnabled(False)
232
+ self.pubsub.close()
233
+ return self.response
234
+
235
+ def _pubsub_readable(self, fd):
236
+ while True:
237
+ msg = self.pubsub.get_message()
238
+ if msg:
239
+ if msg["type"] == "subscribe":
240
+ # get_message buffers, so we may already have the answer
241
+ # let's check...
242
+ continue
243
+ else:
244
+ break
245
+ else:
246
+ return
247
+ channel = msg["channel"].decode()
248
+ msg = MessageObject(topic=channel, value=MsgpackSerialization.loads(msg["data"]))
249
+ self.connector._execute_callback(self._msg_received, msg, {})
250
+
251
+
208
252
  class RPCBase:
209
253
  def __init__(self, gui_id: str = None, config: dict = None, parent=None) -> None:
210
254
  self._client = BECDispatcher().client
@@ -231,7 +275,7 @@ class RPCBase:
231
275
  parent = parent._parent
232
276
  return parent
233
277
 
234
- def _run_rpc(self, method, *args, wait_for_rpc_response=True, **kwargs):
278
+ def _run_rpc(self, method, *args, wait_for_rpc_response=True, timeout=3, **kwargs):
235
279
  """
236
280
  Run the RPC call.
237
281
 
@@ -253,16 +297,24 @@ class RPCBase:
253
297
 
254
298
  # pylint: disable=protected-access
255
299
  receiver = self._root._gui_id
300
+ if wait_for_rpc_response:
301
+ redis_msg = QtRedisMessageWaiter(
302
+ self._client.connector, MessageEndpoints.gui_instruction_response(request_id)
303
+ )
304
+
256
305
  self._client.connector.set_and_publish(MessageEndpoints.gui_instructions(receiver), rpc_msg)
257
306
 
258
- if not wait_for_rpc_response:
259
- return None
260
- response = self._wait_for_response(request_id)
261
- # get class name
262
- if not response.content["accepted"]:
263
- raise ValueError(response.content["message"]["error"])
264
- msg_result = response.content["message"].get("result")
265
- return self._create_widget_from_msg_result(msg_result)
307
+ if wait_for_rpc_response:
308
+ response = redis_msg.wait(timeout)
309
+
310
+ if response is None:
311
+ raise RPCResponseTimeoutError(request_id, timeout)
312
+
313
+ # get class name
314
+ if not response.accepted:
315
+ raise ValueError(response.message["error"])
316
+ msg_result = response.message.get("result")
317
+ return self._create_widget_from_msg_result(msg_result)
266
318
 
267
319
  def _create_widget_from_msg_result(self, msg_result):
268
320
  if msg_result is None:
@@ -285,30 +337,6 @@ class RPCBase:
285
337
  return cls(parent=self, **msg_result)
286
338
  return msg_result
287
339
 
288
- def _wait_for_response(self, request_id: str, timeout: int = 5):
289
- """
290
- Wait for the response from the server.
291
-
292
- Args:
293
- request_id(str): The request ID.
294
- timeout(int): The timeout in seconds.
295
-
296
- Returns:
297
- The response from the server.
298
- """
299
- start_time = time.time()
300
- response = None
301
-
302
- while response is None and self.gui_is_alive() and (time.time() - start_time) < timeout:
303
- response = self._client.connector.get(
304
- MessageEndpoints.gui_instruction_response(request_id)
305
- )
306
- QCoreApplication.processEvents() # keep UI responsive (and execute signals/slots)
307
- if response is None and (time.time() - start_time) >= timeout:
308
- raise RPCResponseTimeoutError(request_id, timeout)
309
-
310
- return response
311
-
312
340
  def gui_is_alive(self):
313
341
  """
314
342
  Check if the GUI is alive.
@@ -136,17 +136,18 @@ if __name__ == "__main__": # pragma: no cover
136
136
 
137
137
  module_path = os.path.dirname(bec_widgets.__file__)
138
138
 
139
- bec_dispatcher = BECDispatcher()
140
- client = bec_dispatcher.client
141
- client.start()
142
-
143
139
  app = QApplication(sys.argv)
144
140
  app.setApplicationName("Jupyter Console")
145
141
  app.setApplicationDisplayName("Jupyter Console")
146
- # qdarktheme.setup_theme("auto")
142
+ qdarktheme.setup_theme("auto")
147
143
  icon = QIcon()
148
144
  icon.addFile(os.path.join(module_path, "assets", "terminal_icon.png"), size=QSize(48, 48))
149
145
  app.setWindowIcon(icon)
146
+
147
+ bec_dispatcher = BECDispatcher()
148
+ client = bec_dispatcher.client
149
+ client.start()
150
+
150
151
  win = JupyterConsoleWindow()
151
152
  win.show()
152
153
 
@@ -9,7 +9,7 @@ import redis
9
9
  from bec_lib.client import BECClient
10
10
  from bec_lib.redis_connector import MessageObject, RedisConnector
11
11
  from bec_lib.service_config import ServiceConfig
12
- from qtpy.QtCore import QObject
12
+ from qtpy.QtCore import QCoreApplication, QObject
13
13
  from qtpy.QtCore import Signal as pyqtSignal
14
14
 
15
15
  if TYPE_CHECKING:
@@ -71,6 +71,7 @@ class BECDispatcher:
71
71
 
72
72
  _instance = None
73
73
  _initialized = False
74
+ qapp = None
74
75
 
75
76
  def __new__(cls, client=None, config: str = None, *args, **kwargs):
76
77
  if cls._instance is None:
@@ -82,6 +83,9 @@ class BECDispatcher:
82
83
  if self._initialized:
83
84
  return
84
85
 
86
+ if not QCoreApplication.instance():
87
+ BECDispatcher.qapp = QCoreApplication([])
88
+
85
89
  self._slots = collections.defaultdict(set)
86
90
  self.client = client
87
91
 
@@ -0,0 +1,123 @@
1
+ (user.customisation)=
2
+ # Customisation
3
+
4
+ ## Leveraging BEC Widgets in custom GUI applications
5
+
6
+ BEC Widgets can be used to compose a complete Qt graphical application, along with
7
+ other QWidgets. The only requirement is to connect to BEC servers in order to get
8
+ data, or to interact with BEC components. This role is devoted to the BECDispatcher,
9
+ a singleton object which has to be instantiated **after the QApplication is created**.
10
+
11
+ A typical BEC Widgets custom application "main" entry point should follow the template
12
+ below:
13
+
14
+ ```
15
+ import argparse
16
+ import sys
17
+
18
+ from bec_widgets.utils.bec_dispatcher import BECDispatcher
19
+ from qtpy.QtWidgets import QApplication
20
+
21
+ # optional command line arguments processing
22
+ parser = argparse.ArgumentParser(description="...")
23
+ parser.add_argument( ...)
24
+ ...
25
+ args = parser.parse_args()
26
+
27
+ # creation of the Qt application
28
+ app = QApplication([])
29
+
30
+ # creation of BEC Dispatcher
31
+ # /!\ important: after the QApplication has been instantiated
32
+ bec_dispatcher = BECDispatcher()
33
+ client = bec_dispatcher.client
34
+ client.start()
35
+
36
+ # (optional) processing of command line args,
37
+ # creation of a main window depending on the command line arguments (or not)
38
+ if args.xxx == "...":
39
+ window = ...
40
+
41
+ # display of the main window and start of Qt event loop
42
+ window.show()
43
+ sys.exit(app.exec())
44
+ ```
45
+
46
+ The main "window" object presents the layout of widgets to the user and allows
47
+ users to interact. BEC Widgets must be placed in the window:
48
+
49
+ ```
50
+ from qtpy.QWidgets import QMainWindow
51
+ from bec_widgets.widgets import BECFigure
52
+
53
+ window = QMainWindow()
54
+ bec_figure = BECFigure(gui_id="my_gui_app_id")
55
+ window.setCentralWidget(bec_figure)
56
+
57
+ # prepare to plot samx motor vs bpm4i value
58
+ bec_figure.plot(x_name="samx", y_name="bpm4i")
59
+ ```
60
+
61
+ In the example just above, the resulting application will show a plot of samx
62
+ positions on the horizontal axis, and beam intensity on the vertical axis
63
+ (when the next scan will be started).
64
+
65
+ It is important to ensure proper cleanup of the resources is done when application
66
+ quits:
67
+
68
+ ```
69
+ def final_cleanup():
70
+ bec_figure.clear_all()
71
+ bec_figure.client.shutdown()
72
+
73
+ window.aboutToQuit.connect(final_cleanup)
74
+ ```
75
+
76
+ Final example:
77
+
78
+ ```
79
+ import sys
80
+ from qtpy.QtWidgets import QMainWindow, QApplication
81
+ from bec_widgets.widgets import BECFigure
82
+ from bec_widgets.utils.bec_dispatcher import BECDispatcher
83
+
84
+ # creation of the Qt application
85
+ app = QApplication([])
86
+
87
+ # creation of BEC Dispatcher
88
+ bec_dispatcher = BECDispatcher()
89
+ client = bec_dispatcher.client
90
+ client.start()
91
+
92
+ # creation of main window
93
+ window = QMainWindow()
94
+
95
+ # inserting BEC Widgets
96
+ bec_figure = BECFigure(parent=window, gui_id="my_gui_app_id")
97
+ window.setCentralWidget(bec_figure)
98
+
99
+ bec_figure.plot(x_name="samx", y_name="bpm4i")
100
+
101
+ # ensuring proper cleanup
102
+ def final_cleanup():
103
+ bec_figure.clear_all()
104
+ bec_figure.client.shutdown()
105
+
106
+ app.aboutToQuit.connect(final_cleanup)
107
+
108
+ # execution
109
+ window.show()
110
+ sys.exit(app.exec())
111
+ ```
112
+
113
+ ## Writing applications using Qt Designer
114
+
115
+ BEC Widgets are designed to be used with QtDesigner to quickly design GUI.
116
+
117
+ ## Example of promoting widgets in Qt Designer
118
+
119
+ _Work in progress_
120
+
121
+ ## Implementation of plugins into Qt Designer
122
+
123
+ _Work in progress_
@@ -4,7 +4,7 @@ build-backend = "hatchling.build"
4
4
 
5
5
  [project]
6
6
  name = "bec_widgets"
7
- version = "0.61.0"
7
+ version = "0.62.0"
8
8
  description = "BEC Widgets"
9
9
  requires-python = ">=3.10"
10
10
  classifiers = [
@@ -1,13 +0,0 @@
1
- (user.customisation)=
2
- # Customisation
3
-
4
- BEC Widgets are designed to be used with QtDesigner to quicly design GUI.
5
-
6
-
7
- ## Example of promoting widgets in Qt Designer
8
-
9
- _Work in progress_
10
-
11
- ## Implementation of plugins into Qt Designer
12
-
13
- _Work in progress_
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes