ruby_ui 1.0.0.pre.alpha.4 → 1.0.0.rc1

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 (227) hide show
  1. checksums.yaml +4 -4
  2. data/LICENSE.txt +21 -0
  3. data/README.md +85 -0
  4. data/lib/generators/ruby_ui/component_generator.rb +94 -0
  5. data/lib/generators/ruby_ui/dependencies.yml +74 -0
  6. data/lib/generators/ruby_ui/install/install_generator.rb +89 -0
  7. data/lib/generators/ruby_ui/install/templates/ruby_ui.rb.erb +18 -0
  8. data/lib/generators/ruby_ui/install/templates/tailwind.css.erb +156 -0
  9. data/lib/generators/ruby_ui/javascript_utils.rb +57 -0
  10. data/lib/{rbui → ruby_ui}/accordion/accordion.rb +1 -1
  11. data/lib/{rbui → ruby_ui}/accordion/accordion_content.rb +2 -2
  12. data/lib/ruby_ui/accordion/accordion_controller.js +97 -0
  13. data/lib/{rbui → ruby_ui}/accordion/accordion_default_content.rb +1 -1
  14. data/lib/{rbui → ruby_ui}/accordion/accordion_default_trigger.rb +3 -3
  15. data/lib/{rbui → ruby_ui}/accordion/accordion_icon.rb +2 -2
  16. data/lib/{rbui → ruby_ui}/accordion/accordion_item.rb +4 -4
  17. data/lib/{rbui → ruby_ui}/accordion/accordion_trigger.rb +3 -2
  18. data/lib/{rbui → ruby_ui}/alert/alert.rb +3 -3
  19. data/lib/{rbui → ruby_ui}/alert/alert_description.rb +1 -1
  20. data/lib/{rbui → ruby_ui}/alert/alert_title.rb +1 -1
  21. data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog.rb +3 -3
  22. data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_action.rb +2 -2
  23. data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_cancel.rb +3 -3
  24. data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_content.rb +5 -5
  25. data/lib/ruby_ui/alert_dialog/alert_dialog_controller.js +31 -0
  26. data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_description.rb +1 -1
  27. data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_footer.rb +2 -2
  28. data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_header.rb +2 -2
  29. data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_title.rb +1 -1
  30. data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_trigger.rb +2 -2
  31. data/lib/{rbui → ruby_ui}/aspect_ratio/aspect_ratio.rb +1 -1
  32. data/lib/{rbui → ruby_ui}/avatar/avatar.rb +2 -2
  33. data/lib/{rbui → ruby_ui}/avatar/avatar_fallback.rb +1 -1
  34. data/lib/{rbui → ruby_ui}/avatar/avatar_image.rb +1 -1
  35. data/lib/{rbui → ruby_ui}/badge/badge.rb +2 -2
  36. data/lib/{rbui → ruby_ui}/base.rb +1 -8
  37. data/lib/ruby_ui/breadcrumb/breadcrumb.rb +17 -0
  38. data/lib/ruby_ui/breadcrumb/breadcrumb_ellipsis.rb +39 -0
  39. data/lib/{rbui/typography/typography_list_item.rb → ruby_ui/breadcrumb/breadcrumb_item.rb} +3 -3
  40. data/lib/ruby_ui/breadcrumb/breadcrumb_link.rb +22 -0
  41. data/lib/ruby_ui/breadcrumb/breadcrumb_list.rb +17 -0
  42. data/lib/ruby_ui/breadcrumb/breadcrumb_page.rb +19 -0
  43. data/lib/ruby_ui/breadcrumb/breadcrumb_separator.rb +38 -0
  44. data/lib/{rbui → ruby_ui}/button/button.rb +13 -13
  45. data/lib/ruby_ui/calendar/calendar.rb +39 -0
  46. data/lib/{rbui → ruby_ui}/calendar/calendar_body.rb +2 -2
  47. data/lib/ruby_ui/calendar/calendar_controller.js +249 -0
  48. data/lib/{rbui → ruby_ui}/calendar/calendar_days.rb +14 -14
  49. data/lib/{rbui → ruby_ui}/calendar/calendar_header.rb +1 -1
  50. data/lib/ruby_ui/calendar/calendar_input_controller.js +8 -0
  51. data/lib/{rbui → ruby_ui}/calendar/calendar_next.rb +2 -2
  52. data/lib/{rbui → ruby_ui}/calendar/calendar_prev.rb +2 -2
  53. data/lib/{rbui → ruby_ui}/calendar/calendar_title.rb +2 -2
  54. data/lib/{rbui → ruby_ui}/calendar/calendar_weekdays.rb +2 -2
  55. data/lib/{rbui → ruby_ui}/card/card.rb +1 -1
  56. data/lib/{rbui → ruby_ui}/card/card_content.rb +1 -1
  57. data/lib/{rbui → ruby_ui}/card/card_description.rb +1 -1
  58. data/lib/{rbui → ruby_ui}/card/card_footer.rb +1 -1
  59. data/lib/{rbui → ruby_ui}/card/card_header.rb +1 -1
  60. data/lib/{rbui → ruby_ui}/card/card_title.rb +1 -1
  61. data/lib/ruby_ui/carousel/carousel.rb +44 -0
  62. data/lib/ruby_ui/carousel/carousel_content.rb +23 -0
  63. data/lib/ruby_ui/carousel/carousel_controller.js +60 -0
  64. data/lib/ruby_ui/carousel/carousel_item.rb +23 -0
  65. data/lib/ruby_ui/carousel/carousel_next.rb +48 -0
  66. data/lib/ruby_ui/carousel/carousel_previous.rb +49 -0
  67. data/lib/{rbui → ruby_ui}/chart/chart.rb +3 -3
  68. data/lib/ruby_ui/chart/chart_controller.js +103 -0
  69. data/lib/{rbui → ruby_ui}/checkbox/checkbox.rb +4 -4
  70. data/lib/{rbui → ruby_ui}/checkbox/checkbox_group.rb +2 -2
  71. data/lib/ruby_ui/checkbox/checkbox_group_controller.js +21 -0
  72. data/lib/{rbui → ruby_ui}/clipboard/clipboard.rb +6 -6
  73. data/lib/ruby_ui/clipboard/clipboard_controller.js +54 -0
  74. data/lib/{rbui → ruby_ui}/clipboard/clipboard_popover.rb +2 -2
  75. data/lib/{rbui → ruby_ui}/clipboard/clipboard_source.rb +2 -2
  76. data/lib/{rbui → ruby_ui}/clipboard/clipboard_trigger.rb +3 -3
  77. data/lib/{rbui → ruby_ui}/codeblock/codeblock.rb +7 -10
  78. data/lib/{rbui → ruby_ui}/collapsible/collapsible.rb +3 -3
  79. data/lib/{rbui → ruby_ui}/collapsible/collapsible_content.rb +2 -2
  80. data/lib/ruby_ui/collapsible/collapsible_controller.js +47 -0
  81. data/lib/{rbui → ruby_ui}/collapsible/collapsible_trigger.rb +2 -2
  82. data/lib/ruby_ui/combobox/combobox.rb +26 -0
  83. data/lib/ruby_ui/combobox/combobox_checkbox.rb +25 -0
  84. data/lib/ruby_ui/combobox/combobox_controller.js +176 -0
  85. data/lib/{rbui/combobox/combobox_empty.rb → ruby_ui/combobox/combobox_empty_state.rb} +3 -3
  86. data/lib/ruby_ui/combobox/combobox_item.rb +25 -0
  87. data/lib/ruby_ui/combobox/combobox_list.rb +18 -0
  88. data/lib/ruby_ui/combobox/combobox_list_group.rb +20 -0
  89. data/lib/ruby_ui/combobox/combobox_popover.rb +30 -0
  90. data/lib/ruby_ui/combobox/combobox_radio.rb +26 -0
  91. data/lib/{rbui → ruby_ui}/combobox/combobox_search_input.rb +22 -25
  92. data/lib/ruby_ui/combobox/combobox_toggle_all_checkbox.rb +25 -0
  93. data/lib/{rbui → ruby_ui}/combobox/combobox_trigger.rb +26 -21
  94. data/lib/{rbui → ruby_ui}/command/command.rb +1 -1
  95. data/lib/ruby_ui/command/command_controller.js +136 -0
  96. data/lib/{rbui → ruby_ui}/command/command_dialog.rb +2 -2
  97. data/lib/{rbui → ruby_ui}/command/command_dialog_content.rb +6 -6
  98. data/lib/{rbui → ruby_ui}/command/command_dialog_trigger.rb +3 -3
  99. data/lib/{rbui → ruby_ui}/command/command_empty.rb +2 -2
  100. data/lib/{rbui → ruby_ui}/command/command_group.rb +2 -2
  101. data/lib/{rbui → ruby_ui}/command/command_input.rb +3 -3
  102. data/lib/{rbui → ruby_ui}/command/command_item.rb +2 -2
  103. data/lib/{rbui → ruby_ui}/command/command_list.rb +1 -1
  104. data/lib/{rbui → ruby_ui}/context_menu/context_menu.rb +2 -2
  105. data/lib/{rbui → ruby_ui}/context_menu/context_menu_content.rb +2 -2
  106. data/lib/ruby_ui/context_menu/context_menu_controller.js +144 -0
  107. data/lib/{rbui → ruby_ui}/context_menu/context_menu_item.rb +3 -3
  108. data/lib/{rbui → ruby_ui}/context_menu/context_menu_label.rb +2 -2
  109. data/lib/{rbui → ruby_ui}/context_menu/context_menu_separator.rb +1 -1
  110. data/lib/{rbui → ruby_ui}/context_menu/context_menu_trigger.rb +3 -3
  111. data/lib/{rbui → ruby_ui}/dialog/dialog.rb +3 -3
  112. data/lib/{rbui → ruby_ui}/dialog/dialog_content.rb +9 -9
  113. data/lib/ruby_ui/dialog/dialog_controller.js +32 -0
  114. data/lib/{rbui → ruby_ui}/dialog/dialog_description.rb +1 -1
  115. data/lib/{rbui → ruby_ui}/dialog/dialog_footer.rb +2 -2
  116. data/lib/{rbui → ruby_ui}/dialog/dialog_header.rb +2 -2
  117. data/lib/{rbui → ruby_ui}/dialog/dialog_middle.rb +1 -1
  118. data/lib/{rbui → ruby_ui}/dialog/dialog_title.rb +1 -1
  119. data/lib/{rbui → ruby_ui}/dialog/dialog_trigger.rb +2 -2
  120. data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu.rb +4 -4
  121. data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu_content.rb +2 -2
  122. data/lib/ruby_ui/dropdown_menu/dropdown_menu_controller.js +120 -0
  123. data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu_item.rb +3 -3
  124. data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu_label.rb +1 -1
  125. data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu_separator.rb +1 -1
  126. data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu_trigger.rb +2 -2
  127. data/lib/{rbui → ruby_ui}/form/form.rb +1 -1
  128. data/lib/{rbui → ruby_ui}/form/form_field.rb +2 -2
  129. data/lib/ruby_ui/form/form_field_controller.js +61 -0
  130. data/lib/{rbui → ruby_ui}/form/form_field_error.rb +2 -2
  131. data/lib/{rbui → ruby_ui}/form/form_field_hint.rb +1 -1
  132. data/lib/{rbui → ruby_ui}/form/form_field_label.rb +1 -1
  133. data/lib/{rbui → ruby_ui}/hover_card/hover_card.rb +3 -3
  134. data/lib/{rbui → ruby_ui}/hover_card/hover_card_content.rb +2 -2
  135. data/lib/ruby_ui/hover_card/hover_card_controller.js +144 -0
  136. data/lib/{rbui → ruby_ui}/hover_card/hover_card_trigger.rb +2 -2
  137. data/lib/{rbui → ruby_ui}/input/input.rb +3 -3
  138. data/lib/{rbui → ruby_ui}/link/link.rb +13 -13
  139. data/lib/ruby_ui/masked_input/masked_input.rb +15 -0
  140. data/lib/ruby_ui/masked_input/masked_input_controller.js +9 -0
  141. data/lib/{rbui → ruby_ui}/pagination/pagination.rb +1 -1
  142. data/lib/{rbui → ruby_ui}/pagination/pagination_content.rb +1 -1
  143. data/lib/{rbui → ruby_ui}/pagination/pagination_ellipsis.rb +1 -1
  144. data/lib/{rbui → ruby_ui}/pagination/pagination_item.rb +4 -4
  145. data/lib/{rbui → ruby_ui}/popover/popover.rb +4 -4
  146. data/lib/{rbui → ruby_ui}/popover/popover_content.rb +2 -2
  147. data/lib/ruby_ui/popover/popover_controller.js +107 -0
  148. data/lib/{rbui → ruby_ui}/popover/popover_trigger.rb +2 -2
  149. data/lib/ruby_ui/progress/progress.rb +37 -0
  150. data/lib/ruby_ui/radio_button/radio_button.rb +25 -0
  151. data/lib/{rbui → ruby_ui}/select/select.rb +5 -5
  152. data/lib/{rbui → ruby_ui}/select/select_content.rb +3 -3
  153. data/lib/ruby_ui/select/select_controller.js +171 -0
  154. data/lib/{rbui → ruby_ui}/select/select_group.rb +1 -1
  155. data/lib/{rbui → ruby_ui}/select/select_input.rb +4 -4
  156. data/lib/{rbui → ruby_ui}/select/select_item.rb +4 -4
  157. data/lib/ruby_ui/select/select_item_controller.js +11 -0
  158. data/lib/{rbui → ruby_ui}/select/select_label.rb +1 -1
  159. data/lib/{rbui → ruby_ui}/select/select_trigger.rb +3 -3
  160. data/lib/{rbui → ruby_ui}/select/select_value.rb +3 -3
  161. data/lib/ruby_ui/separator/separator.rb +38 -0
  162. data/lib/{rbui → ruby_ui}/sheet/sheet.rb +2 -2
  163. data/lib/{rbui → ruby_ui}/sheet/sheet_content.rb +8 -8
  164. data/lib/ruby_ui/sheet/sheet_content_controller.js +7 -0
  165. data/lib/ruby_ui/sheet/sheet_controller.js +9 -0
  166. data/lib/{rbui → ruby_ui}/sheet/sheet_description.rb +1 -1
  167. data/lib/{rbui → ruby_ui}/sheet/sheet_footer.rb +1 -1
  168. data/lib/{rbui → ruby_ui}/sheet/sheet_header.rb +1 -1
  169. data/lib/{rbui → ruby_ui}/sheet/sheet_middle.rb +1 -1
  170. data/lib/{rbui → ruby_ui}/sheet/sheet_title.rb +1 -1
  171. data/lib/{rbui → ruby_ui}/sheet/sheet_trigger.rb +2 -2
  172. data/lib/{rbui → ruby_ui}/shortcut_key/shortcut_key.rb +1 -1
  173. data/lib/ruby_ui/skeleton/skeleton.rb +17 -0
  174. data/lib/ruby_ui/switch/switch.rb +24 -0
  175. data/lib/{rbui → ruby_ui}/table/table.rb +1 -1
  176. data/lib/{rbui → ruby_ui}/table/table_body.rb +1 -1
  177. data/lib/{rbui → ruby_ui}/table/table_caption.rb +1 -1
  178. data/lib/{rbui → ruby_ui}/table/table_cell.rb +1 -1
  179. data/lib/{rbui → ruby_ui}/table/table_footer.rb +1 -1
  180. data/lib/{rbui → ruby_ui}/table/table_head.rb +1 -1
  181. data/lib/{rbui → ruby_ui}/table/table_header.rb +1 -1
  182. data/lib/{rbui → ruby_ui}/table/table_row.rb +1 -1
  183. data/lib/{rbui → ruby_ui}/tabs/tabs.rb +3 -3
  184. data/lib/{rbui → ruby_ui}/tabs/tabs_content.rb +2 -2
  185. data/lib/ruby_ui/tabs/tabs_controller.js +45 -0
  186. data/lib/{rbui → ruby_ui}/tabs/tabs_list.rb +1 -1
  187. data/lib/{rbui → ruby_ui}/tabs/tabs_trigger.rb +3 -3
  188. data/lib/{rbui → ruby_ui}/textarea/textarea.rb +3 -3
  189. data/lib/{rbui → ruby_ui}/theme_toggle/theme_toggle.rb +4 -4
  190. data/lib/ruby_ui/theme_toggle/theme_toggle_controller.js +30 -0
  191. data/lib/{rbui → ruby_ui}/tooltip/tooltip.rb +3 -3
  192. data/lib/{rbui → ruby_ui}/tooltip/tooltip_content.rb +3 -3
  193. data/lib/ruby_ui/tooltip/tooltip_controller.js +37 -0
  194. data/lib/{rbui → ruby_ui}/tooltip/tooltip_trigger.rb +2 -2
  195. data/lib/ruby_ui/typography/heading.rb +60 -0
  196. data/lib/{rbui/typography/typography_inline_code.rb → ruby_ui/typography/inline_code.rb} +2 -2
  197. data/lib/{rbui/typography/typography_inline_link.rb → ruby_ui/typography/inline_link.rb} +2 -2
  198. data/lib/ruby_ui/typography/text.rb +53 -0
  199. data/lib/{rbui → ruby_ui}/typography/typography_blockquote.rb +1 -1
  200. data/lib/ruby_ui.rb +5 -1
  201. metadata +208 -173
  202. data/lib/generators/rbui/base_generator.rb +0 -17
  203. data/lib/generators/rbui/component_generator.rb +0 -137
  204. data/lib/generators/rbui/install/install_generator.rb +0 -194
  205. data/lib/rbui/calendar/calendar.rb +0 -39
  206. data/lib/rbui/combobox/combobox.rb +0 -24
  207. data/lib/rbui/combobox/combobox_content.rb +0 -31
  208. data/lib/rbui/combobox/combobox_group.rb +0 -38
  209. data/lib/rbui/combobox/combobox_input.rb +0 -22
  210. data/lib/rbui/combobox/combobox_item.rb +0 -53
  211. data/lib/rbui/combobox/combobox_list.rb +0 -29
  212. data/lib/rbui/combobox/combobox_separator.rb +0 -15
  213. data/lib/rbui/combobox/combobox_value.rb +0 -27
  214. data/lib/rbui/radio_button/radio_button.rb +0 -22
  215. data/lib/rbui/railtie.rb +0 -52
  216. data/lib/rbui/typography/typography_h1.rb +0 -17
  217. data/lib/rbui/typography/typography_h2.rb +0 -17
  218. data/lib/rbui/typography/typography_h3.rb +0 -17
  219. data/lib/rbui/typography/typography_h4.rb +0 -17
  220. data/lib/rbui/typography/typography_large.rb +0 -17
  221. data/lib/rbui/typography/typography_lead.rb +0 -17
  222. data/lib/rbui/typography/typography_list.rb +0 -47
  223. data/lib/rbui/typography/typography_muted.rb +0 -17
  224. data/lib/rbui/typography/typography_p.rb +0 -17
  225. data/lib/rbui/typography/typography_small.rb +0 -17
  226. data/lib/rbui/version.rb +0 -5
  227. data/lib/rbui.rb +0 -57
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class PopoverContent < Base
5
5
  def view_template(&)
6
6
  div(**attrs, &)
@@ -11,7 +11,7 @@ module RBUI
11
11
  def default_attrs
12
12
  {
13
13
  data: {
14
- rbui__popover_target: "content"
14
+ ruby_ui__popover_target: "content"
15
15
  },
16
16
  class: [
17
17
  "hidden z-50 rounded-md border bg-background p-1 text-foreground shadow-md outline-none",
@@ -0,0 +1,107 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+ import {
3
+ computePosition,
4
+ flip,
5
+ shift,
6
+ offset,
7
+ autoUpdate,
8
+ } from "@floating-ui/dom";
9
+
10
+ export default class extends Controller {
11
+ static targets = ["trigger", "content"];
12
+ static values = {
13
+ open: { type: Boolean, default: false },
14
+ options: { type: Object, default: {} },
15
+ trigger: { type: String, default: "hover" },
16
+ };
17
+
18
+ connect() {
19
+ this.closeTimeout = null;
20
+ this.cleanup = null;
21
+ this.addEventListeners();
22
+ }
23
+
24
+ disconnect() {
25
+ this.removeEventListeners();
26
+ if (this.cleanup) {
27
+ this.cleanup();
28
+ }
29
+ }
30
+
31
+ addEventListeners() {
32
+ if (this.triggerValue === "hover") {
33
+ this.triggerTarget.addEventListener("mouseenter", this.handleMouseEnter);
34
+ this.triggerTarget.addEventListener("mouseleave", this.handleMouseLeave);
35
+ this.contentTarget.addEventListener("mouseenter", this.handleMouseEnter);
36
+ this.contentTarget.addEventListener("mouseleave", this.handleMouseLeave);
37
+ } else if (this.triggerValue === "click") {
38
+ this.triggerTarget.addEventListener("click", this.handleClick);
39
+ document.addEventListener("click", this.handleOutsideClick);
40
+ }
41
+ }
42
+
43
+ removeEventListeners() {
44
+ this.triggerTarget.removeEventListener("mouseenter", this.handleMouseEnter);
45
+ this.triggerTarget.removeEventListener("mouseleave", this.handleMouseLeave);
46
+ this.contentTarget.removeEventListener("mouseenter", this.handleMouseEnter);
47
+ this.contentTarget.removeEventListener("mouseleave", this.handleMouseLeave);
48
+ this.triggerTarget.removeEventListener("click", this.handleClick);
49
+ document.removeEventListener("click", this.handleOutsideClick);
50
+ }
51
+
52
+ handleMouseEnter = () => {
53
+ clearTimeout(this.closeTimeout);
54
+ this.openValue = true;
55
+ this.showPopover();
56
+ };
57
+
58
+ handleMouseLeave = () => {
59
+ this.closeTimeout = setTimeout(() => {
60
+ this.openValue = false;
61
+ this.hidePopover();
62
+ }, 100);
63
+ };
64
+
65
+ handleClick = (event) => {
66
+ event.stopPropagation();
67
+ this.openValue = !this.openValue;
68
+ this.openValue ? this.showPopover() : this.hidePopover();
69
+ };
70
+
71
+ handleOutsideClick = (event) => {
72
+ if (!this.element.contains(event.target) && this.openValue) {
73
+ this.openValue = false;
74
+ this.hidePopover();
75
+ }
76
+ };
77
+
78
+ showPopover() {
79
+ this.contentTarget.classList.remove("hidden");
80
+ this.updatePosition();
81
+ }
82
+
83
+ hidePopover() {
84
+ this.contentTarget.classList.add("hidden");
85
+ if (this.cleanup) {
86
+ this.cleanup();
87
+ }
88
+ }
89
+
90
+ updatePosition() {
91
+ if (this.cleanup) {
92
+ this.cleanup();
93
+ }
94
+
95
+ this.cleanup = autoUpdate(this.triggerTarget, this.contentTarget, () => {
96
+ computePosition(this.triggerTarget, this.contentTarget, {
97
+ placement: this.optionsValue.placement || "bottom",
98
+ middleware: [flip(), shift(), offset(8)],
99
+ }).then(({ x, y }) => {
100
+ Object.assign(this.contentTarget.style, {
101
+ left: `${x}px`,
102
+ top: `${y}px`,
103
+ });
104
+ });
105
+ });
106
+ }
107
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class PopoverTrigger < Base
5
5
  def view_template(&)
6
6
  div(**attrs, &)
@@ -11,7 +11,7 @@ module RBUI
11
11
  def default_attrs
12
12
  {
13
13
  data: {
14
- rbui__popover_target: "trigger"
14
+ ruby_ui__popover_target: "trigger"
15
15
  },
16
16
  class: "inline-block"
17
17
  }
@@ -0,0 +1,37 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RubyUI
4
+ class Progress < Base
5
+ def initialize(value: 0, **attrs)
6
+ @value = value.to_f.clamp(0, 100)
7
+
8
+ super(**attrs)
9
+ end
10
+
11
+ def view_template
12
+ div(**attrs) do
13
+ div(**indicator_attrs)
14
+ end
15
+ end
16
+
17
+ private
18
+
19
+ def default_attrs
20
+ {
21
+ role: "progressbar",
22
+ aria_valuenow: @value,
23
+ aria_valuemin: 0,
24
+ aria_valuemax: 100,
25
+ aria_valuetext: "#{@value}%",
26
+ class: "relative h-2 overflow-hidden rounded-full bg-primary/20 [&>*]:bg-primary"
27
+ }
28
+ end
29
+
30
+ def indicator_attrs
31
+ {
32
+ class: "h-full w-full flex-1",
33
+ style: "transform: translateX(-#{100 - @value}%)"
34
+ }
35
+ end
36
+ end
37
+ end
@@ -0,0 +1,25 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RubyUI
4
+ class RadioButton < Base
5
+ def view_template
6
+ input(**attrs)
7
+ end
8
+
9
+ private
10
+
11
+ def default_attrs
12
+ {
13
+ type: "radio",
14
+ data: {
15
+ ruby_ui__form_field_target: "input",
16
+ action: "change->ruby-ui--form-field#onInput invalid->ruby-ui--form-field#onInvalid"
17
+ },
18
+ class: [
19
+ "h-4 w-4 p-0 border-primary rounded-full flex-none",
20
+ "disabled:cursor-not-allowed disabled:opacity-50"
21
+ ]
22
+ }
23
+ end
24
+ end
25
+ end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Select < Base
5
5
  def view_template(&)
6
6
  div(**attrs, &)
@@ -11,10 +11,10 @@ module RBUI
11
11
  def default_attrs
12
12
  {
13
13
  data: {
14
- controller: "rbui--select",
15
- rbui__select_open_value: "false",
16
- action: "click@window->rbui--select#clickOutside",
17
- rbui__select_rbui__select_item_outlet: ".item"
14
+ controller: "ruby-ui--select",
15
+ ruby_ui__select_open_value: "false",
16
+ action: "click@window->ruby-ui--select#clickOutside",
17
+ ruby_ui__select_ruby_ui__select_item_outlet: ".item"
18
18
  },
19
19
  class: "group/select w-full relative"
20
20
  }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class SelectContent < Base
5
5
  def initialize(**attrs)
6
6
  @id = "content#{SecureRandom.hex(4)}"
@@ -10,7 +10,7 @@ module RBUI
10
10
  def view_template(&block)
11
11
  div(**attrs) do
12
12
  div(
13
- class: "max-h-96 min-w-max overflow-auto rounded-md border bg-background p-1 text-foreground shadow-md animate-out group-data-[rbui--select-open-value=true]/select:animate-in fade-out-0 group-data-[rbui--select-open-value=true]/select:fade-in-0 zoom-out-95 group-data-[rbui--select-open-value=true]/select:zoom-in-95 slide-in-from-top-2", &block
13
+ class: "max-h-96 w-full text-wrap overflow-auto rounded-md border bg-background p-1 text-foreground shadow-md animate-out group-data-[ruby-ui--select-open-value=true]/select:animate-in fade-out-0 group-data-[ruby-ui--select-open-value=true]/select:fade-in-0 zoom-out-95 group-data-[ruby-ui--select-open-value=true]/select:zoom-in-95 slide-in-from-top-2", &block
14
14
  )
15
15
  end
16
16
  end
@@ -23,7 +23,7 @@ module RBUI
23
23
  role: "listbox",
24
24
  tabindex: "-1",
25
25
  data: {
26
- rbui__select_target: "content"
26
+ ruby_ui__select_target: "content"
27
27
  },
28
28
  class: "hidden w-full absolute top-0 left-0 z-50"
29
29
  }
@@ -0,0 +1,171 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+ import { computePosition, autoUpdate, offset, flip } from "@floating-ui/dom";
3
+
4
+ export default class extends Controller {
5
+ static targets = ["trigger", "content", "input", "value", "item"];
6
+ static values = { open: Boolean };
7
+ static outlets = ["ruby-ui--select-item"];
8
+
9
+ constructor(...args) {
10
+ super(...args);
11
+ this.cleanup;
12
+ }
13
+
14
+ connect() {
15
+ this.setFloatingElement();
16
+ this.generateItemsIds();
17
+ }
18
+
19
+ disconnect() {
20
+ this.cleanup();
21
+ }
22
+
23
+ selectItem(event) {
24
+ event.preventDefault();
25
+
26
+ this.rubyUiSelectItemOutlets.forEach((item) =>
27
+ item.handleSelectItem(event),
28
+ );
29
+
30
+ const oldValue = this.inputTarget.value;
31
+ const newValue = event.target.dataset.value;
32
+
33
+ this.inputTarget.value = newValue;
34
+ this.valueTarget.innerText = event.target.innerText;
35
+
36
+ this.dispatchOnChange(oldValue, newValue);
37
+ this.closeContent();
38
+ }
39
+
40
+ onClick() {
41
+ this.toogleContent();
42
+
43
+ if (this.openValue) {
44
+ this.setFocusAndCurrent();
45
+ } else {
46
+ this.resetCurrent();
47
+ }
48
+ }
49
+
50
+ handleKeyDown(event) {
51
+ event.preventDefault();
52
+
53
+ const currentIndex = this.itemTargets.findIndex(
54
+ (item) => item.getAttribute("aria-current") === "true",
55
+ );
56
+
57
+ if (currentIndex + 1 < this.itemTargets.length) {
58
+ this.itemTargets[currentIndex].removeAttribute("aria-current");
59
+ this.setAriaCurrentAndActiveDescendant(currentIndex + 1);
60
+ }
61
+ }
62
+
63
+ handleKeyUp(event) {
64
+ event.preventDefault();
65
+
66
+ const currentIndex = this.itemTargets.findIndex(
67
+ (item) => item.getAttribute("aria-current") === "true",
68
+ );
69
+
70
+ if (currentIndex > 0) {
71
+ this.itemTargets[currentIndex].removeAttribute("aria-current");
72
+ this.setAriaCurrentAndActiveDescendant(currentIndex - 1);
73
+ }
74
+ }
75
+
76
+ handleEsc(event) {
77
+ event.preventDefault();
78
+ this.closeContent();
79
+ }
80
+
81
+ setFocusAndCurrent() {
82
+ const selectedItem = this.itemTargets.find(
83
+ (item) => item.getAttribute("aria-selected") === "true",
84
+ );
85
+
86
+ if (selectedItem) {
87
+ selectedItem.focus({ preventScroll: true });
88
+ selectedItem.setAttribute("aria-current", "true");
89
+ this.triggerTarget.setAttribute(
90
+ "aria-activedescendant",
91
+ selectedItem.getAttribute("id"),
92
+ );
93
+ } else {
94
+ this.itemTarget.focus({ preventScroll: true });
95
+ this.itemTarget.setAttribute("aria-current", "true");
96
+ this.triggerTarget.setAttribute(
97
+ "aria-activedescendant",
98
+ this.itemTarget.getAttribute("id"),
99
+ );
100
+ }
101
+ }
102
+
103
+ resetCurrent() {
104
+ this.itemTargets.forEach((item) => item.removeAttribute("aria-current"));
105
+ }
106
+
107
+ clickOutside(event) {
108
+ if (!this.openValue) return;
109
+ if (this.element.contains(event.target)) return;
110
+
111
+ event.preventDefault();
112
+ this.toogleContent();
113
+ }
114
+
115
+ toogleContent() {
116
+ this.openValue = !this.openValue;
117
+ this.contentTarget.classList.toggle("hidden");
118
+ this.triggerTarget.setAttribute("aria-expanded", this.openValue);
119
+ }
120
+
121
+ setFloatingElement() {
122
+ this.cleanup = autoUpdate(this.triggerTarget, this.contentTarget, () => {
123
+ computePosition(this.triggerTarget, this.contentTarget, {
124
+ middleware: [offset(4), flip()],
125
+ }).then(({ x, y }) => {
126
+ Object.assign(this.contentTarget.style, {
127
+ left: `${x}px`,
128
+ top: `${y}px`,
129
+ });
130
+ });
131
+ });
132
+ }
133
+
134
+ generateItemsIds() {
135
+ const contentId = this.contentTarget.getAttribute("id");
136
+ this.triggerTarget.setAttribute("aria-controls", contentId);
137
+
138
+ this.itemTargets.forEach((item, index) => {
139
+ item.id = `${contentId}-${index}`;
140
+ });
141
+ }
142
+
143
+ setAriaCurrentAndActiveDescendant(currentIndex) {
144
+ const currentItem = this.itemTargets[currentIndex];
145
+ currentItem.focus({ preventScroll: true });
146
+ currentItem.setAttribute("aria-current", "true");
147
+ this.triggerTarget.setAttribute(
148
+ "aria-activedescendant",
149
+ currentItem.getAttribute("id"),
150
+ );
151
+ }
152
+
153
+ closeContent() {
154
+ this.toogleContent();
155
+ this.resetCurrent();
156
+
157
+ this.triggerTarget.setAttribute("aria-activedescendant", true);
158
+ this.triggerTarget.focus({ preventScroll: true });
159
+ }
160
+
161
+ dispatchOnChange(oldValue, newValue) {
162
+ if (oldValue === newValue) return;
163
+
164
+ const event = new InputEvent("change", {
165
+ bubbles: true,
166
+ cancelable: true,
167
+ });
168
+
169
+ this.inputTarget.dispatchEvent(event);
170
+ }
171
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class SelectGroup < Base
5
5
  def view_template(&)
6
6
  div(**attrs, &)
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class SelectInput < Base
5
5
  def view_template
6
6
  input(**attrs)
@@ -12,9 +12,9 @@ module RBUI
12
12
  {
13
13
  class: "hidden",
14
14
  data: {
15
- rbui__select_target: "input",
16
- rbui__form_field_target: "input",
17
- action: "change->rbui--form-field#onChange invalid->rbui--form-field#onInvalid"
15
+ ruby_ui__select_target: "input",
16
+ ruby_ui__form_field_target: "input",
17
+ action: "change->ruby-ui--form-field#onChange invalid->ruby-ui--form-field#onInvalid"
18
18
  }
19
19
  }
20
20
  end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class SelectItem < Base
5
5
  def initialize(value: nil, **attrs)
6
6
  @value = value
@@ -39,9 +39,9 @@ module RBUI
39
39
  tabindex: "0",
40
40
  class: "item group relative flex cursor-pointer select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground focus:bg-accent focus:text-accent-foreground aria-selected:bg-accent aria-selected:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
41
41
  data: {
42
- controller: "rbui--select-item",
43
- action: "click->rbui--select#selectItem keydown.enter->rbui--select#selectItem keydown.down->rbui--select#handleKeyDown keydown.up->rbui--select#handleKeyUp keydown.esc->rbui--select#handleEsc",
44
- rbui__select_target: "item"
42
+ controller: "ruby-ui--select-item",
43
+ action: "click->ruby-ui--select#selectItem keydown.enter->ruby-ui--select#selectItem keydown.down->ruby-ui--select#handleKeyDown keydown.up->ruby-ui--select#handleKeyUp keydown.esc->ruby-ui--select#handleEsc",
44
+ ruby_ui__select_target: "item"
45
45
  },
46
46
  data_value: @value,
47
47
  data_orientation: "vertical",
@@ -0,0 +1,11 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+ export default class extends Controller {
3
+
4
+ handleSelectItem({ target }) {
5
+ if (this.element.dataset.value == target.dataset.value) {
6
+ this.element.setAttribute("aria-selected", true);
7
+ } else {
8
+ this.element.removeAttribute("aria-selected");
9
+ }
10
+ }
11
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class SelectLabel < Base
5
5
  def view_template(&)
6
6
  h3(**attrs, &)
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class SelectTrigger < Base
5
5
  def view_template(&block)
6
6
  button(**attrs) do
@@ -34,8 +34,8 @@ module RBUI
34
34
  def default_attrs
35
35
  {
36
36
  data: {
37
- action: "rbui--select#onClick",
38
- rbui__select_target: "trigger"
37
+ action: "ruby-ui--select#onClick",
38
+ ruby_ui__select_target: "trigger"
39
39
  },
40
40
  type: "button",
41
41
  role: "combobox",
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class SelectValue < Base
5
5
  def initialize(placeholder: nil, **attrs)
6
6
  @placeholder = placeholder
@@ -18,9 +18,9 @@ module RBUI
18
18
  def default_attrs
19
19
  {
20
20
  data: {
21
- rbui__select_target: "value"
21
+ ruby_ui__select_target: "value"
22
22
  },
23
- class: "pointer-events-none"
23
+ class: "truncate pointer-events-none"
24
24
  }
25
25
  end
26
26
  end
@@ -0,0 +1,38 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RubyUI
4
+ class Separator < Base
5
+ ORIENTATIONS = %i[horizontal vertical].freeze
6
+
7
+ def initialize(as: :div, orientation: :horizontal, decorative: true, **attrs)
8
+ raise ArgumentError, "Invalid orientation: #{orientation}" unless ORIENTATIONS.include?(orientation.to_sym)
9
+
10
+ @as = as.to_sym
11
+ @orientation = orientation.to_sym
12
+ @decorative = decorative
13
+ super(**attrs)
14
+ end
15
+
16
+ def view_template(&)
17
+ tag(@as, **attrs, &)
18
+ end
19
+
20
+ private
21
+
22
+ def default_attrs
23
+ {
24
+ role: (@decorative ? "none" : "separator"),
25
+ class: [
26
+ "shrink-0 bg-border",
27
+ orientation_classes
28
+ ]
29
+ }
30
+ end
31
+
32
+ def orientation_classes
33
+ return "h-[1px] w-full" if @orientation == :horizontal
34
+
35
+ "h-full w-[1px]"
36
+ end
37
+ end
38
+ end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Sheet < Base
5
5
  def view_template(&)
6
6
  div(**attrs, &)
@@ -10,7 +10,7 @@ module RBUI
10
10
 
11
11
  def default_attrs
12
12
  {
13
- data: {controller: "rbui--sheet"}
13
+ data: {controller: "ruby-ui--sheet"}
14
14
  }
15
15
  end
16
16
  end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class SheetContent < Base
5
5
  SIDE_CLASS = {
6
6
  top: "inset-x-0 top-0 border-b data-[state=closed]:slide-out-to-top data-[state=open]:slide-in-from-top",
@@ -16,8 +16,8 @@ module RBUI
16
16
  end
17
17
 
18
18
  def view_template(&block)
19
- template_tag(data: {rbui__sheet_target: "content"}) do
20
- div(data: {controller: "rbui--sheet-content"}) do
19
+ template(data: {ruby_ui__sheet_target: "content"}) do
20
+ div(data: {controller: "ruby-ui--sheet-content"}) do
21
21
  backdrop
22
22
  div(**attrs) do
23
23
  block&.call
@@ -32,18 +32,18 @@ module RBUI
32
32
  def default_attrs
33
33
  {
34
34
  data_state: "open", # For animate in
35
- class: tokens(
35
+ class: [
36
36
  "fixed pointer-events-auto z-50 gap-4 bg-background p-6 shadow-lg transition ease-in-out data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:duration-300 data-[state=open]:duration-500",
37
37
  @side_classes
38
- )
38
+ ]
39
39
  }
40
40
  end
41
41
 
42
42
  def close_button
43
43
  button(
44
44
  type: "button",
45
- class: "absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-accent data-[state=open]:text-muted-foreground",
46
- data_action: "click->rbui--sheet-content#close"
45
+ class: "absolute end-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-accent data-[state=open]:text-muted-foreground",
46
+ data_action: "click->ruby-ui--sheet-content#close"
47
47
  ) do
48
48
  svg(
49
49
  width: "15",
@@ -68,7 +68,7 @@ module RBUI
68
68
  def backdrop
69
69
  div(
70
70
  data_state: "open",
71
- data_action: "click->rbui--sheet-content#close",
71
+ data_action: "click->ruby-ui--sheet-content#close",
72
72
  class:
73
73
  "fixed pointer-events-auto inset-0 z-50 bg-background/80 backdrop-blur-sm data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0"
74
74
  )
@@ -0,0 +1,7 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ export default class extends Controller {
4
+ close() {
5
+ this.element.remove()
6
+ }
7
+ }
@@ -0,0 +1,9 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ export default class extends Controller {
4
+ static targets = ["content"]
5
+
6
+ open() {
7
+ document.body.insertAdjacentHTML("beforeend", this.contentTarget.innerHTML)
8
+ }
9
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class SheetDescription < Base
5
5
  def view_template(&)
6
6
  p(**attrs, &)
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class SheetFooter < Base
5
5
  def view_template(&)
6
6
  div(**attrs, &)