ruwi 0.10.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 (117) hide show
  1. checksums.yaml +7 -0
  2. data/.cursor/rules/ruby_comments.mdc +29 -0
  3. data/.github/workflows/playwright.yml +74 -0
  4. data/.github/workflows/rspec.yml +31 -0
  5. data/.node-version +1 -0
  6. data/CODE_OF_CONDUCT.md +132 -0
  7. data/LICENSE.txt +21 -0
  8. data/Makefile +56 -0
  9. data/README.md +237 -0
  10. data/Rakefile +4 -0
  11. data/docs/conditional-rendering.md +119 -0
  12. data/docs/lifecycle-hooks.md +75 -0
  13. data/docs/list-rendering.md +51 -0
  14. data/examples/.gitignore +4 -0
  15. data/examples/Gemfile +5 -0
  16. data/examples/Gemfile.lock +39 -0
  17. data/examples/Makefile +15 -0
  18. data/examples/npm-packages/runtime/counter/index.html +28 -0
  19. data/examples/npm-packages/runtime/counter/index.rb +62 -0
  20. data/examples/npm-packages/runtime/counter/index.spec.js +42 -0
  21. data/examples/npm-packages/runtime/hello/index.html +28 -0
  22. data/examples/npm-packages/runtime/hello/index.rb +29 -0
  23. data/examples/npm-packages/runtime/hello/index.spec.js +53 -0
  24. data/examples/npm-packages/runtime/input/index.html +28 -0
  25. data/examples/npm-packages/runtime/input/index.rb +46 -0
  26. data/examples/npm-packages/runtime/input/index.spec.js +58 -0
  27. data/examples/npm-packages/runtime/list/index.html +27 -0
  28. data/examples/npm-packages/runtime/list/index.rb +33 -0
  29. data/examples/npm-packages/runtime/list/index.spec.js +46 -0
  30. data/examples/npm-packages/runtime/on_mounted_demo/index.html +40 -0
  31. data/examples/npm-packages/runtime/on_mounted_demo/index.rb +59 -0
  32. data/examples/npm-packages/runtime/on_mounted_demo/index.spec.js +50 -0
  33. data/examples/npm-packages/runtime/r_if_attribute_demo/index.html +34 -0
  34. data/examples/npm-packages/runtime/r_if_attribute_demo/index.rb +113 -0
  35. data/examples/npm-packages/runtime/r_if_attribute_demo/index.spec.js +140 -0
  36. data/examples/npm-packages/runtime/random_cocktail/index.html +27 -0
  37. data/examples/npm-packages/runtime/random_cocktail/index.rb +69 -0
  38. data/examples/npm-packages/runtime/random_cocktail/index.spec.js +101 -0
  39. data/examples/npm-packages/runtime/search_field/index.html +27 -0
  40. data/examples/npm-packages/runtime/search_field/index.rb +39 -0
  41. data/examples/npm-packages/runtime/search_field/index.spec.js +59 -0
  42. data/examples/npm-packages/runtime/todos/index.html +28 -0
  43. data/examples/npm-packages/runtime/todos/index.rb +239 -0
  44. data/examples/npm-packages/runtime/todos/index.spec.js +161 -0
  45. data/examples/npm-packages/runtime/todos/todos_repository.rb +23 -0
  46. data/examples/package.json +12 -0
  47. data/examples/src/counter/index.html +23 -0
  48. data/examples/src/counter/index.rb +60 -0
  49. data/examples/src/index.html +21 -0
  50. data/examples/src/index.rb +26 -0
  51. data/examples/src/todos/index.html +23 -0
  52. data/examples/src/todos/index.rb +237 -0
  53. data/examples/src/todos/todos_repository.rb +23 -0
  54. data/exe/ruwi +6 -0
  55. data/lib/ruwi/cli/command/base.rb +192 -0
  56. data/lib/ruwi/cli/command/dev.rb +207 -0
  57. data/lib/ruwi/cli/command/pack.rb +36 -0
  58. data/lib/ruwi/cli/command/rebuild.rb +38 -0
  59. data/lib/ruwi/cli/command/setup.rb +159 -0
  60. data/lib/ruwi/cli/command.rb +48 -0
  61. data/lib/ruwi/runtime/app.rb +53 -0
  62. data/lib/ruwi/runtime/component.rb +215 -0
  63. data/lib/ruwi/runtime/dispatcher.rb +46 -0
  64. data/lib/ruwi/runtime/dom/attributes.rb +105 -0
  65. data/lib/ruwi/runtime/dom/destroy_dom.rb +63 -0
  66. data/lib/ruwi/runtime/dom/events.rb +40 -0
  67. data/lib/ruwi/runtime/dom/mount_dom.rb +108 -0
  68. data/lib/ruwi/runtime/dom/patch_dom.rb +237 -0
  69. data/lib/ruwi/runtime/dom/scheduler.rb +51 -0
  70. data/lib/ruwi/runtime/dom.rb +13 -0
  71. data/lib/ruwi/runtime/nodes_equal.rb +45 -0
  72. data/lib/ruwi/runtime/template/build_conditional_group.rb +150 -0
  73. data/lib/ruwi/runtime/template/build_for_group.rb +125 -0
  74. data/lib/ruwi/runtime/template/build_vdom.rb +220 -0
  75. data/lib/ruwi/runtime/template/parser.rb +134 -0
  76. data/lib/ruwi/runtime/template.rb +11 -0
  77. data/lib/ruwi/runtime/utils/arrays.rb +185 -0
  78. data/lib/ruwi/runtime/utils/objects.rb +37 -0
  79. data/lib/ruwi/runtime/utils/props.rb +25 -0
  80. data/lib/ruwi/runtime/utils/strings.rb +19 -0
  81. data/lib/ruwi/runtime/utils.rb +11 -0
  82. data/lib/ruwi/runtime/vdom.rb +84 -0
  83. data/lib/ruwi/version.rb +5 -0
  84. data/lib/ruwi.rb +14 -0
  85. data/package-lock.json +73 -0
  86. data/package.json +32 -0
  87. data/packages/npm-packages/runtime/README.md +5 -0
  88. data/packages/npm-packages/runtime/eslint.config.mjs +16 -0
  89. data/packages/npm-packages/runtime/package-lock.json +6668 -0
  90. data/packages/npm-packages/runtime/package.json +38 -0
  91. data/packages/npm-packages/runtime/rollup.config.mjs +147 -0
  92. data/packages/npm-packages/runtime/src/__tests__/sample.test.js +5 -0
  93. data/packages/npm-packages/runtime/src/index.js +37 -0
  94. data/packages/npm-packages/runtime/src/ruwi +1 -0
  95. data/packages/npm-packages/runtime/src/ruwi.rb +1 -0
  96. data/packages/npm-packages/runtime/vitest.config.js +8 -0
  97. data/playwright.config.js +78 -0
  98. data/sig/ruwi.rbs +4 -0
  99. data/spec/ruwi/cli/command/base_spec.rb +503 -0
  100. data/spec/ruwi/cli/command/dev_spec.rb +442 -0
  101. data/spec/ruwi/cli/command/pack_spec.rb +131 -0
  102. data/spec/ruwi/cli/command/rebuild_spec.rb +95 -0
  103. data/spec/ruwi/cli/command/setup_spec.rb +251 -0
  104. data/spec/ruwi/cli/command_spec.rb +118 -0
  105. data/spec/ruwi/runtime/component_spec.rb +416 -0
  106. data/spec/ruwi/runtime/dom/scheduler_spec.rb +98 -0
  107. data/spec/ruwi/runtime/nodes_equal_spec.rb +190 -0
  108. data/spec/ruwi/runtime/template/build_conditional_group_spec.rb +505 -0
  109. data/spec/ruwi/runtime/template/build_for_group_spec.rb +377 -0
  110. data/spec/ruwi/runtime/template/build_vdom_spec.rb +573 -0
  111. data/spec/ruwi/runtime/template/parser_spec.rb +627 -0
  112. data/spec/ruwi/runtime/utils/arrays_spec.rb +228 -0
  113. data/spec/ruwi/runtime/utils/objects_spec.rb +127 -0
  114. data/spec/ruwi/runtime/utils/props_spec.rb +205 -0
  115. data/spec/ruwi/runtime/utils/strings_spec.rb +107 -0
  116. data/spec/spec_helper.rb +16 -0
  117. metadata +229 -0
@@ -0,0 +1,84 @@
1
+ module Ruwi
2
+ class Vdom
3
+ DOM_TYPES = {
4
+ TEXT: 'text',
5
+ ELEMENT: 'element',
6
+ FRAGMENT: 'fragment',
7
+ COMPONENT: 'component'
8
+ }
9
+
10
+ # @param tag [String]
11
+ # @param props [Hash]
12
+ # @param children [Array]
13
+ # @param type [Symbol]
14
+ # @param value [Object]
15
+ # @return [Vdom]
16
+ def self.h(tag, props = {}, children = [])
17
+ type = tag.is_a?(String) ? DOM_TYPES[:ELEMENT] : DOM_TYPES[:COMPONENT]
18
+ new(tag, props, type, children, nil)
19
+ end
20
+
21
+ # @param str [String]
22
+ # @return [Vdom]
23
+ def self.h_string(str)
24
+ vdom = new('', {}, DOM_TYPES[:TEXT], [], str.to_s)
25
+ vdom
26
+ end
27
+
28
+ # @param vdoms [Array]
29
+ # @return [Vdom]
30
+ def self.h_fragment(vdoms)
31
+ new('', {}, DOM_TYPES[:FRAGMENT], map_text_nodes(Ruwi::Utils::Arrays.without_nulls(vdoms)), nil)
32
+ end
33
+
34
+ # @param type [Symbol]
35
+ # @param props [Hash]
36
+ # @param children [Array]
37
+ # @param value [Object]
38
+ # @return [Vdom]
39
+ def initialize(tag, props, type, children, value)
40
+ @tag = tag
41
+ @props = props
42
+ @children = self.class.map_text_nodes(Ruwi::Utils::Arrays.without_nulls(children))
43
+ @type = type
44
+ @value = value.to_s
45
+ @el = nil
46
+ @listeners = {}
47
+ @component = nil
48
+ end
49
+
50
+ attr_reader :tag, :props, :children, :type, :value
51
+ attr_accessor :el, :listeners, :component
52
+
53
+ private
54
+
55
+ # @param children [Array]
56
+ def self.map_text_nodes(children)
57
+ children.map { |child| is_text_node?(child) ? h_string(child) : child }
58
+ end
59
+
60
+ # @param child [Object]
61
+ # @return [Boolean]
62
+ def self.is_text_node?(child)
63
+ child.is_a?(String) || child.is_a?(Integer) || child.is_a?(JS::Object)
64
+ end
65
+
66
+ # @param vdom [Vdom]
67
+ # @return [Array]
68
+ def self.extract_children(vdom)
69
+ return [] if vdom.children.nil?
70
+
71
+ children = []
72
+
73
+ vdom.children.each do |child|
74
+ if child.type == DOM_TYPES[:FRAGMENT]
75
+ children.concat(extract_children(child))
76
+ else
77
+ children << child
78
+ end
79
+ end
80
+
81
+ children
82
+ end
83
+ end
84
+ end
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Ruwi
4
+ VERSION = "0.10.0"
5
+ end
data/lib/ruwi.rb ADDED
@@ -0,0 +1,14 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "ruwi/runtime/app"
4
+ require_relative "ruwi/runtime/component"
5
+ require_relative "ruwi/runtime/dispatcher"
6
+ require_relative "ruwi/runtime/dom"
7
+ require_relative "ruwi/runtime/nodes_equal"
8
+ require_relative "ruwi/runtime/template"
9
+ require_relative "ruwi/runtime/utils"
10
+ require_relative "ruwi/runtime/vdom"
11
+ require_relative "ruwi/version"
12
+
13
+ module Ruwi
14
+ end
data/package-lock.json ADDED
@@ -0,0 +1,73 @@
1
+ {
2
+ "name": "ruwi-project",
3
+ "version": "0.10.0",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "ruwi-project",
9
+ "version": "0.10.0",
10
+ "license": "MIT",
11
+ "workspaces": [
12
+ "packages/*"
13
+ ],
14
+ "devDependencies": {
15
+ "@playwright/test": "^1.40.0"
16
+ }
17
+ },
18
+ "node_modules/@playwright/test": {
19
+ "version": "1.55.0",
20
+ "dev": true,
21
+ "license": "Apache-2.0",
22
+ "dependencies": {
23
+ "playwright": "1.55.0"
24
+ },
25
+ "bin": {
26
+ "playwright": "cli.js"
27
+ },
28
+ "engines": {
29
+ "node": ">=18"
30
+ }
31
+ },
32
+ "node_modules/fsevents": {
33
+ "version": "2.3.2",
34
+ "dev": true,
35
+ "license": "MIT",
36
+ "optional": true,
37
+ "os": [
38
+ "darwin"
39
+ ],
40
+ "engines": {
41
+ "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
42
+ }
43
+ },
44
+ "node_modules/playwright": {
45
+ "version": "1.55.0",
46
+ "dev": true,
47
+ "license": "Apache-2.0",
48
+ "dependencies": {
49
+ "playwright-core": "1.55.0"
50
+ },
51
+ "bin": {
52
+ "playwright": "cli.js"
53
+ },
54
+ "engines": {
55
+ "node": ">=18"
56
+ },
57
+ "optionalDependencies": {
58
+ "fsevents": "2.3.2"
59
+ }
60
+ },
61
+ "node_modules/playwright-core": {
62
+ "version": "1.55.0",
63
+ "dev": true,
64
+ "license": "Apache-2.0",
65
+ "bin": {
66
+ "playwright-core": "cli.js"
67
+ },
68
+ "engines": {
69
+ "node": ">=18"
70
+ }
71
+ }
72
+ }
73
+ }
data/package.json ADDED
@@ -0,0 +1,32 @@
1
+ {
2
+ "name": "ruwi-project",
3
+ "version": "0.10.0",
4
+ "private": true,
5
+ "description": "A project to ruby.wasm ui framework",
6
+ "scripts": {
7
+ "serve:examples": "npx http-server . -o './examples/npm-packages/runtime/?env=PRD' --cors -P 'http://localhost:8080?' -c-1",
8
+ "serve:examples:dev": "cd packages/npm-packages/runtime && npm run build:dev && cd ../../.. && npx http-server . -o './examples/npm-packages/runtime/?env=DEV' --cors -P 'http://localhost:8080?' -c-1",
9
+ "serve:examples:test": "cd packages/npm-packages/runtime && npm run build:dev && cd ../../.. && npx http-server . --cors -P 'http://localhost:8080?' -c-1",
10
+ "test": "npx playwright test",
11
+ "test:ui": "npx playwright test --ui",
12
+ "test:report": "npx playwright show-report"
13
+ },
14
+ "repository": {
15
+ "type": "git",
16
+ "url": "git+https://github.com/t0yohei/ruby-wasm-ui.git"
17
+ },
18
+ "keywords": [
19
+ "ruby",
20
+ "wasm",
21
+ "ui"
22
+ ],
23
+ "author": "t0yohei <k.t0yohei@gmail.com>",
24
+ "license": "MIT",
25
+ "type": "module",
26
+ "workspaces": [
27
+ "packages/*"
28
+ ],
29
+ "devDependencies": {
30
+ "@playwright/test": "^1.40.0"
31
+ }
32
+ }
@@ -0,0 +1,5 @@
1
+ # ruwi
2
+
3
+ A modern web frontend framework for Ruby using [ruby.wasm](https://github.com/ruby/ruby.wasm). Write reactive web applications using familiar Ruby syntax and patterns.
4
+
5
+ Please refer to the [GitHub README](https://github.com/t0yohei/ruby-wasm-ui#readme) for full documentation.
@@ -0,0 +1,16 @@
1
+ import js from "@eslint/js";
2
+ import globals from "globals";
3
+
4
+ export default [
5
+ js.configs.recommended,
6
+ {
7
+ languageOptions: {
8
+ ecmaVersion: "latest",
9
+ sourceType: "module",
10
+ globals: {
11
+ ...globals.browser,
12
+ },
13
+ },
14
+ rules: {},
15
+ },
16
+ ];