glimmer-dsl-swt 4.18.3.5 → 4.18.4.4

Sign up to get free protection for your applications and to get access to all the features.
Files changed (162) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +52 -0
  3. data/LICENSE.txt +20 -20
  4. data/README.md +782 -716
  5. data/RUBY_VERSION +1 -1
  6. data/VERSION +1 -1
  7. data/bin/girb +31 -31
  8. data/bin/girb_runner.rb +34 -34
  9. data/bin/glimmer +26 -26
  10. data/glimmer-dsl-swt.gemspec +10 -7
  11. data/lib/ext/glimmer.rb +41 -41
  12. data/lib/ext/glimmer/config.rb +167 -167
  13. data/lib/ext/rouge/themes/glimmer.rb +29 -29
  14. data/lib/glimmer-dsl-swt.rb +44 -44
  15. data/lib/glimmer/Rakefile +26 -26
  16. data/lib/glimmer/data_binding/list_selection_binding.rb +72 -72
  17. data/lib/glimmer/data_binding/observable_widget.rb +38 -38
  18. data/lib/glimmer/data_binding/shine.rb +44 -44
  19. data/lib/glimmer/data_binding/table_items_binding.rb +89 -89
  20. data/lib/glimmer/data_binding/tree_items_binding.rb +108 -108
  21. data/lib/glimmer/data_binding/widget_binding.rb +73 -73
  22. data/lib/glimmer/dsl/swt/animation_expression.rb +43 -43
  23. data/lib/glimmer/dsl/swt/async_exec_expression.rb +35 -35
  24. data/lib/glimmer/dsl/swt/bind_expression.rb +58 -58
  25. data/lib/glimmer/dsl/swt/block_property_expression.rb +41 -41
  26. data/lib/glimmer/dsl/swt/checkbox_group_selection_data_binding_expression.rb +61 -61
  27. data/lib/glimmer/dsl/swt/color_expression.rb +40 -40
  28. data/lib/glimmer/dsl/swt/column_properties_expression.rb +45 -45
  29. data/lib/glimmer/dsl/swt/combo_selection_data_binding_expression.rb +67 -67
  30. data/lib/glimmer/dsl/swt/cursor_expression.rb +44 -44
  31. data/lib/glimmer/dsl/swt/custom_widget_expression.rb +63 -63
  32. data/lib/glimmer/dsl/swt/data_binding_expression.rb +55 -55
  33. data/lib/glimmer/dsl/swt/dialog_expression.rb +48 -48
  34. data/lib/glimmer/dsl/swt/directory_dialog_expression.rb +48 -48
  35. data/lib/glimmer/dsl/swt/display_expression.rb +40 -40
  36. data/lib/glimmer/dsl/swt/dnd_expression.rb +46 -46
  37. data/lib/glimmer/dsl/swt/dsl.rb +63 -63
  38. data/lib/glimmer/dsl/swt/exec_expression.rb +55 -55
  39. data/lib/glimmer/dsl/swt/expand_item_expression.rb +60 -60
  40. data/lib/glimmer/dsl/swt/file_dialog_expression.rb +48 -48
  41. data/lib/glimmer/dsl/swt/font_expression.rb +49 -49
  42. data/lib/glimmer/dsl/swt/image_expression.rb +50 -50
  43. data/lib/glimmer/dsl/swt/layout_data_expression.rb +46 -46
  44. data/lib/glimmer/dsl/swt/layout_expression.rb +50 -50
  45. data/lib/glimmer/dsl/swt/list_selection_data_binding_expression.rb +65 -65
  46. data/lib/glimmer/dsl/swt/menu_bar_expression.rb +54 -54
  47. data/lib/glimmer/dsl/swt/menu_expression.rb +53 -53
  48. data/lib/glimmer/dsl/swt/message_box_expression.rb +54 -54
  49. data/lib/glimmer/dsl/swt/multiply_expression.rb +53 -53
  50. data/lib/glimmer/dsl/swt/observe_expression.rb +53 -53
  51. data/lib/glimmer/dsl/swt/property_expression.rb +46 -46
  52. data/lib/glimmer/dsl/swt/radio_group_selection_data_binding_expression.rb +61 -61
  53. data/lib/glimmer/dsl/swt/rgb_expression.rb +33 -33
  54. data/lib/glimmer/dsl/swt/rgba_expression.rb +33 -33
  55. data/lib/glimmer/dsl/swt/shape_expression.rb +54 -54
  56. data/lib/glimmer/dsl/swt/shell_expression.rb +46 -46
  57. data/lib/glimmer/dsl/swt/swt_expression.rb +46 -46
  58. data/lib/glimmer/dsl/swt/sync_exec_expression.rb +36 -36
  59. data/lib/glimmer/dsl/swt/tab_item_expression.rb +54 -54
  60. data/lib/glimmer/dsl/swt/table_items_data_binding_expression.rb +52 -52
  61. data/lib/glimmer/dsl/swt/timer_exec_expression.rb +35 -0
  62. data/lib/glimmer/dsl/swt/transform_expression.rb +55 -55
  63. data/lib/glimmer/dsl/swt/tree_items_data_binding_expression.rb +52 -52
  64. data/lib/glimmer/dsl/swt/tree_properties_expression.rb +47 -47
  65. data/lib/glimmer/dsl/swt/widget_expression.rb +66 -66
  66. data/lib/glimmer/dsl/swt/widget_listener_expression.rb +53 -53
  67. data/lib/glimmer/rake_task.rb +220 -220
  68. data/lib/glimmer/rake_task/list.rb +97 -97
  69. data/lib/glimmer/rake_task/package.rb +139 -139
  70. data/lib/glimmer/rake_task/scaffold.rb +765 -765
  71. data/lib/glimmer/swt/color_proxy.rb +107 -107
  72. data/lib/glimmer/swt/cursor_proxy.rb +66 -66
  73. data/lib/glimmer/swt/custom/animation.rb +243 -243
  74. data/lib/glimmer/swt/custom/checkbox_group.rb +181 -181
  75. data/lib/glimmer/swt/custom/code_text.rb +150 -93
  76. data/lib/glimmer/swt/custom/drawable.rb +49 -52
  77. data/lib/glimmer/swt/custom/radio_group.rb +176 -176
  78. data/lib/glimmer/swt/custom/shape.rb +297 -253
  79. data/lib/glimmer/swt/date_time_proxy.rb +85 -85
  80. data/lib/glimmer/swt/directory_dialog_proxy.rb +65 -65
  81. data/lib/glimmer/swt/display_proxy.rb +166 -155
  82. data/lib/glimmer/swt/dnd_proxy.rb +51 -51
  83. data/lib/glimmer/swt/expand_item_proxy.rb +97 -97
  84. data/lib/glimmer/swt/file_dialog_proxy.rb +66 -66
  85. data/lib/glimmer/swt/font_proxy.rb +94 -94
  86. data/lib/glimmer/swt/image_proxy.rb +184 -173
  87. data/lib/glimmer/swt/layout_data_proxy.rb +105 -105
  88. data/lib/glimmer/swt/layout_proxy.rb +112 -109
  89. data/lib/glimmer/swt/menu_proxy.rb +126 -126
  90. data/lib/glimmer/swt/message_box_proxy.rb +89 -89
  91. data/lib/glimmer/swt/packages.rb +37 -37
  92. data/lib/glimmer/swt/properties.rb +49 -49
  93. data/lib/glimmer/swt/sash_form_proxy.rb +53 -53
  94. data/lib/glimmer/swt/scrolled_composite_proxy.rb +37 -37
  95. data/lib/glimmer/swt/shell_proxy.rb +4 -1
  96. data/lib/glimmer/swt/style_constantizable.rb +157 -157
  97. data/lib/glimmer/swt/styled_text_proxy.rb +38 -38
  98. data/lib/glimmer/swt/swt_proxy.rb +59 -59
  99. data/lib/glimmer/swt/tab_item_proxy.rb +91 -91
  100. data/lib/glimmer/swt/table_column_proxy.rb +57 -57
  101. data/lib/glimmer/swt/table_proxy.rb +2 -2
  102. data/lib/glimmer/swt/transform_proxy.rb +109 -109
  103. data/lib/glimmer/swt/tree_proxy.rb +145 -145
  104. data/lib/glimmer/swt/widget_listener_proxy.rb +64 -64
  105. data/lib/glimmer/swt/widget_proxy.rb +957 -949
  106. data/lib/glimmer/ui/custom_shell.rb +82 -82
  107. data/lib/glimmer/ui/custom_widget.rb +315 -315
  108. data/lib/glimmer/util/proc_tracker.rb +39 -39
  109. data/samples/elaborate/contact_manager.rb +142 -142
  110. data/samples/elaborate/contact_manager/contact.rb +32 -32
  111. data/samples/elaborate/contact_manager/contact_manager_presenter.rb +47 -47
  112. data/samples/elaborate/contact_manager/contact_repository.rb +169 -169
  113. data/samples/elaborate/login.rb +123 -123
  114. data/samples/elaborate/meta_sample.rb +14 -3
  115. data/samples/elaborate/tetris.rb +16 -6
  116. data/samples/elaborate/tetris/model/block.rb +48 -48
  117. data/samples/elaborate/tetris/model/game.rb +2 -3
  118. data/samples/elaborate/tetris/model/past_game.rb +39 -39
  119. data/samples/elaborate/tetris/view/block.rb +1 -1
  120. data/samples/elaborate/tetris/view/high_score_dialog.rb +0 -7
  121. data/samples/elaborate/tetris/view/playfield.rb +1 -1
  122. data/samples/elaborate/tetris/view/tetris_menu_bar.rb +13 -11
  123. data/samples/elaborate/tic_tac_toe.rb +76 -76
  124. data/samples/elaborate/tic_tac_toe/board.rb +145 -145
  125. data/samples/elaborate/tic_tac_toe/cell.rb +48 -48
  126. data/samples/elaborate/user_profile.rb +76 -76
  127. data/samples/hello/hello_browser.rb +31 -31
  128. data/samples/hello/hello_button.rb +46 -46
  129. data/samples/hello/hello_canvas.rb +64 -63
  130. data/samples/hello/hello_canvas_animation.rb +3 -3
  131. data/samples/hello/hello_canvas_transform.rb +1 -1
  132. data/samples/hello/hello_checkbox.rb +85 -85
  133. data/samples/hello/hello_checkbox_group.rb +71 -71
  134. data/samples/hello/hello_code_text.rb +104 -84
  135. data/samples/hello/hello_combo.rb +63 -63
  136. data/samples/hello/hello_computed.rb +96 -96
  137. data/samples/hello/hello_computed/contact.rb +42 -42
  138. data/samples/hello/hello_custom_shell.rb +155 -155
  139. data/samples/hello/hello_custom_widget.rb +86 -86
  140. data/samples/hello/hello_date_time.rb +63 -63
  141. data/samples/hello/hello_dialog.rb +78 -78
  142. data/samples/hello/hello_directory_dialog.rb +60 -60
  143. data/samples/hello/hello_drag_and_drop.rb +50 -50
  144. data/samples/hello/hello_expand_bar.rb +110 -110
  145. data/samples/hello/hello_file_dialog.rb +60 -60
  146. data/samples/hello/hello_group.rb +104 -104
  147. data/samples/hello/hello_link.rb +80 -80
  148. data/samples/hello/hello_list_multi_selection.rb +74 -74
  149. data/samples/hello/hello_list_single_selection.rb +59 -59
  150. data/samples/hello/hello_menu_bar.rb +241 -241
  151. data/samples/hello/hello_message_box.rb +37 -37
  152. data/samples/hello/hello_pop_up_context_menu.rb +84 -84
  153. data/samples/hello/hello_radio.rb +108 -108
  154. data/samples/hello/hello_radio_group.rb +87 -87
  155. data/samples/hello/hello_sash_form.rb +137 -137
  156. data/samples/hello/hello_spinner.rb +69 -69
  157. data/samples/hello/hello_styled_text.rb +138 -138
  158. data/samples/hello/hello_tab.rb +50 -50
  159. data/samples/hello/hello_table.rb +7 -4
  160. data/samples/hello/hello_table/baseball_park.png +0 -0
  161. data/samples/hello/hello_world.rb +29 -29
  162. metadata +7 -4
@@ -1,82 +1,82 @@
1
- # Copyright (c) 2007-2021 Andy Maleh
2
- #
3
- # Permission is hereby granted, free of charge, to any person obtaining
4
- # a copy of this software and associated documentation files (the
5
- # "Software"), to deal in the Software without restriction, including
6
- # without limitation the rights to use, copy, modify, merge, publish,
7
- # distribute, sublicense, and/or sell copies of the Software, and to
8
- # permit persons to whom the Software is furnished to do so, subject to
9
- # the following conditions:
10
- #
11
- # The above copyright notice and this permission notice shall be
12
- # included in all copies or substantial portions of the Software.
13
- #
14
- # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
- # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
- # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
- # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
- # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
- # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
- # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21
-
22
- require 'glimmer/error'
23
-
24
- module Glimmer
25
- module UI
26
- module CustomShell
27
- include SuperModule
28
- include Glimmer::UI::CustomWidget
29
-
30
- class << self
31
- attr_reader :launched_custom_shell
32
-
33
- def launch(*args, &content)
34
- @launched_custom_shell = send(keyword, *args, &content) if @launched_custom_shell.nil? || @launched_custom_shell.disposed?
35
- @launched_custom_shell.swt_widget.set_data('launched', true)
36
- @launched_custom_shell.open
37
- end
38
- end
39
-
40
- def initialize(parent, *swt_constants, options, &content)
41
- super
42
- @swt_widget.set_data('custom_shell', self)
43
- raise Error, 'Invalid custom shell body root! Must be a shell or another custom shell.' unless body_root.swt_widget.is_a?(org.eclipse.swt.widgets.Shell)
44
- end
45
-
46
- # Classes may override
47
- def open
48
- body_root.open
49
- end
50
-
51
- # DO NOT OVERRIDE. JUST AN ALIAS FOR `#open`. OVERRIDE `#open` INSTEAD.
52
- def show
53
- open
54
- end
55
-
56
- # TODO consider using Forwardable instead
57
- def close
58
- body_root.close
59
- end
60
-
61
- def hide
62
- body_root.hide
63
- end
64
-
65
- def visible?
66
- body_root.visible?
67
- end
68
-
69
- def disposed?
70
- swt_widget.is_disposed
71
- end
72
-
73
- def center_within_display
74
- body_root.center_within_display
75
- end
76
-
77
- def start_event_loop
78
- body_root.start_event_loop
79
- end
80
- end
81
- end
82
- end
1
+ # Copyright (c) 2007-2021 Andy Maleh
2
+ #
3
+ # Permission is hereby granted, free of charge, to any person obtaining
4
+ # a copy of this software and associated documentation files (the
5
+ # "Software"), to deal in the Software without restriction, including
6
+ # without limitation the rights to use, copy, modify, merge, publish,
7
+ # distribute, sublicense, and/or sell copies of the Software, and to
8
+ # permit persons to whom the Software is furnished to do so, subject to
9
+ # the following conditions:
10
+ #
11
+ # The above copyright notice and this permission notice shall be
12
+ # included in all copies or substantial portions of the Software.
13
+ #
14
+ # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21
+
22
+ require 'glimmer/error'
23
+
24
+ module Glimmer
25
+ module UI
26
+ module CustomShell
27
+ include SuperModule
28
+ include Glimmer::UI::CustomWidget
29
+
30
+ class << self
31
+ attr_reader :launched_custom_shell
32
+
33
+ def launch(*args, &content)
34
+ @launched_custom_shell = send(keyword, *args, &content) if @launched_custom_shell.nil? || @launched_custom_shell.disposed?
35
+ @launched_custom_shell.swt_widget.set_data('launched', true)
36
+ @launched_custom_shell.open
37
+ end
38
+ end
39
+
40
+ def initialize(parent, *swt_constants, options, &content)
41
+ super
42
+ @swt_widget.set_data('custom_shell', self)
43
+ raise Error, 'Invalid custom shell body root! Must be a shell or another custom shell.' unless body_root.swt_widget.is_a?(org.eclipse.swt.widgets.Shell)
44
+ end
45
+
46
+ # Classes may override
47
+ def open
48
+ body_root.open
49
+ end
50
+
51
+ # DO NOT OVERRIDE. JUST AN ALIAS FOR `#open`. OVERRIDE `#open` INSTEAD.
52
+ def show
53
+ open
54
+ end
55
+
56
+ # TODO consider using Forwardable instead
57
+ def close
58
+ body_root.close
59
+ end
60
+
61
+ def hide
62
+ body_root.hide
63
+ end
64
+
65
+ def visible?
66
+ body_root.visible?
67
+ end
68
+
69
+ def disposed?
70
+ swt_widget.is_disposed
71
+ end
72
+
73
+ def center_within_display
74
+ body_root.center_within_display
75
+ end
76
+
77
+ def start_event_loop
78
+ body_root.start_event_loop
79
+ end
80
+ end
81
+ end
82
+ end
@@ -1,315 +1,315 @@
1
- # Copyright (c) 2007-2021 Andy Maleh
2
- #
3
- # Permission is hereby granted, free of charge, to any person obtaining
4
- # a copy of this software and associated documentation files (the
5
- # "Software"), to deal in the Software without restriction, including
6
- # without limitation the rights to use, copy, modify, merge, publish,
7
- # distribute, sublicense, and/or sell copies of the Software, and to
8
- # permit persons to whom the Software is furnished to do so, subject to
9
- # the following conditions:
10
- #
11
- # The above copyright notice and this permission notice shall be
12
- # included in all copies or substantial portions of the Software.
13
- #
14
- # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
- # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
- # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
- # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
- # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
- # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
- # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21
-
22
- require 'glimmer'
23
- require 'glimmer/error'
24
- require 'glimmer/swt/swt_proxy'
25
- require 'glimmer/swt/display_proxy'
26
- require 'glimmer/util/proc_tracker'
27
- require 'glimmer/data_binding/observer'
28
- require 'glimmer/data_binding/observable_model'
29
-
30
- module Glimmer
31
- module UI
32
- module CustomWidget
33
- include SuperModule
34
- include DataBinding::ObservableModel
35
-
36
- super_module_included do |klass|
37
- # TODO clear memoization of WidgetProxy.swt_widget_class_for for a keyword if a custom widget was defined with that keyword
38
- klass.include(Glimmer) unless klass.name.include?('Glimmer::UI::CustomShell')
39
- Glimmer::UI::CustomWidget.add_custom_widget_namespaces_for(klass) unless klass.name.include?('Glimmer::UI::CustomShell')
40
- end
41
-
42
- class << self
43
- def for(underscored_custom_widget_name)
44
- unless flyweight_custom_widget_classes.keys.include?(underscored_custom_widget_name)
45
- begin
46
- extracted_namespaces = underscored_custom_widget_name.
47
- to_s.
48
- split(/__/).map do |namespace|
49
- namespace.camelcase(:upper)
50
- end
51
- custom_widget_namespaces.each do |base|
52
- extracted_namespaces.reduce(base) do |result, namespace|
53
- if !result.constants.include?(namespace)
54
- namespace = result.constants.detect {|c| c.to_s.upcase == namespace.to_s.upcase } || namespace
55
- end
56
- begin
57
- flyweight_custom_widget_classes[underscored_custom_widget_name] = constant = result.const_get(namespace)
58
- return constant if constant.ancestors.include?(Glimmer::UI::CustomWidget)
59
- flyweight_custom_widget_classes[underscored_custom_widget_name] = constant
60
- rescue => e
61
- # Glimmer::Config.logger.debug {"#{e.message}\n#{e.backtrace.join("\n")}"}
62
- flyweight_custom_widget_classes[underscored_custom_widget_name] = result
63
- end
64
- end
65
- end
66
- raise "#{underscored_custom_widget_name} has no custom widget class!"
67
- rescue => e
68
- Glimmer::Config.logger.debug {e.message}
69
- Glimmer::Config.logger.debug {"#{e.message}\n#{e.backtrace.join("\n")}"}
70
- flyweight_custom_widget_classes[underscored_custom_widget_name] = nil
71
- end
72
- end
73
- flyweight_custom_widget_classes[underscored_custom_widget_name]
74
- end
75
-
76
- # Flyweight Design Pattern memoization cache. Can be cleared if memory is needed.
77
- def flyweight_custom_widget_classes
78
- @flyweight_custom_widget_classes ||= {}
79
- end
80
-
81
- # Returns keyword to use for this custom widget
82
- def keyword
83
- self.name.underscore.gsub('::', '__')
84
- end
85
-
86
- # Returns shortcut keyword to use for this custom widget (keyword minus namespace)
87
- def shortcut_keyword
88
- self.name.underscore.gsub('::', '__').split('__').last
89
- end
90
-
91
- def add_custom_widget_namespaces_for(klass)
92
- Glimmer::UI::CustomWidget.namespaces_for_class(klass).drop(1).each do |namespace|
93
- Glimmer::UI::CustomWidget.custom_widget_namespaces << namespace
94
- end
95
- end
96
-
97
- def namespaces_for_class(m)
98
- return [m] if m.name.nil?
99
- namespace_constants = m.name.split(/::/).map(&:to_sym)
100
- namespace_constants.reduce([Object]) do |output, namespace_constant|
101
- output += [output.last.const_get(namespace_constant)]
102
- end[1..-1].uniq.reverse
103
- end
104
-
105
- def custom_widget_namespaces
106
- @custom_widget_namespaces ||= reset_custom_widget_namespaces
107
- end
108
-
109
- def reset_custom_widget_namespaces
110
- @custom_widget_namespaces = Set[Object, Glimmer::UI]
111
- end
112
-
113
- # Allows defining convenience option accessors for an array of option names
114
- # Example: `options :color1, :color2` defines `#color1` and `#color2`
115
- # where they return the instance values `options[:color1]` and `options[:color2]`
116
- # respectively.
117
- # Can be called multiple times to set more options additively.
118
- # When passed no arguments, it returns list of all option names captured so far
119
- def options(*new_options)
120
- new_options = new_options.compact.map(&:to_s).map(&:to_sym)
121
- if new_options.empty?
122
- @options ||= {} # maps options to defaults
123
- else
124
- new_options = new_options.reduce({}) {|new_options_hash, new_option| new_options_hash.merge(new_option => nil)}
125
- @options = options.merge(new_options)
126
- def_option_attr_accessors(new_options)
127
- end
128
- end
129
-
130
- def option(new_option, default: nil)
131
- new_option = new_option.to_s.to_sym
132
- new_options = {new_option => default}
133
- @options = options.merge(new_options)
134
- def_option_attr_accessors(new_options)
135
- end
136
-
137
- def def_option_attr_accessors(new_options)
138
- new_options.each do |option, default|
139
- class_eval <<-end_eval, __FILE__, __LINE__
140
- def #{option}
141
- options[:#{option}]
142
- end
143
- def #{option}=(option_value)
144
- self.options[:#{option}] = option_value
145
- end
146
- end_eval
147
- end
148
- end
149
-
150
- def before_body(&block)
151
- @before_body_block = block
152
- end
153
-
154
- def body(&block)
155
- @body_block = block
156
- end
157
-
158
- def after_body(&block)
159
- @after_body_block = block
160
- end
161
- end
162
-
163
- attr_reader :body_root, :swt_widget, :parent, :parent_proxy, :swt_style, :options
164
-
165
- def initialize(parent, *swt_constants, options, &content)
166
- @parent_proxy = @parent = parent
167
- @parent_proxy = @parent&.get_data('proxy') if @parent.respond_to?(:get_data) && @parent.get_data('proxy')
168
- @swt_style = SWT::SWTProxy[*swt_constants]
169
- options ||= {}
170
- @options = self.class.options.merge(options)
171
- @content = Util::ProcTracker.new(content) if content
172
- execute_hook('before_body')
173
- body_block = self.class.instance_variable_get("@body_block")
174
- raise Glimmer::Error, 'Invalid custom widget for having no body! Please define body block!' if body_block.nil?
175
- @body_root = instance_exec(&body_block)
176
- raise Glimmer::Error, 'Invalid custom widget for having an empty body! Please fill body block!' if @body_root.nil?
177
- @swt_widget = @body_root.swt_widget
178
- @swt_widget.set_data('custom_widget', self)
179
- execute_hook('after_body')
180
- end
181
-
182
- # Subclasses may override to perform post initialization work on an added child
183
- def post_initialize_child(child)
184
- # No Op by default
185
- end
186
-
187
- def can_handle_observation_request?(observation_request)
188
- observation_request = observation_request.to_s
189
- result = false
190
- if observation_request.start_with?('on_updated_')
191
- property = observation_request.sub(/^on_updated_/, '')
192
- result = can_add_observer?(property)
193
- end
194
- result || body_root&.can_handle_observation_request?(observation_request)
195
- end
196
-
197
- def handle_observation_request(observation_request, &block)
198
- observation_request = observation_request.to_s
199
- if observation_request.start_with?('on_updated_')
200
- property = observation_request.sub(/^on_updated_/, '') # TODO look into eliminating duplication from above
201
- add_observer(DataBinding::Observer.proc(&block), property) if can_add_observer?(property)
202
- else
203
- body_root.handle_observation_request(observation_request, &block)
204
- end
205
- end
206
-
207
- def can_add_observer?(attribute_name)
208
- has_instance_method?(attribute_name) || has_instance_method?("#{attribute_name}?") || @body_root.can_add_observer?(attribute_name)
209
- end
210
-
211
- def add_observer(observer, attribute_name)
212
- if has_instance_method?(attribute_name)
213
- super
214
- else
215
- @body_root.add_observer(observer, attribute_name)
216
- end
217
- end
218
-
219
- def has_attribute?(attribute_name, *args)
220
- has_instance_method?(attribute_setter(attribute_name)) ||
221
- @body_root.has_attribute?(attribute_name, *args)
222
- end
223
-
224
- def set_attribute(attribute_name, *args)
225
- if has_instance_method?(attribute_setter(attribute_name))
226
- send(attribute_setter(attribute_name), *args)
227
- else
228
- @body_root.set_attribute(attribute_name, *args)
229
- end
230
- end
231
-
232
- # This method ensures it has an instance method not coming from Glimmer DSL
233
- def has_instance_method?(method_name)
234
- respond_to?(method_name) and
235
- !swt_widget&.respond_to?(method_name) and
236
- (method(method_name) rescue nil) and
237
- !method(method_name)&.source_location&.first&.include?('glimmer/dsl/engine.rb') and
238
- !method(method_name)&.source_location&.first&.include?('glimmer/swt/widget_proxy.rb')
239
- end
240
-
241
- def get_attribute(attribute_name)
242
- if has_instance_method?(attribute_name)
243
- send(attribute_name)
244
- else
245
- @body_root.get_attribute(attribute_name)
246
- end
247
- end
248
-
249
- def attribute_setter(attribute_name)
250
- "#{attribute_name}="
251
- end
252
-
253
- def disposed?
254
- swt_widget.isDisposed
255
- end
256
-
257
- def has_style?(style)
258
- (swt_style & SWT::SWTProxy[style]) == SWT::SWTProxy[style]
259
- end
260
-
261
- def pack(*args)
262
- body_root.pack(*args)
263
- end
264
-
265
- # TODO see if it is worth it to eliminate duplication of async_exec/sync_exec
266
- # delegation to DisplayProxy, via a module
267
-
268
- def async_exec(&block)
269
- SWT::DisplayProxy.instance.async_exec(&block)
270
- end
271
-
272
- def sync_exec(&block)
273
- SWT::DisplayProxy.instance.sync_exec(&block)
274
- end
275
-
276
- # Returns content block if used as an attribute reader (no args)
277
- # Otherwise, if a block is passed, it adds it as content to this custom widget
278
- def content(&block)
279
- if block_given?
280
- body_root.content(&block)
281
- else
282
- @content
283
- end
284
- end
285
-
286
- def method_missing(method, *args, &block)
287
- # TODO Consider supporting a glimmer error silencing option for methods defined here
288
- # but fail the glimmer DSL for the right reason to avoid seeing noise in the log output
289
- if can_handle_observation_request?(method)
290
- handle_observation_request(method, &block)
291
- else
292
- body_root.send(method, *args, &block)
293
- end
294
- end
295
-
296
- alias local_respond_to? respond_to?
297
- def respond_to?(method, *args, &block)
298
- super or
299
- can_handle_observation_request?(method) or
300
- body_root.respond_to?(method, *args, &block)
301
- end
302
-
303
- private
304
-
305
- def execute_hook(hook_name)
306
- hook_block = self.class.instance_variable_get("@#{hook_name}_block")
307
- return if hook_block.nil?
308
- temp_method_name = "#{hook_name}_block_#{hook_block.hash.abs}_#{(Time.now.to_f * 1_000_000).to_i}"
309
- singleton_class.define_method(temp_method_name, &hook_block)
310
- send(temp_method_name)
311
- singleton_class.send(:remove_method, temp_method_name)
312
- end
313
- end
314
- end
315
- end
1
+ # Copyright (c) 2007-2021 Andy Maleh
2
+ #
3
+ # Permission is hereby granted, free of charge, to any person obtaining
4
+ # a copy of this software and associated documentation files (the
5
+ # "Software"), to deal in the Software without restriction, including
6
+ # without limitation the rights to use, copy, modify, merge, publish,
7
+ # distribute, sublicense, and/or sell copies of the Software, and to
8
+ # permit persons to whom the Software is furnished to do so, subject to
9
+ # the following conditions:
10
+ #
11
+ # The above copyright notice and this permission notice shall be
12
+ # included in all copies or substantial portions of the Software.
13
+ #
14
+ # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21
+
22
+ require 'glimmer'
23
+ require 'glimmer/error'
24
+ require 'glimmer/swt/swt_proxy'
25
+ require 'glimmer/swt/display_proxy'
26
+ require 'glimmer/util/proc_tracker'
27
+ require 'glimmer/data_binding/observer'
28
+ require 'glimmer/data_binding/observable_model'
29
+
30
+ module Glimmer
31
+ module UI
32
+ module CustomWidget
33
+ include SuperModule
34
+ include DataBinding::ObservableModel
35
+
36
+ super_module_included do |klass|
37
+ # TODO clear memoization of WidgetProxy.swt_widget_class_for for a keyword if a custom widget was defined with that keyword
38
+ klass.include(Glimmer) unless klass.name.include?('Glimmer::UI::CustomShell')
39
+ Glimmer::UI::CustomWidget.add_custom_widget_namespaces_for(klass) unless klass.name.include?('Glimmer::UI::CustomShell')
40
+ end
41
+
42
+ class << self
43
+ def for(underscored_custom_widget_name)
44
+ unless flyweight_custom_widget_classes.keys.include?(underscored_custom_widget_name)
45
+ begin
46
+ extracted_namespaces = underscored_custom_widget_name.
47
+ to_s.
48
+ split(/__/).map do |namespace|
49
+ namespace.camelcase(:upper)
50
+ end
51
+ custom_widget_namespaces.each do |base|
52
+ extracted_namespaces.reduce(base) do |result, namespace|
53
+ if !result.constants.include?(namespace)
54
+ namespace = result.constants.detect {|c| c.to_s.upcase == namespace.to_s.upcase } || namespace
55
+ end
56
+ begin
57
+ flyweight_custom_widget_classes[underscored_custom_widget_name] = constant = result.const_get(namespace)
58
+ return constant if constant.ancestors.include?(Glimmer::UI::CustomWidget)
59
+ flyweight_custom_widget_classes[underscored_custom_widget_name] = constant
60
+ rescue => e
61
+ # Glimmer::Config.logger.debug {"#{e.message}\n#{e.backtrace.join("\n")}"}
62
+ flyweight_custom_widget_classes[underscored_custom_widget_name] = result
63
+ end
64
+ end
65
+ end
66
+ raise "#{underscored_custom_widget_name} has no custom widget class!"
67
+ rescue => e
68
+ Glimmer::Config.logger.debug {e.message}
69
+ Glimmer::Config.logger.debug {"#{e.message}\n#{e.backtrace.join("\n")}"}
70
+ flyweight_custom_widget_classes[underscored_custom_widget_name] = nil
71
+ end
72
+ end
73
+ flyweight_custom_widget_classes[underscored_custom_widget_name]
74
+ end
75
+
76
+ # Flyweight Design Pattern memoization cache. Can be cleared if memory is needed.
77
+ def flyweight_custom_widget_classes
78
+ @flyweight_custom_widget_classes ||= {}
79
+ end
80
+
81
+ # Returns keyword to use for this custom widget
82
+ def keyword
83
+ self.name.underscore.gsub('::', '__')
84
+ end
85
+
86
+ # Returns shortcut keyword to use for this custom widget (keyword minus namespace)
87
+ def shortcut_keyword
88
+ self.name.underscore.gsub('::', '__').split('__').last
89
+ end
90
+
91
+ def add_custom_widget_namespaces_for(klass)
92
+ Glimmer::UI::CustomWidget.namespaces_for_class(klass).drop(1).each do |namespace|
93
+ Glimmer::UI::CustomWidget.custom_widget_namespaces << namespace
94
+ end
95
+ end
96
+
97
+ def namespaces_for_class(m)
98
+ return [m] if m.name.nil?
99
+ namespace_constants = m.name.split(/::/).map(&:to_sym)
100
+ namespace_constants.reduce([Object]) do |output, namespace_constant|
101
+ output += [output.last.const_get(namespace_constant)]
102
+ end[1..-1].uniq.reverse
103
+ end
104
+
105
+ def custom_widget_namespaces
106
+ @custom_widget_namespaces ||= reset_custom_widget_namespaces
107
+ end
108
+
109
+ def reset_custom_widget_namespaces
110
+ @custom_widget_namespaces = Set[Object, Glimmer::UI]
111
+ end
112
+
113
+ # Allows defining convenience option accessors for an array of option names
114
+ # Example: `options :color1, :color2` defines `#color1` and `#color2`
115
+ # where they return the instance values `options[:color1]` and `options[:color2]`
116
+ # respectively.
117
+ # Can be called multiple times to set more options additively.
118
+ # When passed no arguments, it returns list of all option names captured so far
119
+ def options(*new_options)
120
+ new_options = new_options.compact.map(&:to_s).map(&:to_sym)
121
+ if new_options.empty?
122
+ @options ||= {} # maps options to defaults
123
+ else
124
+ new_options = new_options.reduce({}) {|new_options_hash, new_option| new_options_hash.merge(new_option => nil)}
125
+ @options = options.merge(new_options)
126
+ def_option_attr_accessors(new_options)
127
+ end
128
+ end
129
+
130
+ def option(new_option, default: nil)
131
+ new_option = new_option.to_s.to_sym
132
+ new_options = {new_option => default}
133
+ @options = options.merge(new_options)
134
+ def_option_attr_accessors(new_options)
135
+ end
136
+
137
+ def def_option_attr_accessors(new_options)
138
+ new_options.each do |option, default|
139
+ class_eval <<-end_eval, __FILE__, __LINE__
140
+ def #{option}
141
+ options[:#{option}]
142
+ end
143
+ def #{option}=(option_value)
144
+ self.options[:#{option}] = option_value
145
+ end
146
+ end_eval
147
+ end
148
+ end
149
+
150
+ def before_body(&block)
151
+ @before_body_block = block
152
+ end
153
+
154
+ def body(&block)
155
+ @body_block = block
156
+ end
157
+
158
+ def after_body(&block)
159
+ @after_body_block = block
160
+ end
161
+ end
162
+
163
+ attr_reader :body_root, :swt_widget, :parent, :parent_proxy, :swt_style, :options
164
+
165
+ def initialize(parent, *swt_constants, options, &content)
166
+ @parent_proxy = @parent = parent
167
+ @parent_proxy = @parent&.get_data('proxy') if @parent.respond_to?(:get_data) && @parent.get_data('proxy')
168
+ @swt_style = SWT::SWTProxy[*swt_constants]
169
+ options ||= {}
170
+ @options = self.class.options.merge(options)
171
+ @content = Util::ProcTracker.new(content) if content
172
+ execute_hook('before_body')
173
+ body_block = self.class.instance_variable_get("@body_block")
174
+ raise Glimmer::Error, 'Invalid custom widget for having no body! Please define body block!' if body_block.nil?
175
+ @body_root = instance_exec(&body_block)
176
+ raise Glimmer::Error, 'Invalid custom widget for having an empty body! Please fill body block!' if @body_root.nil?
177
+ @swt_widget = @body_root.swt_widget
178
+ @swt_widget.set_data('custom_widget', self)
179
+ execute_hook('after_body')
180
+ end
181
+
182
+ # Subclasses may override to perform post initialization work on an added child
183
+ def post_initialize_child(child)
184
+ # No Op by default
185
+ end
186
+
187
+ def can_handle_observation_request?(observation_request)
188
+ observation_request = observation_request.to_s
189
+ result = false
190
+ if observation_request.start_with?('on_updated_')
191
+ property = observation_request.sub(/^on_updated_/, '')
192
+ result = can_add_observer?(property)
193
+ end
194
+ result || body_root&.can_handle_observation_request?(observation_request)
195
+ end
196
+
197
+ def handle_observation_request(observation_request, &block)
198
+ observation_request = observation_request.to_s
199
+ if observation_request.start_with?('on_updated_')
200
+ property = observation_request.sub(/^on_updated_/, '') # TODO look into eliminating duplication from above
201
+ add_observer(DataBinding::Observer.proc(&block), property) if can_add_observer?(property)
202
+ else
203
+ body_root.handle_observation_request(observation_request, &block)
204
+ end
205
+ end
206
+
207
+ def can_add_observer?(attribute_name)
208
+ has_instance_method?(attribute_name) || has_instance_method?("#{attribute_name}?") || @body_root.can_add_observer?(attribute_name)
209
+ end
210
+
211
+ def add_observer(observer, attribute_name)
212
+ if has_instance_method?(attribute_name)
213
+ super
214
+ else
215
+ @body_root.add_observer(observer, attribute_name)
216
+ end
217
+ end
218
+
219
+ def has_attribute?(attribute_name, *args)
220
+ has_instance_method?(attribute_setter(attribute_name)) ||
221
+ @body_root.has_attribute?(attribute_name, *args)
222
+ end
223
+
224
+ def set_attribute(attribute_name, *args)
225
+ if has_instance_method?(attribute_setter(attribute_name))
226
+ send(attribute_setter(attribute_name), *args)
227
+ else
228
+ @body_root.set_attribute(attribute_name, *args)
229
+ end
230
+ end
231
+
232
+ # This method ensures it has an instance method not coming from Glimmer DSL
233
+ def has_instance_method?(method_name)
234
+ respond_to?(method_name) and
235
+ !swt_widget&.respond_to?(method_name) and
236
+ (method(method_name) rescue nil) and
237
+ !method(method_name)&.source_location&.first&.include?('glimmer/dsl/engine.rb') and
238
+ !method(method_name)&.source_location&.first&.include?('glimmer/swt/widget_proxy.rb')
239
+ end
240
+
241
+ def get_attribute(attribute_name)
242
+ if has_instance_method?(attribute_name)
243
+ send(attribute_name)
244
+ else
245
+ @body_root.get_attribute(attribute_name)
246
+ end
247
+ end
248
+
249
+ def attribute_setter(attribute_name)
250
+ "#{attribute_name}="
251
+ end
252
+
253
+ def disposed?
254
+ swt_widget.isDisposed
255
+ end
256
+
257
+ def has_style?(style)
258
+ (swt_style & SWT::SWTProxy[style]) == SWT::SWTProxy[style]
259
+ end
260
+
261
+ def pack(*args)
262
+ body_root.pack(*args)
263
+ end
264
+
265
+ # TODO see if it is worth it to eliminate duplication of async_exec/sync_exec
266
+ # delegation to DisplayProxy, via a module
267
+
268
+ def async_exec(&block)
269
+ SWT::DisplayProxy.instance.async_exec(&block)
270
+ end
271
+
272
+ def sync_exec(&block)
273
+ SWT::DisplayProxy.instance.sync_exec(&block)
274
+ end
275
+
276
+ # Returns content block if used as an attribute reader (no args)
277
+ # Otherwise, if a block is passed, it adds it as content to this custom widget
278
+ def content(&block)
279
+ if block_given?
280
+ body_root.content(&block)
281
+ else
282
+ @content
283
+ end
284
+ end
285
+
286
+ def method_missing(method, *args, &block)
287
+ # TODO Consider supporting a glimmer error silencing option for methods defined here
288
+ # but fail the glimmer DSL for the right reason to avoid seeing noise in the log output
289
+ if can_handle_observation_request?(method)
290
+ handle_observation_request(method, &block)
291
+ else
292
+ body_root.send(method, *args, &block)
293
+ end
294
+ end
295
+
296
+ alias local_respond_to? respond_to?
297
+ def respond_to?(method, *args, &block)
298
+ super or
299
+ can_handle_observation_request?(method) or
300
+ body_root.respond_to?(method, *args, &block)
301
+ end
302
+
303
+ private
304
+
305
+ def execute_hook(hook_name)
306
+ hook_block = self.class.instance_variable_get("@#{hook_name}_block")
307
+ return if hook_block.nil?
308
+ temp_method_name = "#{hook_name}_block_#{hook_block.hash.abs}_#{(Time.now.to_f * 1_000_000).to_i}"
309
+ singleton_class.define_method(temp_method_name, &hook_block)
310
+ send(temp_method_name)
311
+ singleton_class.send(:remove_method, temp_method_name)
312
+ end
313
+ end
314
+ end
315
+ end