glimmer-dsl-web 0.2.0 → 0.2.3

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 429afe58eb54b98a9aa229faeecbe09ceefb82ebae7e240ef3dd68ce63110164
4
- data.tar.gz: '03469f3f04757d5b95aab910b1469047e467481aeaf6169ed69a33a8ab900b6f'
3
+ metadata.gz: c418754ebdb0e40b75d62576c678942a321ee5d8b0dbb0f9d3bafbd4d0622cbb
4
+ data.tar.gz: a4e98d44358353332f166e3191bad00393532ac13dcbe0af214db53f91921b11
5
5
  SHA512:
6
- metadata.gz: 93b2b65475f2609476997bb7139e2078f8851b88e881d123af4298df2f293a8a2537eecf3f697d734d09e0a8d5ad87771b32899873b9161b74f3e5387e86df00
7
- data.tar.gz: 7a37b8b255486545016a982522a52ee3cf00c8238557d396a874e38da8d5689fda9e8a8b021cb698f380f641e68d341b9b9e2ea10e035692192e534693cfac21
6
+ metadata.gz: d756e84e43ba8cce780251ab96c40234e2b5a9d53724b2be329d5cdef53d02d33a2ab629035a8c742d7ff75b888f76d2f8e92fcb5354a012bbdd061024c0db66
7
+ data.tar.gz: 221e1d4472510c02d403439d6fde7c33ace135f10337f2a258c1641d3685d41b8c5d049f2a96e99cbdad1dd2587c1728aa03bce04cb844705f083e2664bb49b0
data/CHANGELOG.md CHANGED
@@ -1,5 +1,24 @@
1
1
  # Change Log
2
2
 
3
+ ## 0.2.3
4
+
5
+ - Ensure that setting `element.class_name=classnameval` does not override its Glimmer built-in CSS classes
6
+ - Alias `Kernel#p` as `Kernel#pi` (puts inspect) to allow using `pi` in place of `p` for printing inspected objects given that `p` is used up by the HTML DSL.
7
+ - Remove `element` from dynamic part of Glimmer DSL to allow it to fail faster if an invalid HTML element name was used
8
+ - Fix use of `span` keyword under `p`, fixing Hello, Paragraph! sample.
9
+
10
+ ## 0.2.2
11
+
12
+ - Fix bug in content data-binding that was caused by recent performance optimizations (+ fix Hello, Content Data-Binding! sample)
13
+ - Fix bug in rendering formatting elements that was caused by recent performance optimizations (+ fix Hello, Paragraph! sample)
14
+
15
+ ## 0.2.1
16
+
17
+ - Optimize performance (~800% faster):
18
+ - Optimize performance (~20% faster than 0.2.0) by not using glimmer-dsl-xml for rendering dom elements in the frontend as it is more suitable for backend rendering and we already have a separate dsl for the frontend
19
+ - Optimize performance (~287% faster than previous bullet point) by adding Glimmer shortcut methods for all HTML element DSL keywords
20
+ - Optimize performance (~235% faster than previous bullet point) by not selecting ElementProxy subclass dynamically for each element, yet always using ElementProxy
21
+
3
22
  ## 0.2.0
4
23
 
5
24
  - Support `style` element content as Glimmer DSL for CSS syntax (Ruby Programmable CSS) as an alternative to a CSS `String`
data/README.md CHANGED
@@ -1,13 +1,15 @@
1
- # [<img src="https://raw.githubusercontent.com/AndyObtiva/glimmer/master/images/glimmer-logo-hi-res.png" height=85 />](https://github.com/AndyObtiva/glimmer) Glimmer DSL for Web 0.2.0 (Beta)
1
+ # [<img src="https://raw.githubusercontent.com/AndyObtiva/glimmer/master/images/glimmer-logo-hi-res.png" height=85 />](https://github.com/AndyObtiva/glimmer) Glimmer DSL for Web 0.2.3 (Beta)
2
2
  ## Ruby in the Browser Web GUI Frontend Library
3
3
  [![Gem Version](https://badge.fury.io/rb/glimmer-dsl-web.svg)](http://badge.fury.io/rb/glimmer-dsl-web)
4
4
  [![Join the chat at https://gitter.im/AndyObtiva/glimmer](https://badges.gitter.im/AndyObtiva/glimmer.svg)](https://gitter.im/AndyObtiva/glimmer?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
5
5
 
6
- [Glimmer](https://github.com/AndyObtiva/glimmer) DSL for Web enables building Web GUI frontends using [Ruby in the Browser](https://www.youtube.com/watch?v=4AdcfbI6A4c), as per [Matz's recommendation in his RubyConf 2022 keynote speech to replace JavaScript with Ruby](https://youtu.be/knutsgHTrfQ?t=789). It aims at providing the simplest, most intuitive, most straight-forward, and most productive frontend library in existence. The library follows the Ruby way (with [DSLs](https://martinfowler.com/books/dsl.html) and [TIMTOWTDI](https://en.wiktionary.org/wiki/TMTOWTDI#English)) and the Rails way ([Convention over Configuration](https://rubyonrails.org/doctrine)) while supporting both Unidirectional (One-Way) [Data-Binding](#hello-data-binding) (using `<=`) and Bidirectional (Two-Way) [Data-Binding](#hello-data-binding) (using `<=>`). Dynamic rendering (and re-rendering) of HTML content is also supported via [Content Data-Binding](#hello-content-data-binding). And, modular design is supported with [Glimmer Web Components](#hello-component). You can finally live in pure Rubyland on the Web in both the frontend and backend with [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web)!
6
+ **(Based on Original [Glimmer](https://github.com/AndyObtiva/glimmer) Library Handling World’s Ruby GUI Needs Since 2007. Beware of Imitators!)**
7
+
8
+ [Glimmer](https://github.com/AndyObtiva/glimmer) DSL for Web enables building Web GUI frontends using [Ruby in the Browser](https://www.youtube.com/watch?v=4AdcfbI6A4c), as per [Matz's recommendation in his RubyConf 2022 keynote speech to replace JavaScript with Ruby](https://youtu.be/knutsgHTrfQ?t=789). It aims at providing the simplest, most intuitive, most straight-forward, and most productive frontend library in existence. The library follows the Ruby way (with [DSLs](https://martinfowler.com/books/dsl.html) and [TIMTOWTDI](https://en.wiktionary.org/wiki/TMTOWTDI#English)) and the Rails way ([Convention over Configuration](https://rubyonrails.org/doctrine)) in building Isomorphic Ruby on Rails Applications. It supports both Unidirectional (One-Way) [Data-Binding](#hello-data-binding) (using `<=`) and Bidirectional (Two-Way) [Data-Binding](#hello-data-binding) (using `<=>`). Dynamic rendering (and re-rendering) of HTML content is also supported via [Content Data-Binding](#hello-content-data-binding). And, modular design is supported with [Glimmer Web Components](#hello-component). You can finally live in pure Rubyland on the Web in both the frontend and backend with [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web)!
7
9
 
8
10
  **Hello, World! Sample**
9
11
 
10
- Glimmer GUI code:
12
+ Glimmer GUI code in the frontend:
11
13
 
12
14
  ```ruby
13
15
  require 'glimmer-dsl-web'
@@ -31,11 +33,13 @@ That produces the following under `<body></body>`:
31
33
 
32
34
  ![setup is working](/images/glimmer-dsl-web-setup-example-working.png)
33
35
 
36
+ You can also mount the `div` elsewhere by passing the `parent: parent_css_selector` option (e.g. `div(parent: 'div#app-container') { 'Hello, World!' }`).
37
+
34
38
  **Hello, Button!**
35
39
 
36
- Event listeners can be setup on any element using the same event names used in HTML (e.g. `onclick`) while passing in a standard Ruby block to handle behavior. `$$` gives access to `window` to invoke functions like `alert`.
40
+ Event listeners can be setup on any element using the same event names used in HTML (e.g. `onclick`) while passing in a standard Ruby block to handle behavior. `$$` gives access to JS global scope from Ruby to invoke functions like `alert`.
37
41
 
38
- Glimmer GUI code:
42
+ Glimmer GUI code in the frontend:
39
43
 
40
44
  ```ruby
41
45
  require 'glimmer-dsl-web'
@@ -69,7 +73,7 @@ Screenshot:
69
73
 
70
74
  [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web) gives access to all Web Browser built-in features like HTML form validations, input focus, events, and element functions from a very terse and productive Ruby GUI DSL.
71
75
 
72
- Glimmer GUI code:
76
+ Glimmer GUI code in the frontend:
73
77
 
74
78
  ```ruby
75
79
  require 'glimmer-dsl-web'
@@ -221,95 +225,11 @@ Screenshot:
221
225
 
222
226
  ![Hello, Form!](/images/glimmer-dsl-web-samples-hello-hello-form.gif)
223
227
 
224
- **Hello, Observer!**
225
-
226
- [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web) provides the `observe(model, attribute) { ... }` keyword to employ the [Observer Design Pattern](https://en.wikipedia.org/wiki/Observer_pattern) as per [MVC](https://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93controller) (Model View Controller), enabling Views to observe Models and update themselves in response to changes. If the `observe` keyword is used from inside a [Component](#hello-component) (covered later), when the Component is removed or its top-level element is removed, the observer is automatically cleaned up. The need for such explicit observers is significantly diminished by the availablility of the more advanced Unidirectional [Data-Binding](#hello-data-binding) Support and Bidirectional [Data-Binding](#hello-data-binding) Support (covered later).
227
-
228
- Glimmer GUI code:
229
-
230
- ```ruby
231
- require 'glimmer-dsl-web'
232
-
233
- class NumberHolder
234
- attr_accessor :number
235
-
236
- def initialize
237
- self.number = 50
238
- end
239
- end
240
-
241
- class HelloObserver
242
- include Glimmer::Web::Component
243
-
244
- before_render do
245
- @number_holder = NumberHolder.new
246
- end
247
-
248
- after_render do
249
- @number_input.value = @number_holder.number
250
- @range_input.value = @number_holder.number
251
- # Observe Model attribute @number_holder.number for changes and update View
252
- # Observer is automatically cleaned up if remove method is called on rendered HelloObserver
253
- # or its top-level element
254
- observe(@number_holder, :number) do
255
- number_string = @number_holder.number.to_s
256
- @number_input.value = number_string unless @number_input.value == number_string
257
- @range_input.value = number_string unless @range_input.value == number_string
258
- end
259
- # Bidirectional Data-Binding does the same thing automatically
260
- # Just disable the observe block above as well as the oninput listeners below
261
- # and enable the `value <=> [@number_holder, :number]` lines to try the data-binding version
262
- # Learn more about Bidirectional and Unidirectional Data-Binding in hello_data_binding.rb
263
- end
264
-
265
- markup {
266
- div {
267
- div {
268
- @number_input = input(type: 'number', min: 0, max: 100) {
269
- # oninput listener updates Model attribute @number_holder.number
270
- oninput do
271
- @number_holder.number = @number_input.value.to_i
272
- end
273
-
274
- # Bidirectional Data-Binding simplifies the implementation significantly
275
- # by enabling the following line and disabling oninput listeners as well
276
- # as the after_body observe block observer
277
- # Learn more about Bidirectional and Unidirectional Data-Binding in hello_data_binding.rb
278
- # value <=> [@number_holder, :number]
279
- }
280
- }
281
- div {
282
- @range_input = input(type: 'range', min: 0, max: 100) {
283
- # oninput listener updates Model attribute @number_holder.number
284
- oninput do
285
- @number_holder.number = @range_input.value.to_i
286
- end
287
-
288
- # Bidirectional Data-Binding simplifies the implementation significantly
289
- # by enabling the following line and disabling oninput listeners as well
290
- # as the after_body observe block observer
291
- # Learn more about Bidirectional and Unidirectional Data-Binding in hello_data_binding.rb
292
- # value <=> [@number_holder, :number]
293
- }
294
- }
295
- }
296
- }
297
- end
298
-
299
- Document.ready? do
300
- HelloObserver.render
301
- end
302
- ```
303
-
304
- Screenshot:
305
-
306
- ![Hello, Observer!](/images/glimmer-dsl-web-samples-hello-hello-observer.gif)
307
-
308
228
  **Hello, Data-Binding!**
309
229
 
310
230
  [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web) intuitively supports both Unidirectional (One-Way) Data-Binding via the `<=` operator and Bidirectional (Two-Way) Data-Binding via the `<=>` operator, incredibly simplifying how to sync View properties with Model attributes with the simplest code to reason about.
311
231
 
312
- Glimmer GUI code:
232
+ Glimmer GUI code in the frontend:
313
233
 
314
234
  ```ruby
315
235
  require 'glimmer-dsl-web'
@@ -428,7 +348,7 @@ If you need to regenerate HTML element content dynamically, you can use Content
428
348
  rebuild HTML elements based on changes in a Model attribute that provides the source data.
429
349
  In this example, we generate multiple address forms based on the number of addresses the user has.
430
350
 
431
- Glimmer GUI code:
351
+ Glimmer GUI code in the frontend:
432
352
 
433
353
  ```ruby
434
354
  require 'glimmer-dsl-web'
@@ -560,12 +480,14 @@ Screenshot:
560
480
  You can define Glimmer web components (View components) to reuse visual concepts to your heart's content,
561
481
  by simply defining a class with `include Glimmer::Web::Component` and encasing the reusable markup inside
562
482
  a `markup {...}` block. Glimmer web components automatically extend the Glimmer GUI DSL with new keywords
563
- that match the underscored versions of the component class names (e.g. a `OrderSummary` class yields
564
- the `order_summary` keyword for reusing that component within the Glimmer GUI DSL).
565
- Below, we define an `AddressForm` component that generates a `address_form` keyword, and then we
483
+ that match the underscored versions of the component class names (e.g. an `OrderSummary` class yields
484
+ the `order_summary` keyword for reusing that component within the Glimmer GUI DSL).
485
+ You may insert a Glimmer component anywhere into a Rails View using `glimmer_component(component_path, *args)`
486
+ Rails helper (more about it in [Hello, glimmer_component Rails Helper!](#hello-glimmer_component-rails-helper)).
487
+ Below, we define an `AddressForm` component that generates an `address_form` keyword, and then we
566
488
  reuse it twice inside an `AddressPage` component displaying a Shipping Address and a Billing Address.
567
489
 
568
- Glimmer GUI code:
490
+ Glimmer GUI code in the frontend:
569
491
 
570
492
  ```ruby
571
493
  require 'glimmer-dsl-web'
@@ -814,7 +736,7 @@ Rails View code:
814
736
  </div>
815
737
  ```
816
738
 
817
- Glimmer GUI code:
739
+ Glimmer GUI code in the frontend (`app/assets/opal/address_form.rb`):
818
740
 
819
741
  ```ruby
820
742
  require 'glimmer-dsl-web'
@@ -979,7 +901,7 @@ Screenshot:
979
901
 
980
902
  **Hello, Paragraph!**
981
903
 
982
- To facilitate building formatted textual paragraphs in Ruby, the Glimmer GUI DSL is advanced enough to behave differently when using HTML formatting elements: `<br>`, `<strong>`, `<em>`, `<br>`, `<i>`, `<sub>`, `<sup>`, `<del>`, `<ins>`, `<small>`, `<mark>`
904
+ To facilitate building formatted textual paragraphs in Ruby, thanks to [Glimmer](https://github.com/AndyObtiva/glimmer#dsl-engine), the most advanced DSL engine in Ruby, the Glimmer GUI DSL is advanced enough to intelligently behave differently under different situations, like when using HTML formatting elements: `<br>`, `<strong>`, `<em>`, `<br>`, `<i>`, `<sub>`, `<sup>`, `<del>`, `<ins>`, `<small>`, `<mark>`
983
905
 
984
906
  Instead of returning Ruby objects that are nested as children within their parent, the Glimmer GUI DSL returns `String` objects directly that can be concatenated to or embedded within other `String` objects via interpolation.
985
907
 
@@ -1011,7 +933,7 @@ it returns a `String` to enable code like this:
1011
933
 
1012
934
  In any case, below is a full example leveraging the Glimmer GUI DSL alternative approach when utilizing formatting elements underneath a paragraph.
1013
935
 
1014
- Glimmer GUI code:
936
+ Glimmer GUI code in the frontend:
1015
937
 
1016
938
  ```ruby
1017
939
  require 'glimmer-dsl-web'
@@ -1068,6 +990,92 @@ Screenshot:
1068
990
 
1069
991
  ![Hello, Paragraph!](/images/glimmer-dsl-web-samples-hello-hello-paragraph.png)
1070
992
 
993
+ **Hello, Observer!**
994
+
995
+ [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web) provides the `observe(model, attribute) { ... }` keyword to employ the [Observer Design Pattern](https://en.wikipedia.org/wiki/Observer_pattern) as per [MVC](https://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93controller) (Model View Controller), enabling Views to observe Models and update themselves in response to changes. If the `observe` keyword is used from inside a [Component](#hello-component), when the Component is removed or its top-level element is removed, the observer is automatically cleaned up. The need for such explicit observers is significantly diminished by the availablility of the more advanced Unidirectional [Data-Binding](#hello-data-binding) Support and Bidirectional [Data-Binding](#hello-data-binding) Support.
996
+
997
+ Glimmer GUI code in the frontend:
998
+
999
+ ```ruby
1000
+ require 'glimmer-dsl-web'
1001
+
1002
+ class NumberHolder
1003
+ attr_accessor :number
1004
+
1005
+ def initialize
1006
+ self.number = 50
1007
+ end
1008
+ end
1009
+
1010
+ class HelloObserver
1011
+ include Glimmer::Web::Component
1012
+
1013
+ before_render do
1014
+ @number_holder = NumberHolder.new
1015
+ end
1016
+
1017
+ after_render do
1018
+ @number_input.value = @number_holder.number
1019
+ @range_input.value = @number_holder.number
1020
+ # Observe Model attribute @number_holder.number for changes and update View
1021
+ # Observer is automatically cleaned up if remove method is called on rendered HelloObserver
1022
+ # or its top-level element
1023
+ observe(@number_holder, :number) do
1024
+ number_string = @number_holder.number.to_s
1025
+ @number_input.value = number_string unless @number_input.value == number_string
1026
+ @range_input.value = number_string unless @range_input.value == number_string
1027
+ end
1028
+ # Bidirectional Data-Binding does the same thing automatically
1029
+ # Just disable the observe block above as well as the oninput listeners below
1030
+ # and enable the `value <=> [@number_holder, :number]` lines to try the data-binding version
1031
+ # Learn more about Bidirectional and Unidirectional Data-Binding in hello_data_binding.rb
1032
+ end
1033
+
1034
+ markup {
1035
+ div {
1036
+ div {
1037
+ @number_input = input(type: 'number', min: 0, max: 100) {
1038
+ # oninput listener updates Model attribute @number_holder.number
1039
+ oninput do
1040
+ @number_holder.number = @number_input.value.to_i
1041
+ end
1042
+
1043
+ # Bidirectional Data-Binding simplifies the implementation significantly
1044
+ # by enabling the following line and disabling oninput listeners as well
1045
+ # as the after_body observe block observer
1046
+ # Learn more about Bidirectional and Unidirectional Data-Binding in hello_data_binding.rb
1047
+ # value <=> [@number_holder, :number]
1048
+ }
1049
+ }
1050
+ div {
1051
+ @range_input = input(type: 'range', min: 0, max: 100) {
1052
+ # oninput listener updates Model attribute @number_holder.number
1053
+ oninput do
1054
+ @number_holder.number = @range_input.value.to_i
1055
+ end
1056
+
1057
+ # Bidirectional Data-Binding simplifies the implementation significantly
1058
+ # by enabling the following line and disabling oninput listeners as well
1059
+ # as the after_body observe block observer
1060
+ # Learn more about Bidirectional and Unidirectional Data-Binding in hello_data_binding.rb
1061
+ # value <=> [@number_holder, :number]
1062
+ }
1063
+ }
1064
+ }
1065
+ }
1066
+ end
1067
+
1068
+ Document.ready? do
1069
+ HelloObserver.render
1070
+ end
1071
+ ```
1072
+
1073
+ Screenshot:
1074
+
1075
+ ![Hello, Observer!](/images/glimmer-dsl-web-samples-hello-hello-observer.gif)
1076
+
1077
+ To get started, [Setup](#setup) Ruby gem, read [Usage](#usage) instructions, and check out more [Samples](#samples).
1078
+
1071
1079
  --
1072
1080
 
1073
1081
  NOTE: Glimmer DSL for Web is a Beta project. If you want it developed faster, please [open an issue report](https://github.com/AndyObtiva/glimmer-dsl-web/issues/new). I have completed some GitHub project features much faster before due to [issue reports](https://github.com/AndyObtiva/glimmer-dsl-web/issues) and [pull requests](https://github.com/AndyObtiva/glimmer-dsl-web/pulls). Please help make better by contributing, adopting for small or low risk projects, and providing feedback. It is still a Beta, so the more feedback and issues you report the better.
@@ -1114,14 +1122,12 @@ Learn more about the differences between various [Glimmer](https://github.com/An
1114
1122
  [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web) will begin by supporting [Opal Ruby](https://opalrb.com/) on [Rails](https://rubyonrails.org/). [Opal](https://opalrb.com/) is a lightweight Ruby to JavaScript transpiler that results in small downloadables compared to WASM. In the future, the project might grow to support [Ruby WASM](https://github.com/ruby/ruby.wasm) as an alternative to [Opal Ruby](https://opalrb.com/) that could be switched to with a simple configuration change.
1115
1123
 
1116
1124
  - Ruby 3.0+
1117
- - Rails 6-7: [https://github.com/rails/rails](https://github.com/rails/rails)
1125
+ - Rails 7: [https://github.com/rails/rails](https://github.com/rails/rails)
1118
1126
 
1119
1127
  ## Setup
1120
1128
 
1121
1129
  (NOTE: Keep in mind this is a Beta. If you run into issues, try to go back to a [previous revision](https://rubygems.org/gems/glimmer-dsl-web/versions). Also, there is a slight chance any issues you encounter are fixed in master or some other branch that you could check out instead)
1122
1130
 
1123
- The [glimmer-dsl-web](https://rubygems.org/gems/glimmer-dsl-web) gem is a [Rails Engine](https://guides.rubyonrails.org/engines.html) gem that includes assets.
1124
-
1125
1131
  ### Rails 7
1126
1132
 
1127
1133
  Please follow the following steps to setup.
@@ -1141,7 +1147,7 @@ rails new glimmer_app_server
1141
1147
  Add the following to `Gemfile`:
1142
1148
 
1143
1149
  ```
1144
- gem 'glimmer-dsl-web', '~> 0.2.0'
1150
+ gem 'glimmer-dsl-web', '~> 0.2.3'
1145
1151
  ```
1146
1152
 
1147
1153
  Run:
@@ -1168,9 +1174,11 @@ Opal.append_path Rails.root.join('app', 'assets', 'opal')
1168
1174
  To enable Opal Browser Debugging in Ruby with the [Source Maps](https://opalrb.com/docs/guides/v1.4.1/source_maps.html) feature, edit `config/initializers/opal.rb` and add the following inside the `Rails.application.configure do; end` block at the bottom of it:
1169
1175
 
1170
1176
  ```ruby
1171
- config.assets.debug = true
1177
+ config.assets.debug = true if Rails.env.development?
1172
1178
  ```
1173
1179
 
1180
+ Assuming this is a brand new Rails application and you do not have any Rails resources, you can scaffold the welcome resource just for testing purposes.
1181
+
1174
1182
  Run:
1175
1183
 
1176
1184
  ```
@@ -1233,7 +1241,7 @@ end
1233
1241
 
1234
1242
  Example to confirm setup is working:
1235
1243
 
1236
- Glimmer GUI code:
1244
+ Glimmer GUI code in the frontend:
1237
1245
 
1238
1246
  ```ruby
1239
1247
  require 'glimmer-dsl-web'
@@ -1284,7 +1292,7 @@ HTML:
1284
1292
  ...
1285
1293
  ```
1286
1294
 
1287
- Glimmer GUI code:
1295
+ Glimmer GUI code in the frontend:
1288
1296
 
1289
1297
  ```ruby
1290
1298
  require 'glimmer-dsl-web'
@@ -1337,6 +1345,8 @@ If you run into any issues in setup, refer to the [Sample Glimmer DSL for Web Ra
1337
1345
 
1338
1346
  Otherwise, if you still cannot setup successfully (even with the help of the sample project, or if the sample project stops working), please do not hesitate to report an [Issue request](https://github.com/AndyObtiva/glimmer-dsl-web/issues) or fix and submit a [Pull Request](https://github.com/AndyObtiva/glimmer-dsl-web/pulls).
1339
1347
 
1348
+ Next, read [Usage](#usage) instructions, and check out [Samples](#samples).
1349
+
1340
1350
  ### Rails 6
1341
1351
 
1342
1352
  Please follow the following steps to setup.
@@ -1362,7 +1372,7 @@ Disable the `webpacker` gem line in `Gemfile`:
1362
1372
  Add the following to `Gemfile`:
1363
1373
 
1364
1374
  ```ruby
1365
- gem 'glimmer-dsl-web', '~> 0.2.0'
1375
+ gem 'glimmer-dsl-web', '~> 0.2.3'
1366
1376
  ```
1367
1377
 
1368
1378
  Run:
@@ -1389,9 +1399,11 @@ Opal.append_path Rails.root.join('app', 'assets', 'opal')
1389
1399
  To enable Opal Browser Debugging in Ruby with the [Source Maps](https://opalrb.com/docs/guides/v1.4.1/source_maps.html) feature, edit `config/initializers/opal.rb` and add the following inside the `Rails.application.configure do; end` block at the bottom of it:
1390
1400
 
1391
1401
  ```ruby
1392
- config.assets.debug = true
1402
+ config.assets.debug = true if Rails.env.development?
1393
1403
  ```
1394
1404
 
1405
+ Assuming this is a brand new Rails application and you do not have any Rails resources, you can scaffold the welcome resource just for testing purposes.
1406
+
1395
1407
  Run:
1396
1408
 
1397
1409
  ```
@@ -1453,7 +1465,7 @@ Initial HTML Markup:
1453
1465
  ...
1454
1466
  ```
1455
1467
 
1456
- Glimmer GUI code:
1468
+ Glimmer GUI code in the frontend:
1457
1469
 
1458
1470
  ```ruby
1459
1471
  require 'glimmer-dsl-web'
@@ -1519,9 +1531,11 @@ If you run into any issues in setup, refer to the [Sample Glimmer DSL for Web Ra
1519
1531
 
1520
1532
  Otherwise, if you still cannot setup successfully (even with the help of the sample project, or if the sample project stops working), please do not hesitate to report an [Issue request](https://github.com/AndyObtiva/glimmer-dsl-web/issues) or fix and submit a [Pull Request](https://github.com/AndyObtiva/glimmer-dsl-web/pulls).
1521
1533
 
1534
+ Next, read [Usage](#usage) instructions, and check out [Samples](#samples).
1535
+
1522
1536
  ## Usage
1523
1537
 
1524
- Glimmer DSL for Web offers a GUI DSL for building HTML Web User Interfaces declaratively in Ruby.
1538
+ Glimmer DSL for Web offers a GUI DSL ([Graphical User Interface](https://en.wikipedia.org/wiki/Graphical_user_interface) [Domain Specific Language](https://en.wikipedia.org/wiki/Domain-specific_language)) for building HTML Web User Interfaces declaratively in Ruby.
1525
1539
 
1526
1540
  1- **Keywords (HTML Elements)**
1527
1541
 
@@ -1537,7 +1551,7 @@ Also, if the element has a little bit of text content that can fit in one line,
1537
1551
 
1538
1552
  3- **Content Block (Properties + Listeners + Nested Elements + Text Content)**
1539
1553
 
1540
- Element methods can accept a Ruby content block. It intentionally has a `{...}` style even as a multi-line block to indicate that the code is declarative GUI structure code (intentionally breaking away from Ruby imperative code conventions given this is a declarative GUI DSL, meaning a different language that has its own conventions, embedded within Ruby).
1554
+ Element methods can accept a Ruby content block. It intentionally has a `{...}` style even as a multi-line block to indicate that the code is declarative GUI structure code (intentionally breaking away from Ruby imperative code conventions given this is a declarative GUI DSL ([Domain Specific Language](https://en.wikipedia.org/wiki/Domain-specific_language)), meaning a different language that has its own conventions, embedded within Ruby).
1541
1555
 
1542
1556
  You can nest HTML element properties under an element like:
1543
1557
 
@@ -1591,6 +1605,8 @@ p(class: 'summary') {
1591
1605
 
1592
1606
  You can get/set any element property or invoke any element function by simply calling the lowercase underscored version of their name in Ruby like `input.check_validity`, `input.value`, and `input.id`.
1593
1607
 
1608
+ Next, check out [Samples](#samples).
1609
+
1594
1610
  ## Supported Glimmer DSL Keywords
1595
1611
 
1596
1612
  [All HTML elements](https://developer.mozilla.org/en-US/docs/Web/HTML/Element), following the Ruby method name standard of lowercase and underscored names.
@@ -1613,7 +1629,7 @@ https://github.com/AndyObtiva/sample-glimmer-dsl-web-rails7-app
1613
1629
 
1614
1630
  #### Hello, World!
1615
1631
 
1616
- Glimmer GUI code:
1632
+ Glimmer GUI code in the frontend:
1617
1633
 
1618
1634
  ```ruby
1619
1635
  require 'glimmer-dsl-web'
@@ -1659,11 +1675,13 @@ That produces the following under `<body></body>`:
1659
1675
 
1660
1676
  ![setup is working](/images/glimmer-dsl-web-setup-example-working.png)
1661
1677
 
1678
+ You can also mount the `div` elsewhere by passing the `parent: parent_css_selector` option (e.g. `div(parent: 'div#app-container') { 'Hello, World!' }`).
1679
+
1662
1680
  #### Hello, Button!
1663
1681
 
1664
- Event listeners can be setup on any element using the same event names used in HTML (e.g. `onclick`) while passing in a standard Ruby block to handle behavior. `$$` gives access to `window` to invoke functions like `alert`.
1682
+ Event listeners can be setup on any element using the same event names used in HTML (e.g. `onclick`) while passing in a standard Ruby block to handle behavior. `$$` gives access to JS global scope from Ruby to invoke functions like `alert`.
1665
1683
 
1666
- Glimmer GUI code:
1684
+ Glimmer GUI code in the frontend:
1667
1685
 
1668
1686
  ```ruby
1669
1687
  require 'glimmer-dsl-web'
@@ -1697,7 +1715,7 @@ Screenshot:
1697
1715
 
1698
1716
  [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web) gives access to all Web Browser built-in features like HTML form validations, input focus, events, and element functions from a very terse and productive Ruby GUI DSL.
1699
1717
 
1700
- Glimmer GUI code:
1718
+ Glimmer GUI code in the frontend:
1701
1719
 
1702
1720
  ```ruby
1703
1721
  require 'glimmer-dsl-web'
@@ -1851,9 +1869,9 @@ Screenshot:
1851
1869
 
1852
1870
  #### Hello, Observer!
1853
1871
 
1854
- [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web) provides the `observe(model, attribute) { ... }` keyword to employ the [Observer Design Pattern](https://en.wikipedia.org/wiki/Observer_pattern) as per [MVC](https://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93controller) (Model View Controller), enabling Views to observe Models and update themselves in response to changes. If the `observe` keyword is used from inside a [Component](#hello-component) (covered later), when the Component is removed or its top-level element is removed, the observer is automatically cleaned up. The need for such explicit observers is significantly diminished by the availablility of the more advanced Unidirectional [Data-Binding](#hello-data-binding) Support and Bidirectional [Data-Binding](#hello-data-binding) Support (covered later).
1872
+ [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web) provides the `observe(model, attribute) { ... }` keyword to employ the [Observer Design Pattern](https://en.wikipedia.org/wiki/Observer_pattern) as per [MVC](https://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93controller) (Model View Controller), enabling Views to observe Models and update themselves in response to changes. If the `observe` keyword is used from inside a [Component](#hello-component), when the Component is removed or its top-level element is removed, the observer is automatically cleaned up. The need for such explicit observers is significantly diminished by the availablility of the more advanced Unidirectional [Data-Binding](#hello-data-binding) Support and Bidirectional [Data-Binding](#hello-data-binding) Support.
1855
1873
 
1856
- Glimmer GUI code:
1874
+ Glimmer GUI code in the frontend:
1857
1875
 
1858
1876
  ```ruby
1859
1877
  require 'glimmer-dsl-web'
@@ -1937,7 +1955,7 @@ Screenshot:
1937
1955
 
1938
1956
  [Glimmer DSL for Web](https://rubygems.org/gems/glimmer-dsl-web) intuitively supports both Unidirectional (One-Way) Data-Binding via the `<=` operator and Bidirectional (Two-Way) Data-Binding via the `<=>` operator, incredibly simplifying how to sync View properties with Model attributes with the simplest code to reason about.
1939
1957
 
1940
- Glimmer GUI code:
1958
+ Glimmer GUI code in the frontend:
1941
1959
 
1942
1960
  ```ruby
1943
1961
  require 'glimmer-dsl-web'
@@ -2110,7 +2128,7 @@ If you need to regenerate HTML element content dynamically, you can use Content
2110
2128
  rebuild HTML elements based on changes in a Model attribute that provides the source data.
2111
2129
  In this example, we generate multiple address forms based on the number of addresses the user has.
2112
2130
 
2113
- Glimmer GUI code:
2131
+ Glimmer GUI code in the frontend:
2114
2132
 
2115
2133
  ```ruby
2116
2134
  require 'glimmer-dsl-web'
@@ -2242,12 +2260,14 @@ Screenshot:
2242
2260
  You can define Glimmer web components (View components) to reuse visual concepts to your heart's content,
2243
2261
  by simply defining a class with `include Glimmer::Web::Component` and encasing the reusable markup inside
2244
2262
  a `markup {...}` block. Glimmer web components automatically extend the Glimmer GUI DSL with new keywords
2245
- that match the underscored versions of the component class names (e.g. a `OrderSummary` class yields
2263
+ that match the underscored versions of the component class names (e.g. an `OrderSummary` class yields
2246
2264
  the `order_summary` keyword for reusing that component within the Glimmer GUI DSL).
2247
- Below, we define an `AddressForm` component that generates a `address_form` keyword, and then we
2265
+ You may insert a Glimmer component anywhere into a Rails View using `glimmer_component(component_path, *args)`
2266
+ Rails helper (more about it in [Hello, glimmer_component Rails Helper!](#hello-glimmer_component-rails-helper)).
2267
+ Below, we define an `AddressForm` component that generates an `address_form` keyword, and then we
2248
2268
  reuse it twice inside an `AddressPage` component displaying a Shipping Address and a Billing Address.
2249
2269
 
2250
- Glimmer GUI code:
2270
+ Glimmer GUI code in the frontend:
2251
2271
 
2252
2272
  ```ruby
2253
2273
  require 'glimmer-dsl-web'
@@ -2496,7 +2516,7 @@ Rails View code:
2496
2516
  </div>
2497
2517
  ```
2498
2518
 
2499
- Glimmer GUI code:
2519
+ Glimmer GUI code in the frontend (`app/assets/opal/address_form.rb`):
2500
2520
 
2501
2521
  ```ruby
2502
2522
  require 'glimmer-dsl-web'
@@ -2661,7 +2681,7 @@ Screenshot:
2661
2681
 
2662
2682
  #### Hello, Paragraph!
2663
2683
 
2664
- To facilitate building formatted textual paragraphs in Ruby, the Glimmer GUI DSL is advanced enough to behave differently when using HTML formatting elements: `<br>`, `<strong>`, `<em>`, `<br>`, `<i>`, `<sub>`, `<sup>`, `<del>`, `<ins>`, `<small>`, `<mark>`
2684
+ To facilitate building formatted textual paragraphs in Ruby, thanks to [Glimmer](https://github.com/AndyObtiva/glimmer#dsl-engine), the most advanced DSL engine in Ruby, the Glimmer GUI DSL is advanced enough to intelligently behave differently under different situations, like when using HTML formatting elements: `<br>`, `<strong>`, `<em>`, `<br>`, `<i>`, `<sub>`, `<sup>`, `<del>`, `<ins>`, `<small>`, `<mark>`
2665
2685
 
2666
2686
  Instead of returning Ruby objects that are nested as children within their parent, the Glimmer GUI DSL returns `String` objects directly that can be concatenated to or embedded within other `String` objects via interpolation.
2667
2687
 
@@ -2693,7 +2713,7 @@ it returns a `String` to enable code like this:
2693
2713
 
2694
2714
  In any case, below is a full example leveraging the Glimmer GUI DSL alternative approach when utilizing formatting elements underneath a paragraph.
2695
2715
 
2696
- Glimmer GUI code:
2716
+ Glimmer GUI code in the frontend:
2697
2717
 
2698
2718
  ```ruby
2699
2719
  require 'glimmer-dsl-web'
@@ -2754,7 +2774,7 @@ Screenshot:
2754
2774
 
2755
2775
  #### Hello, Input (Date/Time)!
2756
2776
 
2757
- Glimmer GUI code:
2777
+ Glimmer GUI code in the frontend:
2758
2778
 
2759
2779
  ```ruby
2760
2780
  require 'glimmer-dsl-web'
@@ -2859,7 +2879,7 @@ Screenshot:
2859
2879
 
2860
2880
  #### Button Counter
2861
2881
 
2862
- Glimmer GUI code demonstrating MVC + Glimmer Web Components (Views) + Data-Binding:
2882
+ Glimmer GUI code in the frontend demonstrating MVC + Glimmer Web Components (Views) + Data-Binding:
2863
2883
 
2864
2884
  ```ruby
2865
2885
  require 'glimmer-dsl-web'
data/VERSION CHANGED
@@ -1 +1 @@
1
- 0.2.0
1
+ 0.2.3
@@ -2,17 +2,17 @@
2
2
  # DO NOT EDIT THIS FILE DIRECTLY
3
3
  # Instead, edit Jeweler::Tasks in Rakefile, and run 'rake gemspec'
4
4
  # -*- encoding: utf-8 -*-
5
- # stub: glimmer-dsl-web 0.2.0 ruby lib
5
+ # stub: glimmer-dsl-web 0.2.3 ruby lib
6
6
 
7
7
  Gem::Specification.new do |s|
8
8
  s.name = "glimmer-dsl-web".freeze
9
- s.version = "0.2.0".freeze
9
+ s.version = "0.2.3".freeze
10
10
 
11
11
  s.required_rubygems_version = Gem::Requirement.new(">= 0".freeze) if s.respond_to? :required_rubygems_version=
12
12
  s.require_paths = ["lib".freeze]
13
13
  s.authors = ["Andy Maleh".freeze]
14
- s.date = "2024-01-08"
15
- s.description = "Glimmer DSL for Web (Ruby in the Browser Web GUI Frontend Library) enables building Web GUI frontends using Ruby in the Browser, as per Matz's recommendation in his RubyConf 2022 keynote speech to replace JavaScript with Ruby. It aims at providing the simplest, most intuitive, most straight-forward, and most productive frontend library in existence. The library follows the Ruby way (with DSLs and TIMTOWTDI) and the Rails way (Convention over Configuration) while supporting both Unidirectional (One-Way) Data-Binding (using <=) and Bidirectional (Two-Way) Data-Binding (using <=>). Dynamic rendering (and re-rendering) of HTML content is also supported via Content Data-Binding. And, modular design is supported with Glimmer Web Components. You can finally live in pure Rubyland on the Web in both the frontend and backend with Glimmer DSL for Web! This library relies on Opal Ruby.".freeze
14
+ s.date = "2024-01-13"
15
+ s.description = "Glimmer DSL for Web (Ruby in the Browser Web GUI Frontend Library) enables building Web GUI frontends using Ruby in the Browser, as per Matz's recommendation in his RubyConf 2022 keynote speech to replace JavaScript with Ruby. It aims at providing the simplest, most intuitive, most straight-forward, and most productive frontend library in existence. The library follows the Ruby way (with DSLs and TIMTOWTDI) and the Rails way (Convention over Configuration) in building Isomorphic Ruby on Rails Applications. It supports both Unidirectional (One-Way) Data-Binding (using <=) and Bidirectional (Two-Way) Data-Binding (using <=>). Dynamic rendering (and re-rendering) of HTML content is also supported via Content Data-Binding. And, modular design is supported with Glimmer Web Components. You can finally live in pure Rubyland on the Web in both the frontend and backend with Glimmer DSL for Web! This library relies on Opal Ruby.".freeze
16
16
  s.email = "andy.am@gmail.com".freeze
17
17
  s.extra_rdoc_files = [
18
18
  "CHANGELOG.md",
@@ -54,10 +54,9 @@ Gem::Specification.new do |s|
54
54
  "lib/glimmer/dsl/web/general_element_expression.rb",
55
55
  "lib/glimmer/dsl/web/listener_expression.rb",
56
56
  "lib/glimmer/dsl/web/observe_expression.rb",
57
- "lib/glimmer/dsl/web/p_expression.rb",
58
57
  "lib/glimmer/dsl/web/property_expression.rb",
59
- "lib/glimmer/dsl/web/select_expression.rb",
60
58
  "lib/glimmer/dsl/web/shine_data_binding_expression.rb",
59
+ "lib/glimmer/dsl/web/span_expression.rb",
61
60
  "lib/glimmer/dsl/web/style_expression.rb",
62
61
  "lib/glimmer/helpers/glimmer_helper.rb",
63
62
  "lib/glimmer/util/proc_tracker.rb",
@@ -24,8 +24,7 @@ require 'glimmer/dsl/web/element_expression'
24
24
  require 'glimmer/dsl/web/formatting_element_expression'
25
25
  require 'glimmer/dsl/web/listener_expression'
26
26
  require 'glimmer/dsl/web/property_expression'
27
- require 'glimmer/dsl/web/p_expression'
28
- require 'glimmer/dsl/web/select_expression'
27
+ require 'glimmer/dsl/web/span_expression'
29
28
  require 'glimmer/dsl/web/style_expression'
30
29
  require 'glimmer/dsl/web/bind_expression'
31
30
  require 'glimmer/dsl/web/data_binding_expression'
@@ -47,7 +46,6 @@ module Glimmer
47
46
  content_data_binding
48
47
  shine_data_binding
49
48
  formatting_element
50
- element
51
49
  ]
52
50
  )
53
51
  end
@@ -16,3 +16,18 @@ module Glimmer
16
16
  end
17
17
  end
18
18
  end
19
+
20
+ module Kernel
21
+ alias pi p
22
+ end
23
+
24
+ module Glimmer
25
+ # Optimize performance through shortcut methods for all HTML elements that circumvent the DSL chain of responsibility
26
+ element_expression = Glimmer::DSL::Web::ElementExpression.new
27
+ (Glimmer::Web::ElementProxy::ELEMENT_KEYWORDS - ['span', 'style']).each do |keyword|
28
+ Glimmer::DSL::Engine.static_expressions[keyword] ||= Concurrent::Hash.new
29
+ element_expression_dsl = element_expression.class.dsl
30
+ Glimmer::DSL::Engine.static_expressions[keyword][element_expression_dsl] = element_expression
31
+ Glimmer.send(:define_method, keyword, &Glimmer::DSL::Engine::STATIC_EXPRESSION_METHOD_FACTORY.call(keyword))
32
+ end
33
+ end
@@ -8,7 +8,7 @@ module Glimmer
8
8
  include ParentExpression
9
9
 
10
10
  def interpret(parent, keyword, *args, &block)
11
- Glimmer::Web::ElementProxy.for(keyword, parent, args, block)
11
+ Glimmer::Web::ElementProxy.new(keyword, parent, args, block)
12
12
  end
13
13
 
14
14
  def add_content(parent, keyword, *args, &block)
@@ -0,0 +1,17 @@
1
+ require 'glimmer/dsl/static_expression'
2
+ require 'glimmer/dsl/web/general_element_expression'
3
+
4
+ module Glimmer
5
+ module DSL
6
+ module Web
7
+ class SpanExpression < StaticExpression
8
+ include GeneralElementExpression
9
+
10
+ def can_interpret?(parent, keyword, *args, &block)
11
+ (parent.nil? ||
12
+ (parent.respond_to?(:keyword) && parent.keyword != 'p'))
13
+ end
14
+ end
15
+ end
16
+ end
17
+ end
@@ -31,11 +31,14 @@ module Glimmer
31
31
  ELEMENT_KEYWORDS.include?(keyword.to_s)
32
32
  end
33
33
 
34
+ # NOTE: Avoid using this method until we start supporting ElementProxy subclasses
35
+ # in which case, we must cache them to avoid the slow performance of element_type
34
36
  # Factory Method that translates a Glimmer DSL keyword into a ElementProxy object
35
37
  def for(keyword, parent, args, block)
36
38
  element_type(keyword).new(keyword, parent, args, block)
37
39
  end
38
40
 
41
+ # NOTE: Avoid using this method for now as it has slow performance
39
42
  # returns Ruby proxy class (type) that would handle this keyword
40
43
  def element_type(keyword)
41
44
  class_name_main = "#{keyword.camelcase(:upper)}Proxy"
@@ -67,6 +70,19 @@ module Glimmer
67
70
  def widget_handling_listener
68
71
  @@widget_handling_listener
69
72
  end
73
+
74
+ def render_html(element, attributes, content = nil)
75
+ attributes = attributes.reduce('') do |output, option_pair|
76
+ attribute, value = option_pair
77
+ value = value.to_s.sub('"', '&quot;').sub("'", '&apos;')
78
+ output += " #{attribute}=\"#{value}\""
79
+ end
80
+ if content.nil?
81
+ "<#{element}#{attributes} />"
82
+ else
83
+ "<#{element}#{attributes}>#{content}</#{element}>"
84
+ end
85
+ end
70
86
  end
71
87
 
72
88
  include Glimmer
@@ -74,18 +90,18 @@ module Glimmer
74
90
  Event = Struct.new(:widget, keyword_init: true)
75
91
 
76
92
  ELEMENT_KEYWORDS = [
77
- "a", "abbr", "acronym", "address", "applet", "area", "article", "aside", "audio", "b",
78
- "base", "basefont", "bdi", "bdo", "bgsound", "big", "blink", "blockquote", "body", "br",
79
- "button", "canvas", "caption", "center", "cite", "code", "col", "colgroup", "content", "data",
80
- "datalist", "dd", "decorator", "del", "details", "dfn", "dir", "div", "dl", "dt",
81
- "element", "em", "embed", "fieldset", "figcaption", "figure", "font", "footer", "form", "frame",
93
+ "a", "abbr", "acronym", "address", "applet", "area", "article", "aside", "audio",
94
+ "base", "basefont", "bdi", "bdo", "bgsound", "big", "blink", "blockquote", "body",
95
+ "button", "canvas", "caption", "center", "cite", "code", "col", "colgroup", "data",
96
+ "datalist", "dd", "decorator", "details", "dfn", "dir", "div", "dl", "dt",
97
+ "element", "embed", "fieldset", "figcaption", "figure", "font", "footer", "form", "frame",
82
98
  "frameset", "h1", "h2", "h3", "h4", "h5", "h6", "head", "header", "hgroup",
83
- "hr", "html", "i", "iframe", "img", "input", "ins", "isindex", "kbd", "keygen",
84
- "label", "legend", "li", "link", "listing", "main", "map", "mark", "marquee", "menu",
99
+ "hr", "html", "iframe", "img", "input", "isindex", "kbd", "keygen",
100
+ "label", "legend", "li", "link", "listing", "main", "map", "marquee", "menu",
85
101
  "menuitem", "meta", "meter", "nav", "nobr", "noframes", "noscript", "object", "ol", "optgroup",
86
102
  "option", "output", "p", "param", "plaintext", "pre", "progress", "q", "rp", "rt",
87
- "ruby", "s", "samp", "script", "section", "select", "shadow", "small", "source", "spacer",
88
- "span", "strike", "strong", "style", "sub", "summary", "sup", "table", "tbody", "td",
103
+ "ruby", "s", "samp", "script", "section", "select", "shadow", "source", "spacer",
104
+ "span", "strike", "style", "summary", "table", "tbody", "td",
89
105
  "template", "textarea", "tfoot", "th", "thead", "time", "title", "tr", "track", "tt",
90
106
  "u", "ul", "var", "video", "wbr", "xmp",
91
107
  ]
@@ -249,8 +265,7 @@ module Glimmer
249
265
  @parent = nil
250
266
  end
251
267
  the_parent_dom_element = custom_parent_dom_element || parent_dom_element
252
- old_element = dom_element
253
- brand_new = @dom.nil? || old_element.empty? || !options[:parent].to_s.empty? || brand_new
268
+ brand_new ||= @dom.nil? || !options[:parent].to_s.empty? || (old_element = dom_element).empty?
254
269
  build_dom(layout: !custom_parent_dom_element) # TODO handle custom parent layout by passing parent instead of parent dom element
255
270
  if brand_new
256
271
  attach(the_parent_dom_element)
@@ -310,19 +325,16 @@ module Glimmer
310
325
 
311
326
  def build_dom(layout: true)
312
327
  # TODO consider passing parent element instead and having table item include a table cell widget only for opal
313
- @dom = nil
314
328
  @dom = dom # TODO unify how to build dom for most widgets based on element, id, and name (class)
315
- @dom
316
329
  end
317
330
 
318
331
  def dom
319
332
  # TODO auto-convert known glimmer attributes like parent to data attributes like data-parent
320
333
  # TODO check if we need to avoid rendering content block if no content is available
321
- @dom ||= html {
322
- send(keyword, html_options) {
323
- args.first if args.first.is_a?(String)
324
- }
325
- }.to_s
334
+ @dom ||= begin
335
+ content = args.first if args.first.is_a?(String)
336
+ ElementProxy.render_html(keyword, html_options, content)
337
+ end
326
338
  end
327
339
 
328
340
  def html_options
@@ -359,6 +371,12 @@ module Glimmer
359
371
  @element_id ||= "element-#{ElementProxy.next_id_number_for(name)}"
360
372
  end
361
373
 
374
+ def class_name=(value)
375
+ value = value.is_a?(Array) ? value.join(' ') : value.to_s
376
+ new_class_name = "#{name} #{element_id} #{value}"
377
+ dom_element.prop('className', new_class_name)
378
+ end
379
+
362
380
  def add_css_class(css_class)
363
381
  dom_element.add_class(css_class)
364
382
  end
@@ -21,6 +21,8 @@
21
21
  # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
22
  # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23
23
 
24
+ require 'glimmer/web/element_proxy'
25
+
24
26
  module Glimmer
25
27
  module Web
26
28
  class FormattingElementProxy
@@ -31,7 +33,7 @@ module Glimmer
31
33
  keyword = keyword.to_s
32
34
  (
33
35
  FORMATTING_ELEMENT_KEYWORDS.include?(keyword) ||
34
- (parent&.keyword == 'p' && keyword == 'span')
36
+ (keyword == 'span' && parent&.keyword == 'p')
35
37
  )
36
38
  end
37
39
 
@@ -43,10 +45,7 @@ module Glimmer
43
45
  content = args.first.to_s
44
46
  end
45
47
  attribute_hash = args.last.is_a?(Hash) ? args.last : {}
46
- content_block = proc { content } unless content.nil?
47
- html {
48
- send(keyword, attribute_hash, &content_block)
49
- }.to_s
48
+ ElementProxy.render_html(keyword, attribute_hash, content)
50
49
  end
51
50
  end
52
51
 
@@ -77,7 +77,6 @@ else
77
77
 
78
78
  require 'glimmer/dsl/web/dsl'
79
79
  require 'glimmer/config/opal_logger'
80
- require 'glimmer-dsl-xml'
81
80
  require 'glimmer-dsl-css'
82
81
 
83
82
  Glimmer::Config.loop_max_count = 50 # TODO consider disabling if preferred
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: glimmer-dsl-web
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.2.0
4
+ version: 0.2.3
5
5
  platform: ruby
6
6
  authors:
7
7
  - Andy Maleh
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2024-01-08 00:00:00.000000000 Z
11
+ date: 2024-01-13 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: glimmer
@@ -235,12 +235,13 @@ description: Glimmer DSL for Web (Ruby in the Browser Web GUI Frontend Library)
235
235
  in his RubyConf 2022 keynote speech to replace JavaScript with Ruby. It aims at
236
236
  providing the simplest, most intuitive, most straight-forward, and most productive
237
237
  frontend library in existence. The library follows the Ruby way (with DSLs and TIMTOWTDI)
238
- and the Rails way (Convention over Configuration) while supporting both Unidirectional
239
- (One-Way) Data-Binding (using <=) and Bidirectional (Two-Way) Data-Binding (using
240
- <=>). Dynamic rendering (and re-rendering) of HTML content is also supported via
241
- Content Data-Binding. And, modular design is supported with Glimmer Web Components.
242
- You can finally live in pure Rubyland on the Web in both the frontend and backend
243
- with Glimmer DSL for Web! This library relies on Opal Ruby.
238
+ and the Rails way (Convention over Configuration) in building Isomorphic Ruby on
239
+ Rails Applications. It supports both Unidirectional (One-Way) Data-Binding (using
240
+ <=) and Bidirectional (Two-Way) Data-Binding (using <=>). Dynamic rendering (and
241
+ re-rendering) of HTML content is also supported via Content Data-Binding. And, modular
242
+ design is supported with Glimmer Web Components. You can finally live in pure Rubyland
243
+ on the Web in both the frontend and backend with Glimmer DSL for Web! This library
244
+ relies on Opal Ruby.
244
245
  email: andy.am@gmail.com
245
246
  executables: []
246
247
  extensions: []
@@ -283,10 +284,9 @@ files:
283
284
  - lib/glimmer/dsl/web/general_element_expression.rb
284
285
  - lib/glimmer/dsl/web/listener_expression.rb
285
286
  - lib/glimmer/dsl/web/observe_expression.rb
286
- - lib/glimmer/dsl/web/p_expression.rb
287
287
  - lib/glimmer/dsl/web/property_expression.rb
288
- - lib/glimmer/dsl/web/select_expression.rb
289
288
  - lib/glimmer/dsl/web/shine_data_binding_expression.rb
289
+ - lib/glimmer/dsl/web/span_expression.rb
290
290
  - lib/glimmer/dsl/web/style_expression.rb
291
291
  - lib/glimmer/helpers/glimmer_helper.rb
292
292
  - lib/glimmer/util/proc_tracker.rb
@@ -1,12 +0,0 @@
1
- require 'glimmer/dsl/static_expression'
2
- require 'glimmer/dsl/web/general_element_expression'
3
-
4
- module Glimmer
5
- module DSL
6
- module Web
7
- class PExpression < StaticExpression
8
- include GeneralElementExpression
9
- end
10
- end
11
- end
12
- end
@@ -1,12 +0,0 @@
1
- require 'glimmer/dsl/static_expression'
2
- require 'glimmer/dsl/web/general_element_expression'
3
-
4
- module Glimmer
5
- module DSL
6
- module Web
7
- class SelectExpression < StaticExpression
8
- include GeneralElementExpression
9
- end
10
- end
11
- end
12
- end