glimmer-dsl-opal 0.7.1 → 0.8.0

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 (56) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +45 -0
  3. data/README.md +453 -48
  4. data/VERSION +1 -1
  5. data/lib/display.rb +31 -0
  6. data/lib/glimmer-dsl-opal.rb +15 -36
  7. data/lib/glimmer-dsl-opal/ext/class.rb +10 -0
  8. data/lib/glimmer-dsl-opal/ext/file.rb +29 -0
  9. data/lib/glimmer-dsl-opal/ext/struct.rb +37 -0
  10. data/lib/glimmer-dsl-opal/samples/elaborate/contact_manager.rb +50 -23
  11. data/lib/glimmer-dsl-opal/samples/elaborate/login.rb +22 -5
  12. data/lib/glimmer-dsl-opal/samples/hello/hello_browser.rb +24 -1
  13. data/lib/glimmer-dsl-opal/samples/hello/hello_button.rb +46 -0
  14. data/lib/glimmer-dsl-opal/samples/hello/hello_computed.rb +27 -0
  15. data/lib/glimmer-dsl-opal/samples/hello/hello_custom_shell.rb +7 -7
  16. data/lib/glimmer-dsl-opal/samples/hello/hello_list_multi_selection.rb +62 -32
  17. data/lib/glimmer-dsl-opal/samples/hello/hello_list_single_selection.rb +47 -22
  18. data/lib/glimmer-dsl-opal/samples/hello/hello_message_box.rb +37 -0
  19. data/lib/glimmer-dsl-opal/samples/hello/hello_pop_up_context_menu.rb +84 -0
  20. data/lib/glimmer-dsl-opal/samples/hello/hello_table.rb +2 -2
  21. data/lib/glimmer/data_binding/observable_element.rb +1 -1
  22. data/lib/glimmer/data_binding/table_items_binding.rb +3 -3
  23. data/lib/glimmer/dsl/opal/custom_widget_expression.rb +6 -0
  24. data/lib/glimmer/dsl/opal/dsl.rb +2 -0
  25. data/lib/glimmer/dsl/opal/menu_bar_expression.rb +54 -0
  26. data/lib/glimmer/dsl/opal/menu_expression.rb +61 -0
  27. data/lib/glimmer/dsl/opal/widget_expression.rb +3 -2
  28. data/lib/glimmer/dsl/opal/widget_listener_expression.rb +2 -2
  29. data/lib/glimmer/swt/combo_proxy.rb +40 -1
  30. data/lib/glimmer/swt/control_editor.rb +2 -1
  31. data/lib/glimmer/swt/custom/checkbox_group.rb +2 -2
  32. data/lib/glimmer/swt/custom/radio_group.rb +2 -2
  33. data/lib/glimmer/swt/date_time_proxy.rb +66 -1
  34. data/lib/glimmer/swt/event_listener_proxy.rb +14 -4
  35. data/lib/glimmer/swt/font_proxy.rb +4 -4
  36. data/lib/glimmer/swt/grid_layout_proxy.rb +21 -12
  37. data/lib/glimmer/swt/label_proxy.rb +17 -6
  38. data/lib/glimmer/swt/layout_data_proxy.rb +10 -7
  39. data/lib/glimmer/swt/list_proxy.rb +33 -0
  40. data/lib/glimmer/swt/menu_item_proxy.rb +87 -0
  41. data/lib/glimmer/swt/menu_proxy.rb +162 -0
  42. data/lib/glimmer/swt/message_box_proxy.rb +53 -67
  43. data/lib/glimmer/swt/property_owner.rb +2 -0
  44. data/lib/glimmer/swt/radio_proxy.rb +1 -1
  45. data/lib/glimmer/swt/shell_proxy.rb +32 -187
  46. data/lib/glimmer/swt/tab_folder_proxy.rb +43 -0
  47. data/lib/glimmer/swt/table_column_proxy.rb +4 -3
  48. data/lib/glimmer/swt/table_editor.rb +2 -2
  49. data/lib/glimmer/swt/table_item_proxy.rb +15 -5
  50. data/lib/glimmer/swt/table_proxy.rb +34 -12
  51. data/lib/glimmer/swt/text_proxy.rb +1 -1
  52. data/lib/glimmer/swt/widget_proxy.rb +335 -38
  53. data/lib/glimmer/ui/custom_shell.rb +9 -7
  54. data/lib/glimmer/ui/custom_widget.rb +3 -3
  55. data/lib/os.rb +36 -0
  56. metadata +36 -3
@@ -35,7 +35,7 @@ module Glimmer
35
35
  model_cells = new_model_collection.to_a.map {|m| @table.cells_for(m)}
36
36
  return if table_cells == model_cells
37
37
  if new_model_collection and new_model_collection.is_a?(Array)
38
- # @table_items_observer_registration&.unobserve # TODO re-enable
38
+ @table_items_observer_registration&.unobserve
39
39
  @table_items_observer_registration = observe(new_model_collection, @column_properties)
40
40
  add_dependent(@table_observer_registration => @table_items_observer_registration)
41
41
  @model_collection = new_model_collection
@@ -62,8 +62,7 @@ module Glimmer
62
62
  table_item.set_data(model)
63
63
  table_item.id = old_item_ids_per_model[model.hash] if old_item_ids_per_model[model.hash]
64
64
  end
65
- selected_table_items = parent.search {|item| selected_table_item_models.include?(item.get_data) }
66
- parent.selection = selected_table_items
65
+ parent.selection = parent.search {|item| selected_table_item_models.include?(item.get_data) }
67
66
  parent.redraw
68
67
  end
69
68
 
@@ -71,6 +70,7 @@ module Glimmer
71
70
  return if model_collection == @last_sorted_model_collection
72
71
  if model_collection == @last_populated_model_collection
73
72
  # Reapply the last table sort. The model collection has just been populated since it diverged from what it was before
73
+ # TODO optimize in the future by sorting elements in DOM directly
74
74
  parent.sort!
75
75
  else
76
76
  # The model collection was sorted by the model, but beyond sorting, it did not change from the last populated model collection.
@@ -42,6 +42,12 @@ module Glimmer
42
42
  end
43
43
 
44
44
  def interpret(parent, keyword, *args, &block)
45
+ begin
46
+ require_path = LocalStorage[keyword]
47
+ require(require_path) if require_path
48
+ rescue => e
49
+ Glimmer::Config.logger.debug e.message
50
+ end
45
51
  custom_widget_class = UI::CustomWidget.for(keyword)
46
52
  # TODO clean code by extracting methods into CustomShell
47
53
  if !Glimmer::UI::CustomShell.requested? && custom_widget_class&.ancestors&.to_a.include?(Glimmer::UI::CustomShell)
@@ -26,6 +26,8 @@ require 'glimmer/dsl/opal/swt_expression'
26
26
  require 'glimmer/dsl/opal/radio_group_selection_data_binding_expression'
27
27
  require 'glimmer/dsl/opal/checkbox_group_selection_data_binding_expression'
28
28
  require 'glimmer/dsl/opal/block_property_expression'
29
+ require 'glimmer/dsl/opal/menu_expression'
30
+ # require 'glimmer/dsl/opal/menu_bar_expression'
29
31
 
30
32
  module Glimmer
31
33
  module DSL
@@ -0,0 +1,54 @@
1
+ # Copyright (c) 2007-2020 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/dsl/static_expression'
24
+ require 'glimmer/dsl/parent_expression'
25
+ require 'glimmer/swt/menu_proxy'
26
+
27
+ # TODO implement for Opal
28
+
29
+ module Glimmer
30
+ module DSL
31
+ module SWT
32
+ class MenuBarExpression < StaticExpression
33
+ include ParentExpression
34
+
35
+ def can_interpret?(parent, keyword, *args, &block)
36
+ initial_condition = (keyword == 'menu_bar')
37
+ if initial_condition
38
+ if parent.swt_widget.is_a?(Shell)
39
+ return true
40
+ else
41
+ raise Glimmer::Error, "menu_bar may only be nested under a shell!"
42
+ end
43
+ end
44
+ false
45
+ end
46
+
47
+ def interpret(parent, keyword, *args, &block)
48
+ args = args.unshift(:bar)
49
+ Glimmer::SWT::MenuProxy.new(parent, args)
50
+ end
51
+ end
52
+ end
53
+ end
54
+ end
@@ -0,0 +1,61 @@
1
+ # Copyright (c) 2020 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/dsl/static_expression'
24
+ require 'glimmer/dsl/parent_expression'
25
+ require 'glimmer/swt/widget_proxy'
26
+ require 'glimmer/swt/menu_proxy'
27
+
28
+ module Glimmer
29
+ module DSL
30
+ module Opal
31
+ class MenuExpression < StaticExpression
32
+ include ParentExpression
33
+
34
+ def can_interpret?(parent, keyword, *args, &block)
35
+ initial_condition = (keyword == 'menu')
36
+ if initial_condition
37
+ if parent.is_a?(Glimmer::SWT::WidgetProxy)
38
+ return true
39
+ else
40
+ raise Glimmer::Error, "menu may only be nested under a widget (like shell or another menu)!"
41
+ end
42
+ end
43
+ false
44
+ end
45
+
46
+ def interpret(parent, keyword, *args, &block)
47
+ Glimmer::SWT::MenuProxy.new(parent, args)
48
+ end
49
+
50
+ def add_content(parent, &block)
51
+ super(parent, &block)
52
+ parent.post_add_content
53
+ end
54
+
55
+ end
56
+
57
+ end
58
+
59
+ end
60
+
61
+ end
@@ -7,6 +7,7 @@ module Glimmer
7
7
  module Opal
8
8
  class WidgetExpression < Expression
9
9
  include ParentExpression
10
+
10
11
  EXCLUDED_KEYWORDS = %w[shell display]
11
12
 
12
13
  def can_interpret?(parent, keyword, *args, &block)
@@ -18,9 +19,9 @@ module Glimmer
18
19
  def interpret(parent, keyword, *args, &block)
19
20
  Glimmer::SWT::WidgetProxy.for(keyword, parent, args, block)
20
21
  end
21
-
22
+
22
23
  def add_content(parent, &block)
23
- if parent.rendered?
24
+ if parent.rendered? || parent.skip_content_on_render_blocks?
24
25
  super(parent, &block)
25
26
  parent.post_add_content
26
27
  else
@@ -20,10 +20,10 @@ module Glimmer
20
20
  # Glimmer::Config.logger.debug {"can add listener? #{result}"}
21
21
  # raise Glimmer::Error, "Invalid listener keyword: #{keyword}" unless result
22
22
  true
23
- end
23
+ end
24
24
 
25
25
  def interpret(parent, keyword, *args, &block)
26
- parent.handle_observation_request(keyword, &block)
26
+ parent.handle_observation_request(keyword, block)
27
27
  end
28
28
  end
29
29
  end
@@ -45,7 +45,46 @@ module Glimmer
45
45
  event_listener.call(event)
46
46
  }
47
47
  }
48
- }
48
+ },
49
+ 'on_key_pressed' => {
50
+ event: 'keydown',
51
+ event_handler: -> (event_listener) {
52
+ -> (event) {
53
+ @last_key_pressed_event = event
54
+ @text = event.target.value
55
+ # TODO generalize this solution to all widgets that support key presses
56
+ # TODO support event.location once DOM3 is supported by opal-jquery
57
+ event.define_singleton_method(:keyCode) {event.which}
58
+ event.define_singleton_method(:key_code, &event.method(:keyCode))
59
+ event.define_singleton_method(:character) {event.which.chr}
60
+ event.define_singleton_method(:stateMask) do
61
+ state_mask = 0
62
+ state_mask |= SWTProxy[:alt] if event.alt_key
63
+ state_mask |= SWTProxy[:ctrl] if event.ctrl_key
64
+ state_mask |= SWTProxy[:shift] if event.shift_key
65
+ state_mask |= SWTProxy[:command] if event.meta_key
66
+ state_mask
67
+ end
68
+ event.define_singleton_method(:state_mask, &event.method(:stateMask))
69
+ doit = true
70
+ event.define_singleton_method(:doit=) do |value|
71
+ doit = value
72
+ end
73
+ event.define_singleton_method(:doit) { doit }
74
+ event_listener.call(event)
75
+
76
+ # TODO Fix doit false, it's not stopping input
77
+ unless doit
78
+ event.prevent
79
+ event.prevent_default
80
+ event.stop_propagation
81
+ event.stop_immediate_propagation
82
+ end
83
+
84
+ doit
85
+ }
86
+ }
87
+ },
49
88
  }
50
89
  end
51
90
 
@@ -23,8 +23,9 @@ module Glimmer
23
23
  module SWT
24
24
  # Emulates SWT's native org.eclipse.swt.custom.ControlEditor
25
25
  class ControlEditor
26
+ # TODO implement behavior for all these attributes
26
27
  ATTRIBUTES = [:grabHorizontal, :grabVertical, :horizontalAlignment, :verticalAlignment, :minimumWidth, :minimumHeight]
27
- attr_accessor *ATTRIBUTES
28
+ attr_accessor(*ATTRIBUTES)
28
29
  ATTRIBUTES.each do |attribute|
29
30
  alias_method attribute.underscore, attribute
30
31
  alias_method "#{attribute.underscore}=", "#{attribute}="
@@ -78,7 +78,7 @@ module Glimmer
78
78
  checkboxes.first&.can_handle_observation_request?(observation_request) || super(observation_request)
79
79
  end
80
80
 
81
- def handle_observation_request(observation_request, &block)
81
+ def handle_observation_request(observation_request, block)
82
82
  observation_requests << [observation_request, block]
83
83
  delegate_observation_request_to_checkboxes(observation_request, &block)
84
84
  super
@@ -88,7 +88,7 @@ module Glimmer
88
88
  if observation_request != 'on_widget_disposed'
89
89
  checkboxes.count.times do |index|
90
90
  checkbox = checkboxes[index]
91
- checkbox.handle_observation_request(observation_request, &block) if checkbox.can_handle_observation_request?(observation_request)
91
+ checkbox.handle_observation_request(observation_request, block) if checkbox.can_handle_observation_request?(observation_request)
92
92
  end
93
93
  end
94
94
  end
@@ -81,7 +81,7 @@ module Glimmer
81
81
  radios.first&.can_handle_observation_request?(observation_request) || super(observation_request)
82
82
  end
83
83
 
84
- def handle_observation_request(observation_request, &block)
84
+ def handle_observation_request(observation_request, block)
85
85
  observation_requests << [observation_request, block]
86
86
  delegate_observation_request_to_radios(observation_request, &block)
87
87
  super
@@ -91,7 +91,7 @@ module Glimmer
91
91
  if observation_request != 'on_widget_disposed'
92
92
  radios.count.times do |index|
93
93
  radio = radios[index]
94
- radio.handle_observation_request(observation_request, &block) if radio.can_handle_observation_request?(observation_request)
94
+ radio.handle_observation_request(observation_request, block) if radio.can_handle_observation_request?(observation_request)
95
95
  end
96
96
  end
97
97
  end
@@ -31,7 +31,13 @@ module Glimmer
31
31
  showPeriod: true,
32
32
  showLeadingZero: true,
33
33
  showOn: 'both',
34
+ showNowButton: true,
35
+ showCloseButton: true,
34
36
  button: "##{time_button_id}",
37
+ onClose: ->(v) {
38
+ @timepicker_done = true
39
+ dom_element.trigger('change')
40
+ },
35
41
  })
36
42
  else
37
43
  options = {}
@@ -105,9 +111,68 @@ module Glimmer
105
111
  {
106
112
  'on_widget_selected' => [
107
113
  {
108
- event: 'change'
114
+ event: 'change',
115
+ event_handler: -> (event_listener) {
116
+ -> (event) {
117
+ if calendar? || date? || (time? && @timepicker_done)
118
+ @timepicker_done = false if time?
119
+ event_listener.call(event)
120
+ end
121
+ }
122
+ }
109
123
  },
110
124
  ],
125
+ 'on_focus_lost' => [
126
+ {
127
+ event: 'blur',
128
+ event_handler: -> (event_listener) {
129
+ -> (event) {
130
+ # TODO support blur event for date?
131
+ if time? && @timepicker_done
132
+ @timepicker_done = false if time?
133
+ event_listener.call(event)
134
+ end
135
+ }
136
+ }
137
+ },
138
+ ],
139
+ 'on_key_pressed' => {
140
+ event: 'keydown',
141
+ event_handler: -> (event_listener) {
142
+ -> (event) {
143
+ # TODO generalize this solution to all widgets that support key presses
144
+ # TODO support event.location once DOM3 is supported by opal-jquery
145
+ event.define_singleton_method(:keyCode) {event.which}
146
+ event.define_singleton_method(:key_code, &event.method(:keyCode))
147
+ event.define_singleton_method(:character) {event.which.chr}
148
+ event.define_singleton_method(:stateMask) do
149
+ state_mask = 0
150
+ state_mask |= SWTProxy[:alt] if event.alt_key
151
+ state_mask |= SWTProxy[:ctrl] if event.ctrl_key
152
+ state_mask |= SWTProxy[:shift] if event.shift_key
153
+ state_mask |= SWTProxy[:command] if event.meta_key
154
+ state_mask
155
+ end
156
+ event.define_singleton_method(:state_mask, &event.method(:stateMask))
157
+ doit = true
158
+ event.define_singleton_method(:doit=) do |value|
159
+ doit = value
160
+ end
161
+ event.define_singleton_method(:doit) { doit }
162
+ event_listener.call(event)
163
+
164
+ # TODO Fix doit false, it's not stopping input
165
+ unless doit
166
+ event.prevent
167
+ event.prevent_default
168
+ event.stop_propagation
169
+ event.stop_immediate_propagation
170
+ end
171
+
172
+ doit
173
+ }
174
+ }
175
+ },
111
176
  }
112
177
  end
113
178
 
@@ -1,18 +1,28 @@
1
1
  module Glimmer
2
2
  module SWT
3
3
  class EventListenerProxy
4
- attr_reader :element_proxy, :event, :selector, :delegate
4
+ attr_reader :element_proxy, :event, :dom_element, :selector, :listener, :original_event_listener
5
5
 
6
- def initialize(element_proxy:, event:, selector:, delegate:)
6
+ def initialize(element_proxy:, event:, dom_element:, selector:, listener:)
7
7
  @element_proxy = element_proxy
8
8
  @event = event
9
+ @dom_element = dom_element
9
10
  @selector = selector
10
- @delegate = delegate
11
+ @listener = listener
12
+ @original_event_listener = original_event_listener
11
13
  end
12
14
 
15
+ def register
16
+ @dom_element.on(@event, @delegate)
17
+ end
18
+ alias observe register
19
+ alias reregister register
20
+
13
21
  def unregister
14
- $document.off(@delegate)
22
+ @dom_element.off(@event, @delegate)
15
23
  end
24
+ alias unobserve unregister
25
+ alias deregister unregister
16
26
  end
17
27
  end
18
28
  end
@@ -1,5 +1,5 @@
1
1
  # Copyright (c) 2020 Andy Maleh
2
- #
2
+ #
3
3
  # Permission is hereby granted, free of charge, to any person obtaining
4
4
  # a copy of this software and associated documentation files (the
5
5
  # "Software"), to deal in the Software without restriction, including
@@ -7,10 +7,10 @@
7
7
  # distribute, sublicense, and/or sell copies of the Software, and to
8
8
  # permit persons to whom the Software is furnished to do so, subject to
9
9
  # the following conditions:
10
- #
10
+ #
11
11
  # The above copyright notice and this permission notice shall be
12
12
  # included in all copies or substantial portions of the Software.
13
- #
13
+ #
14
14
  # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
15
  # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
16
  # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
@@ -68,7 +68,7 @@ module Glimmer
68
68
  private
69
69
 
70
70
  def detect_invalid_font_property(font_properties)
71
- [font_properties[:style]].flatten.select do |style|
71
+ font_properties[:style].to_collection(false).select do |style|
72
72
  style.is_a?(Symbol) || style.is_a?(String)
73
73
  end.each do |style|
74
74
  raise Error, style.to_s + ERROR_INVALID_FONT_STYLE if !FONT_STYLES.include?(style.to_sym)
@@ -7,18 +7,20 @@ module Glimmer
7
7
  .grid-layout {
8
8
  display: grid;
9
9
  grid-template-rows: min-content;
10
- justify-content: start;
11
10
  place-content: start;
12
- align-items: stretch;
11
+ align-items: stretch;
13
12
  }
14
13
  CSS
14
+
15
15
  attr_reader :num_columns, :make_columns_equal_width, :horizontal_spacing, :vertical_spacing, :margin_width, :margin_height
16
16
 
17
17
  def initialize(parent, args)
18
18
  super(parent, args)
19
19
  self.horizontal_spacing = 10
20
20
  self.vertical_spacing = 10
21
- self.num_columns = @args.first || 1
21
+ self.margin_width = 15
22
+ self.margin_height = 15
23
+ self.num_columns = @args.first || 1
22
24
  reapply
23
25
  end
24
26
 
@@ -30,7 +32,7 @@ module Glimmer
30
32
  end
31
33
 
32
34
  def make_columns_equal_width=(equal_width)
33
- @make_columns_equal_width = equal_width
35
+ @make_columns_equal_width = equal_width
34
36
  # @parent.add_css_class('make_columns_equal_width') if @make_columns_equal_width
35
37
  reapply
36
38
  end
@@ -50,14 +52,18 @@ module Glimmer
50
52
  def margin_width=(pixels)
51
53
  @margin_width = pixels
52
54
  # Using padding for width since margin-right isn't getting respected with width 100%
53
- @parent.dom_element.css('padding-left', @margin_width)
54
- @parent.dom_element.css('padding-right', @margin_width)
55
+ effective_margin_width = @margin_width
56
+ effective_margin_width += 6 if @parent.is_a?(GroupProxy)
57
+ @parent.dom_element.css('padding-left', effective_margin_width)
58
+ @parent.dom_element.css('padding-right', effective_margin_width)
55
59
  end
56
60
 
57
61
  def margin_height=(pixels)
58
62
  @margin_height = pixels
59
- @parent.dom_element.css('padding-top', @margin_height)
60
- @parent.dom_element.css('padding-bottom', @margin_height)
63
+ effective_margin_height = @margin_height
64
+ effective_margin_height += 9 if @parent.is_a?(GroupProxy)
65
+ @parent.dom_element.css('padding-top', effective_margin_height)
66
+ @parent.dom_element.css('padding-bottom', effective_margin_height)
61
67
  end
62
68
 
63
69
  def reapply
@@ -68,15 +74,18 @@ module Glimmer
68
74
  grid-column-gap: #{@horizontal_spacing}px;
69
75
  CSS
70
76
  if @parent.css_classes.include?('grid-layout')
71
- layout_css.split(";").map(&:strip).map {|l| l.split(':').map(&:strip)}.each do |key, value|
77
+ layout_css.split(";").map(&:strip).map {|l| l.split(':').map(&:strip)}.each do |key, value|
72
78
  @parent.dom_element.css(key, value) unless key.nil?
73
79
  end
80
+ if @parent.is_a?(GroupProxy)
81
+ @parent.dom_element.find('legend').css('grid-column-start', "span #{@num_columns.to_i}")
82
+ end
74
83
  else
75
- layout_css.split(";").map(&:strip).map {|l| l.split(':').map(&:strip)}.each do |key, value|
84
+ layout_css.split(";").map(&:strip).map {|l| l.split(':').map(&:strip)}.each do |key, value|
76
85
  @parent.dom_element.css(key, 'initial') unless key.nil?
77
- end
86
+ end
78
87
  end
79
- end
88
+ end
80
89
  end
81
90
  end
82
91
  end