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
@@ -0,0 +1,120 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+ import { computePosition, flip, shift, offset } from "@floating-ui/dom";
3
+
4
+ export default class extends Controller {
5
+ static targets = ["trigger", "content", "menuItem"];
6
+ static values = {
7
+ open: {
8
+ type: Boolean,
9
+ default: false,
10
+ },
11
+ options: {
12
+ type: Object,
13
+ default: {},
14
+ },
15
+ }
16
+
17
+ connect() {
18
+ this.boundHandleKeydown = this.#handleKeydown.bind(this); // Bind the function so we can remove it later
19
+ this.selectedIndex = -1;
20
+ }
21
+
22
+ #computeTooltip() {
23
+ computePosition(this.triggerTarget, this.contentTarget, {
24
+ placement: this.optionsValue.placement || "top",
25
+ middleware: [flip(), shift(), offset(8)],
26
+ }).then(({ x, y }) => {
27
+ Object.assign(this.contentTarget.style, {
28
+ left: `${x}px`,
29
+ top: `${y}px`,
30
+ });
31
+ });
32
+ }
33
+
34
+ onClickOutside(event) {
35
+ if (!this.openValue) return;
36
+ if (this.element.contains(event.target)) return;
37
+
38
+ event.preventDefault();
39
+ this.close();
40
+ }
41
+
42
+ toggle() {
43
+ this.contentTarget.classList.contains("hidden") ? this.#open() : this.close();
44
+ }
45
+
46
+ #open() {
47
+ this.openValue = true;
48
+ this.#deselectAll();
49
+ this.#addEventListeners();
50
+ this.#computeTooltip()
51
+ this.contentTarget.classList.remove("hidden");
52
+ }
53
+
54
+ close() {
55
+ this.openValue = false;
56
+ this.#removeEventListeners();
57
+ this.contentTarget.classList.add("hidden");
58
+ }
59
+
60
+ #handleKeydown(e) {
61
+ // return if no menu items (one line fix for)
62
+ if (this.menuItemTargets.length === 0) { return; }
63
+
64
+ if (e.key === 'ArrowDown') {
65
+ e.preventDefault();
66
+ this.#updateSelectedItem(1);
67
+ } else if (e.key === 'ArrowUp') {
68
+ e.preventDefault();
69
+ this.#updateSelectedItem(-1);
70
+ } else if (e.key === 'Enter' && this.selectedIndex !== -1) {
71
+ e.preventDefault();
72
+ this.menuItemTargets[this.selectedIndex].click();
73
+ }
74
+ }
75
+
76
+ #updateSelectedItem(direction) {
77
+ // Check if any of the menuItemTargets have aria-selected="true" and set the selectedIndex to that index
78
+ this.menuItemTargets.forEach((item, index) => {
79
+ if (item.getAttribute('aria-selected') === 'true') {
80
+ this.selectedIndex = index;
81
+ }
82
+ });
83
+
84
+ if (this.selectedIndex >= 0) {
85
+ this.#toggleAriaSelected(this.menuItemTargets[this.selectedIndex], false);
86
+ }
87
+
88
+ this.selectedIndex += direction;
89
+
90
+ if (this.selectedIndex < 0) {
91
+ this.selectedIndex = this.menuItemTargets.length - 1;
92
+ } else if (this.selectedIndex >= this.menuItemTargets.length) {
93
+ this.selectedIndex = 0;
94
+ }
95
+
96
+ this.#toggleAriaSelected(this.menuItemTargets[this.selectedIndex], true);
97
+ }
98
+
99
+ #toggleAriaSelected(element, isSelected) {
100
+ // Add or remove attribute
101
+ if (isSelected) {
102
+ element.setAttribute('aria-selected', 'true');
103
+ } else {
104
+ element.removeAttribute('aria-selected');
105
+ }
106
+ }
107
+
108
+ #deselectAll() {
109
+ this.menuItemTargets.forEach(item => this.#toggleAriaSelected(item, false));
110
+ this.selectedIndex = -1;
111
+ }
112
+
113
+ #addEventListeners() {
114
+ document.addEventListener('keydown', this.boundHandleKeydown);
115
+ }
116
+
117
+ #removeEventListeners() {
118
+ document.removeEventListener('keydown', this.boundHandleKeydown);
119
+ }
120
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class DropdownMenuItem < Base
5
5
  def initialize(href: "#", **attrs)
6
6
  @href = href
@@ -18,8 +18,8 @@ module RBUI
18
18
  href: @href,
19
19
  role: "menuitem",
20
20
  class: "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",
21
- data_action: "click->rbui--dropdown-menu#close",
22
- data_rbui__dropdown_menu_target: "menuItem",
21
+ data_action: "click->ruby-ui--dropdown-menu#close",
22
+ data_ruby_ui__dropdown_menu_target: "menuItem",
23
23
  tabindex: "-1",
24
24
  data_orientation: "vertical"
25
25
  }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class DropdownMenuLabel < 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 DropdownMenuSeparator < 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 DropdownMenuTrigger < 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: {rbui__dropdown_menu_target: "trigger", action: "click->rbui--dropdown-menu#toggle"},
13
+ data: {ruby_ui__dropdown_menu_target: "trigger", action: "click->ruby-ui--dropdown-menu#toggle"},
14
14
  class: "inline-block"
15
15
  }
16
16
  end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Form < Base
5
5
  def view_template(&)
6
6
  form(**attrs, &)
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class FormField < 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
- controller: "rbui--form-field"
14
+ controller: "ruby-ui--form-field"
15
15
  },
16
16
  class: "space-y-2"
17
17
  }
@@ -0,0 +1,61 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+
3
+ export default class extends Controller {
4
+ static targets = ["input", "error"];
5
+ static values = { shouldValidate: false };
6
+
7
+ connect() {
8
+ if (this.hasErrorTarget) {
9
+ if (this.errorTarget.textContent) {
10
+ this.shouldValidateValue = true;
11
+ } else {
12
+ this.errorTarget.classList.add("hidden");
13
+ }
14
+ }
15
+ }
16
+
17
+ onInvalid(error) {
18
+ error.preventDefault();
19
+
20
+ this.shouldValidateValue = true;
21
+ this.#setErrorMessage();
22
+ }
23
+
24
+ onInput() {
25
+ this.#setErrorMessage();
26
+ }
27
+
28
+ onChange() {
29
+ this.#setErrorMessage();
30
+ }
31
+
32
+ #setErrorMessage() {
33
+ if (!this.shouldValidateValue) return;
34
+
35
+ if (this.inputTarget.validity.valid) {
36
+ this.errorTarget.textContent = "";
37
+ this.errorTarget.classList.add("hidden");
38
+ } else {
39
+ this.errorTarget.textContent = this.#getValidationMessage();
40
+ this.errorTarget.classList.remove("hidden");
41
+ }
42
+ }
43
+
44
+ #getValidationMessage() {
45
+ let errorMessage;
46
+
47
+ const { validity, dataset, validationMessage } = this.inputTarget;
48
+
49
+ if (validity.tooLong) errorMessage = dataset.tooLong;
50
+ if (validity.tooShort) errorMessage = dataset.tooShort;
51
+ if (validity.badInput) errorMessage = dataset.badInput;
52
+ if (validity.typeMismatch) errorMessage = dataset.typeMismatch;
53
+ if (validity.stepMismatch) errorMessage = dataset.stepMismatch;
54
+ if (validity.valueMissing) errorMessage = dataset.valueMissing;
55
+ if (validity.rangeOverflow) errorMessage = dataset.rangeOverflow;
56
+ if (validity.rangeUnderflow) errorMessage = dataset.rangeUnderflow;
57
+ if (validity.patternMismatch) errorMessage = dataset.patternMismatch;
58
+
59
+ return errorMessage || validationMessage;
60
+ }
61
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class FormFieldError < Base
5
5
  def view_template(&)
6
6
  p(**attrs, &)
@@ -11,7 +11,7 @@ module RBUI
11
11
  def default_attrs
12
12
  {
13
13
  data: {
14
- rbui__form_field_target: "error"
14
+ ruby_ui__form_field_target: "error"
15
15
  },
16
16
  class: "text-sm font-medium text-destructive"
17
17
  }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class FormFieldHint < 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 FormFieldLabel < Base
5
5
  def view_template(&)
6
6
  label(**attrs, &)
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class HoverCard < Base
5
5
  def initialize(option: {}, **attrs)
6
6
  @options = option
@@ -18,8 +18,8 @@ module RBUI
18
18
  def default_attrs
19
19
  {
20
20
  data: {
21
- controller: "rbui--hover-card",
22
- rbui__hover_card_options_value: @options.to_json
21
+ controller: "ruby-ui--hover-card",
22
+ ruby_ui__hover_card_options_value: @options.to_json
23
23
  }
24
24
  }
25
25
  end
@@ -1,9 +1,9 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class HoverCardContent < Base
5
5
  def view_template(&block)
6
- template_tag(data: {rbui__hover_card_target: "content"}) do
6
+ template(data: {ruby_ui__hover_card_target: "content"}) do
7
7
  div(**attrs, &block)
8
8
  end
9
9
  end
@@ -0,0 +1,144 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+ import tippy from "tippy.js";
3
+
4
+ export default class extends Controller {
5
+ static targets = ["trigger", "content", "menuItem"];
6
+ static values = {
7
+ options: {
8
+ type: Object,
9
+ default: {},
10
+ },
11
+ // make content width of the trigger element (true/false)
12
+ matchWidth: {
13
+ type: Boolean,
14
+ default: false,
15
+ }
16
+ }
17
+
18
+ connect() {
19
+ this.boundHandleKeydown = this.handleKeydown.bind(this); // Bind the function so we can remove it later
20
+ this.initializeTippy();
21
+ this.selectedIndex = -1;
22
+ }
23
+
24
+ disconnect() {
25
+ this.destroyTippy();
26
+ }
27
+
28
+ initializeTippy() {
29
+ const defaultOptions = {
30
+ content: this.contentTarget.innerHTML,
31
+ allowHTML: true,
32
+ interactive: true,
33
+ onShow: (instance) => {
34
+ this.matchWidthValue && this.setContentWidth(instance); // ensure content width matches trigger width
35
+ this.addEventListeners();
36
+ },
37
+ onHide: () => {
38
+ this.removeEventListeners();
39
+ this.deselectAll();
40
+ },
41
+ popperOptions: {
42
+ modifiers: [
43
+ {
44
+ name: "offset",
45
+ options: {
46
+ offset: [0, 4]
47
+ },
48
+ },
49
+ ],
50
+ }
51
+ };
52
+
53
+ const mergedOptions = { ...this.optionsValue, ...defaultOptions };
54
+ this.tippy = tippy(this.triggerTarget, mergedOptions);
55
+ }
56
+
57
+ destroyTippy() {
58
+ if (this.tippy) {
59
+ this.tippy.destroy();
60
+ }
61
+ }
62
+
63
+ setContentWidth(instance) {
64
+ // box-sizing: border-box
65
+ const content = instance.popper.querySelector('.tippy-content');
66
+ if (content) {
67
+ content.style.width = `${instance.reference.offsetWidth}px`;
68
+ }
69
+ }
70
+
71
+ handleContextMenu(event) {
72
+ event.preventDefault();
73
+ this.open();
74
+ }
75
+
76
+ open() {
77
+ this.tippy.show();
78
+ }
79
+
80
+ close() {
81
+ this.tippy.hide();
82
+ }
83
+
84
+ handleKeydown(e) {
85
+ // return if no menu items (one line fix for)
86
+ if (this.menuItemTargets.length === 0) { return; }
87
+
88
+ if (e.key === 'ArrowDown') {
89
+ e.preventDefault();
90
+ this.updateSelectedItem(1);
91
+ } else if (e.key === 'ArrowUp') {
92
+ e.preventDefault();
93
+ this.updateSelectedItem(-1);
94
+ } else if (e.key === 'Enter' && this.selectedIndex !== -1) {
95
+ e.preventDefault();
96
+ this.menuItemTargets[this.selectedIndex].click();
97
+ }
98
+ }
99
+
100
+ updateSelectedItem(direction) {
101
+ // Check if any of the menuItemTargets have aria-selected="true" and set the selectedIndex to that index
102
+ this.menuItemTargets.forEach((item, index) => {
103
+ if (item.getAttribute('aria-selected') === 'true') {
104
+ this.selectedIndex = index;
105
+ }
106
+ });
107
+
108
+ if (this.selectedIndex >= 0) {
109
+ this.toggleAriaSelected(this.menuItemTargets[this.selectedIndex], false);
110
+ }
111
+
112
+ this.selectedIndex += direction;
113
+
114
+ if (this.selectedIndex < 0) {
115
+ this.selectedIndex = this.menuItemTargets.length - 1;
116
+ } else if (this.selectedIndex >= this.menuItemTargets.length) {
117
+ this.selectedIndex = 0;
118
+ }
119
+
120
+ this.toggleAriaSelected(this.menuItemTargets[this.selectedIndex], true);
121
+ }
122
+
123
+ toggleAriaSelected(element, isSelected) {
124
+ // Add or remove attribute
125
+ if (isSelected) {
126
+ element.setAttribute('aria-selected', 'true');
127
+ } else {
128
+ element.removeAttribute('aria-selected');
129
+ }
130
+ }
131
+
132
+ deselectAll() {
133
+ this.menuItemTargets.forEach(item => this.toggleAriaSelected(item, false));
134
+ this.selectedIndex = -1;
135
+ }
136
+
137
+ addEventListeners() {
138
+ document.addEventListener('keydown', this.boundHandleKeydown);
139
+ }
140
+
141
+ removeEventListeners() {
142
+ document.removeEventListener('keydown', this.boundHandleKeydown);
143
+ }
144
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class HoverCardTrigger < 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__hover_card_target: "trigger"
14
+ ruby_ui__hover_card_target: "trigger"
15
15
  },
16
16
  class: "inline-block"
17
17
  }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Input < Base
5
5
  def initialize(type: :string, **attrs)
6
6
  @type = type.to_sym
@@ -16,8 +16,8 @@ module RBUI
16
16
  def default_attrs
17
17
  {
18
18
  data: {
19
- rbui__form_field_target: "input",
20
- action: "input->rbui--form-field#onInput invalid->rbui--form-field#onInvalid"
19
+ ruby_ui__form_field_target: "input",
20
+ action: "input->ruby-ui--form-field#onInput invalid->ruby-ui--form-field#onInvalid"
21
21
  },
22
22
  class: "flex h-9 w-full rounded-md border bg-background px-3 py-1 text-sm shadow-sm transition-colors file:border-0 file:bg-transparent file:text-sm file:font-medium focus-visible:outline-none focus-visible:ring-1 disabled:cursor-not-allowed disabled:opacity-50 border-border focus-visible:ring-ring placeholder:text-muted-foreground"
23
23
  }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Link < Base
5
5
  def initialize(href: "#", variant: :link, size: :md, icon: false, **attrs)
6
6
  @href = href
@@ -35,45 +35,45 @@ module RBUI
35
35
  end
36
36
 
37
37
  def primary_classes
38
- tokens(
38
+ [
39
39
  "whitespace-nowrap inline-flex items-center justify-center rounded-md font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 bg-primary text-primary-foreground shadow hover:bg-primary/90",
40
40
  size_classes
41
- )
41
+ ]
42
42
  end
43
43
 
44
44
  def link_classes
45
- tokens(
45
+ [
46
46
  "whitespace-nowrap inline-flex items-center justify-center rounded-md font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 text-primary underline-offset-4 hover:underline",
47
47
  size_classes
48
- )
48
+ ]
49
49
  end
50
50
 
51
51
  def secondary_classes
52
- tokens(
52
+ [
53
53
  "whitespace-nowrap inline-flex items-center justify-center rounded-md font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 bg-secondary text-secondary-foreground hover:bg-opacity-80",
54
54
  size_classes
55
- )
55
+ ]
56
56
  end
57
57
 
58
58
  def destructive_classes
59
- tokens(
59
+ [
60
60
  "whitespace-nowrap inline-flex items-center justify-center rounded-md font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 bg-destructive text-destructive-foreground shadow-sm hover:bg-destructive/90",
61
61
  size_classes
62
- )
62
+ ]
63
63
  end
64
64
 
65
65
  def outline_classes
66
- tokens(
66
+ [
67
67
  "whitespace-nowrap inline-flex items-center justify-center rounded-md font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 border border-input bg-background shadow-sm hover:bg-accent hover:text-accent-foreground",
68
68
  size_classes
69
- )
69
+ ]
70
70
  end
71
71
 
72
72
  def ghost_classes
73
- tokens(
73
+ [
74
74
  "whitespace-nowrap inline-flex items-center justify-center rounded-md font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 hover:bg-accent hover:text-accent-foreground",
75
75
  size_classes
76
- )
76
+ ]
77
77
  end
78
78
 
79
79
  def default_classes
@@ -0,0 +1,15 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RubyUI
4
+ class MaskedInput < Base
5
+ def view_template
6
+ Input(type: "text", **attrs)
7
+ end
8
+
9
+ private
10
+
11
+ def default_attrs
12
+ {data: {controller: "ruby-ui--masked-input"}}
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,9 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+ import { MaskInput } from "maska";
3
+
4
+ // Connects to data-controller="ruby-ui--masked-input"
5
+ export default class extends Controller {
6
+ connect() {
7
+ new MaskInput(this.element)
8
+ }
9
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Pagination < Base
5
5
  def view_template(&)
6
6
  nav(**attrs, &)
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class PaginationContent < Base
5
5
  def view_template(&)
6
6
  ul(**attrs, &)
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class PaginationEllipsis < Base
5
5
  def view_template(&block)
6
6
  li do
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class PaginationItem < Base
5
5
  def initialize(href: "#", active: false, **attrs)
6
6
  @href = href
@@ -19,9 +19,9 @@ module RBUI
19
19
  def default_attrs
20
20
  {
21
21
  aria: {current: @active ? "page" : nil},
22
- class: tokens(
23
- RBUI::Button.new(variant: @active ? :outline : :ghost).attrs[:class]
24
- )
22
+ class: [
23
+ RubyUI::Button.new(variant: @active ? :outline : :ghost).attrs[:class]
24
+ ]
25
25
  }
26
26
  end
27
27
  end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Popover < Base
5
5
  def initialize(options: {}, **attrs)
6
6
  @options = options
@@ -16,9 +16,9 @@ module RBUI
16
16
  def default_attrs
17
17
  {
18
18
  data: {
19
- controller: "rbui--popover",
20
- rbui__popover_options_value: @options.to_json,
21
- rbui__popover_trigger_value: @options[:trigger] || "hover"
19
+ controller: "ruby-ui--popover",
20
+ ruby_ui__popover_options_value: @options.to_json,
21
+ ruby_ui__popover_trigger_value: @options[:trigger] || "hover"
22
22
  }
23
23
  }
24
24
  end