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,60 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+ import EmblaCarousel from 'embla-carousel'
3
+
4
+ const DEFAULT_OPTIONS = {
5
+ loop: true
6
+ }
7
+
8
+ export default class extends Controller {
9
+ static values = {
10
+ options: {
11
+ type: Object,
12
+ default: {},
13
+ }
14
+ }
15
+ static targets = ["viewport", "nextButton", "prevButton"]
16
+
17
+ connect() {
18
+ this.initCarousel(this.#mergedOptions)
19
+ }
20
+
21
+ disconnect() {
22
+ this.destroyCarousel()
23
+ }
24
+
25
+ initCarousel(options, plugins = []) {
26
+ this.carousel = EmblaCarousel(this.viewportTarget, options, plugins)
27
+
28
+ this.carousel.on("init", this.#updateControls.bind(this))
29
+ this.carousel.on("reInit", this.#updateControls.bind(this))
30
+ this.carousel.on("select", this.#updateControls.bind(this))
31
+ }
32
+
33
+ destroyCarousel() {
34
+ this.carousel.destroy()
35
+ }
36
+
37
+ scrollNext() {
38
+ this.carousel.scrollNext()
39
+ }
40
+
41
+ scrollPrev() {
42
+ this.carousel.scrollPrev()
43
+ }
44
+
45
+ #updateControls() {
46
+ this.#toggleButtonsDisabledState(this.nextButtonTargets, !this.carousel.canScrollNext())
47
+ this.#toggleButtonsDisabledState(this.prevButtonTargets, !this.carousel.canScrollPrev())
48
+ }
49
+
50
+ #toggleButtonsDisabledState(buttons, isDisabled) {
51
+ buttons.forEach((button) => button.disabled = isDisabled)
52
+ }
53
+
54
+ get #mergedOptions() {
55
+ return {
56
+ ...DEFAULT_OPTIONS,
57
+ ...this.optionsValue
58
+ }
59
+ }
60
+ }
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RubyUI
4
+ class CarouselItem < Base
5
+ def view_template(&)
6
+ div(**attrs, &)
7
+ end
8
+
9
+ private
10
+
11
+ def default_attrs
12
+ {
13
+ role: "group",
14
+ aria_roledescription: "slide",
15
+ class: [
16
+ "min-w-0 shrink-0 grow-0 basis-full",
17
+ "group-[.is-horizontal]:pl-4",
18
+ "group-[.is-vertical]:pt-4"
19
+ ]
20
+ }
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,48 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RubyUI
4
+ class CarouselNext < Base
5
+ def view_template(&)
6
+ Button(**attrs) do
7
+ icon
8
+ end
9
+ end
10
+
11
+ private
12
+
13
+ def default_attrs
14
+ {
15
+ variant: :outline,
16
+ icon: true,
17
+ class: [
18
+ "absolute h-8 w-8 rounded-full",
19
+ "group-[.is-horizontal]:-right-12 group-[.is-horizontal]:top-1/2 group-[.is-horizontal]:-translate-y-1/2",
20
+ "group-[.is-vertical]:-bottom-12 group-[.is-vertical]:left-1/2 group-[.is-vertical]:-translate-x-1/2 group-[.is-vertical]:rotate-90"
21
+ ],
22
+ disabled: true,
23
+ data: {
24
+ action: "click->ruby-ui--carousel#scrollNext",
25
+ ruby_ui__carousel_target: "nextButton"
26
+ }
27
+ }
28
+ end
29
+
30
+ def icon
31
+ svg(
32
+ width: "24",
33
+ height: "24",
34
+ viewBox: "0 0 24 24",
35
+ fill: "none",
36
+ stroke: "currentColor",
37
+ stroke_width: "2",
38
+ stroke_linecap: "round",
39
+ stroke_linejoin: "round",
40
+ xmlns: "http://www.w3.org/2000/svg",
41
+ class: "w-4 h-4"
42
+ ) do |s|
43
+ s.path(d: "M5 12h14")
44
+ s.path(d: "m12 5 7 7-7 7")
45
+ end
46
+ end
47
+ end
48
+ end
@@ -0,0 +1,49 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RubyUI
4
+ class CarouselPrevious < Base
5
+ def view_template(&)
6
+ Button(**attrs) do
7
+ icon
8
+ span(class: "sr-only") { "Next slide" }
9
+ end
10
+ end
11
+
12
+ private
13
+
14
+ def default_attrs
15
+ {
16
+ variant: :outline,
17
+ icon: true,
18
+ class: [
19
+ "absolute h-8 w-8 rounded-full",
20
+ "group-[.is-horizontal]:-left-12 group-[.is-horizontal]:top-1/2 group-[.is-horizontal]:-translate-y-1/2",
21
+ "group-[.is-vertical]:-top-12 group-[.is-vertical]:left-1/2 group-[.is-vertical]:-translate-x-1/2 group-[.is-vertical]:rotate-90"
22
+ ],
23
+ disabled: true,
24
+ data: {
25
+ action: "click->ruby-ui--carousel#scrollPrev",
26
+ ruby_ui__carousel_target: "prevButton"
27
+ }
28
+ }
29
+ end
30
+
31
+ def icon
32
+ svg(
33
+ width: "24",
34
+ height: "24",
35
+ viewBox: "0 0 24 24",
36
+ fill: "none",
37
+ stroke: "currentColor",
38
+ stroke_width: "2",
39
+ stroke_linecap: "round",
40
+ stroke_linejoin: "round",
41
+ xmlns: "http://www.w3.org/2000/svg",
42
+ class: "w-4 h-4"
43
+ ) do |s|
44
+ s.path(d: "m12 19-7-7 7-7")
45
+ s.path(d: "M19 12H5")
46
+ end
47
+ end
48
+ end
49
+ end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Chart < Base
5
5
  def initialize(options: {}, **attrs)
6
6
  @options = options.to_json
@@ -15,8 +15,8 @@ module RBUI
15
15
 
16
16
  def default_attrs
17
17
  {
18
- data_controller: "rbui--chart",
19
- data_rbui__chart_options_value: @options
18
+ data_controller: "ruby-ui--chart",
19
+ data_ruby_ui__chart_options_value: @options
20
20
  }
21
21
  end
22
22
  end
@@ -0,0 +1,103 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+ import Chart from 'chart.js/auto'
3
+
4
+ // Chart controller
5
+ export default class extends Controller {
6
+ static values = {
7
+ options: {
8
+ type: Object,
9
+ default: {},
10
+ }
11
+ }
12
+
13
+ // Function to initialize the chart when the controller is connected
14
+ connect() {
15
+ this.initDarkModeObserver()
16
+ this.initChart()
17
+ }
18
+
19
+ disconnect() {
20
+ this.darkModeObserver?.disconnect()
21
+ this.chart?.destroy()
22
+ }
23
+
24
+ // Function to initialize the chart
25
+ initChart() {
26
+ this.setColors()
27
+ const ctx = this.element.getContext('2d');
28
+ this.chart = new Chart(ctx, this.mergeOptionsWithDefaults());
29
+ }
30
+
31
+ setColors() {
32
+ this.setDefaultColorsForChart()
33
+ }
34
+
35
+ getThemeColor(name) {
36
+ const color = getComputedStyle(document.documentElement).getPropertyValue(`--${name}`)
37
+ const [hue, saturation, lightness] = color.split(' ')
38
+ return `hsl(${hue}, ${saturation}, ${lightness})`
39
+ }
40
+
41
+ defaultThemeColor() {
42
+ return {
43
+ backgroundColor: this.getThemeColor('background'),
44
+ hoverBackgroundColor: this.getThemeColor('accent'),
45
+ borderColor: this.getThemeColor('primary'),
46
+ borderWidth: 1,
47
+ }
48
+ }
49
+
50
+ // Function to set chart default colors
51
+ setDefaultColorsForChart() {
52
+ Chart.defaults.color = this.getThemeColor('muted-foreground') // font color
53
+ Chart.defaults.borderColor = this.getThemeColor('border') // border color
54
+ Chart.defaults.backgroundColor = this.getThemeColor('background') // background color
55
+
56
+ // tooltip colors
57
+ Chart.defaults.plugins.tooltip.backgroundColor = this.getThemeColor('background')
58
+ Chart.defaults.plugins.tooltip.borderColor = this.getThemeColor('border')
59
+ Chart.defaults.plugins.tooltip.titleColor = this.getThemeColor('foreground')
60
+ Chart.defaults.plugins.tooltip.bodyColor = this.getThemeColor('muted-foreground')
61
+ Chart.defaults.plugins.tooltip.borderWidth = 1
62
+
63
+ // legend
64
+ // options.plugins.legend.labels
65
+ Chart.defaults.plugins.legend.labels.boxWidth = 12
66
+ Chart.defaults.plugins.legend.labels.boxHeight = 12
67
+ Chart.defaults.plugins.legend.labels.borderWidth = 0
68
+ Chart.defaults.plugins.legend.labels.useBorderRadius = true
69
+ Chart.defaults.plugins.legend.labels.borderRadius = this.getThemeColor('radius')
70
+ }
71
+
72
+ // Function to refresh the chart
73
+ refreshChart() {
74
+ // Destroy the chart if it's a valid Chart.js instance
75
+ this.chart?.destroy()
76
+ // Reinitialize the chart
77
+ this.initChart()
78
+ }
79
+
80
+ // Function to initialize the dark mode observer
81
+ initDarkModeObserver() {
82
+ this.darkModeObserver = new MutationObserver(() => {
83
+ this.refreshChart()
84
+ })
85
+ this.darkModeObserver.observe(document.documentElement, { attributeFilter: ['class'] })
86
+ }
87
+
88
+ // Function to merge the options with the defaults
89
+ mergeOptionsWithDefaults() {
90
+ return {
91
+ ...this.optionsValue,
92
+ data: {
93
+ ...this.optionsValue.data,
94
+ datasets: this.optionsValue.data.datasets.map((dataset) => {
95
+ return {
96
+ ...this.defaultThemeColor(),
97
+ ...dataset,
98
+ }
99
+ })
100
+ }
101
+ }
102
+ }
103
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Checkbox < Base
5
5
  def view_template
6
6
  input(**attrs)
@@ -12,9 +12,9 @@ module RBUI
12
12
  {
13
13
  type: "checkbox",
14
14
  data: {
15
- rbui__form_field_target: "input",
16
- rbui__checkbox_group_target: "checkbox",
17
- action: "input->rbui--form-field#onInput invalid->rbui--form-field#onInvalid change->rbui--checkbox-group#onChange"
15
+ ruby_ui__form_field_target: "input",
16
+ ruby_ui__checkbox_group_target: "checkbox",
17
+ action: "change->ruby-ui--checkbox-group#onChange change->ruby-ui--form-field#onInput invalid->ruby-ui--form-field#onInvalid"
18
18
  },
19
19
  class: "peer h-4 w-4 shrink-0 rounded-sm border border-primary ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50 accent-primary"
20
20
  }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class CheckboxGroup < Base
5
5
  def view_template(&)
6
6
  div(**attrs, &)
@@ -12,7 +12,7 @@ module RBUI
12
12
  {
13
13
  role: "group",
14
14
  data: {
15
- controller: "rbui--checkbox-group"
15
+ controller: "ruby-ui--checkbox-group"
16
16
  }
17
17
  }
18
18
  end
@@ -0,0 +1,21 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+
3
+ export default class extends Controller {
4
+ static targets = ["checkbox"];
5
+
6
+ connect() {
7
+ this.#handleRequired();
8
+ }
9
+
10
+ onChange() {
11
+ this.#handleRequired();
12
+ }
13
+
14
+ #handleRequired() {
15
+ if (!this.element.hasAttribute("data-required")) return;
16
+
17
+ const checked = this.checkboxTargets.some(({ checked }) => checked);
18
+
19
+ this.checkboxTargets.forEach((checkbox) => (checkbox.required = !checked));
20
+ }
21
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Clipboard < Base
5
5
  def initialize(options: {}, success: "Copied!", error: "Copy Failed!", **attrs)
6
6
  @options = options
@@ -30,11 +30,11 @@ module RBUI
30
30
  def default_attrs
31
31
  {
32
32
  data: {
33
- controller: "rbui--clipboard",
34
- action: "click@window->rbui--clipboard#onClickOutside",
35
- rbui__clipboard_success_value: @success,
36
- rbui__clipboard_error_value: @error,
37
- rbui__clipboard_options_value: @options.to_json
33
+ controller: "ruby-ui--clipboard",
34
+ action: "click@window->ruby-ui--clipboard#onClickOutside",
35
+ ruby_ui__clipboard_success_value: @success,
36
+ ruby_ui__clipboard_error_value: @error,
37
+ ruby_ui__clipboard_options_value: @options.to_json
38
38
  }
39
39
  }
40
40
  end
@@ -0,0 +1,54 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+ import { computePosition, flip, shift } from "@floating-ui/dom";
3
+
4
+ // Connects to data-controller="accordion"
5
+ export default class extends Controller {
6
+ static targets = ['trigger', 'source', 'successPopover', 'errorPopover']
7
+ static values = {
8
+ options: {
9
+ type: Object,
10
+ default: {},
11
+ },
12
+ }
13
+
14
+ copy() {
15
+ let sourceElement = this.sourceTarget.children[0];
16
+ if (!sourceElement) {
17
+ this.showErrorPopover();
18
+ return;
19
+ }
20
+ let textToCopy = sourceElement.tagName === 'INPUT' ? sourceElement.value : sourceElement.innerText;
21
+ navigator.clipboard.writeText(textToCopy).then(() => {
22
+ this.#showSuccessPopover();
23
+ }).catch(() => {
24
+ this.#showErrorPopover();
25
+ })
26
+ }
27
+
28
+ onClickOutside() {
29
+ if (!this.successPopoverTarget.classList.contains("hidden")) this.successPopoverTarget.classList.add("hidden");
30
+ if (!this.errorPopoverTarget.classList.contains("hidden")) this.errorPopoverTarget.classList.add("hidden");
31
+ }
32
+
33
+ #computeTooltip(popoverElement) {
34
+ computePosition(this.triggerTarget, popoverElement, {
35
+ placement: this.optionsValue.placement || "top",
36
+ middleware: [flip(), shift()],
37
+ }).then(({ x, y }) => {
38
+ Object.assign(popoverElement.style, {
39
+ left: `${x}px`,
40
+ top: `${y}px`,
41
+ });
42
+ });
43
+ }
44
+
45
+ #showSuccessPopover() {
46
+ this.#computeTooltip(this.successPopoverTarget);
47
+ this.successPopoverTarget.classList.remove("hidden");
48
+ }
49
+
50
+ #showErrorPopover() {
51
+ this.#computeTooltip(this.errorPopoverTarget);
52
+ this.errorPopoverTarget.classList.remove("hidden");
53
+ }
54
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class ClipboardPopover < Base
5
5
  def initialize(type:, **attrs)
6
6
  @type = type
@@ -11,7 +11,7 @@ module RBUI
11
11
  div(
12
12
  class: "hidden",
13
13
  style: "width: max-content; position: absolute; top: 0; left: 0;",
14
- data: {rbui__clipboard_target: clipboard_target}
14
+ data: {ruby_ui__clipboard_target: clipboard_target}
15
15
  ) do
16
16
  div(**attrs, &block)
17
17
  end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class ClipboardSource < 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__clipboard_target: "source"
14
+ ruby_ui__clipboard_target: "source"
15
15
  }
16
16
  }
17
17
  end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class ClipboardTrigger < Base
5
5
  def view_template(&)
6
6
  div(**attrs, &)
@@ -11,8 +11,8 @@ module RBUI
11
11
  def default_attrs
12
12
  {
13
13
  data: {
14
- rbui__clipboard_target: "trigger",
15
- action: "click->rbui--clipboard#copy"
14
+ ruby_ui__clipboard_target: "trigger",
15
+ action: "click->ruby-ui--clipboard#copy"
16
16
  }
17
17
  }
18
18
  end
@@ -2,7 +2,7 @@
2
2
 
3
3
  require "rouge"
4
4
 
5
- module RBUI
5
+ module RubyUI
6
6
  class Codeblock < Base
7
7
  FORMATTER = ::Rouge::Formatters::HTML.new
8
8
  ROUGE_CSS = Rouge::Themes::Github.mode(:dark).render(scope: ".highlight") # See themes here: https://rouge-ruby.github.io/docs/Rouge/CSSTheme.html
@@ -34,18 +34,19 @@ module RBUI
34
34
 
35
35
  def default_attrs
36
36
  {
37
+ style: {tab_size: 2},
37
38
  class: "highlight text-sm max-h-[350px] after:content-none flex font-mono overflow-auto overflow-x rounded-md border !bg-stone-900 [&_pre]:p-4"
38
39
  }
39
40
  end
40
41
 
41
42
  def with_clipboard
42
- RBUI.Clipboard(success: @clipboard_success, error: @clipboard_error, class: "relative") do
43
- RBUI.ClipboardSource do
43
+ RubyUI.Clipboard(success: @clipboard_success, error: @clipboard_error, class: "relative") do
44
+ RubyUI.ClipboardSource do
44
45
  codeblock
45
46
  end
46
47
  div(class: "absolute top-2 right-2") do
47
- RBUI.ClipboardTrigger do
48
- RBUI.Button(variant: :ghost, size: :sm, icon: true, class: "text-white hover:text-white hover:bg-white/20") { clipboard_icon }
48
+ RubyUI.ClipboardTrigger do
49
+ RubyUI.Button(variant: :ghost, size: :sm, icon: true, class: "text-white hover:text-white hover:bg-white/20") { clipboard_icon }
49
50
  end
50
51
  end
51
52
  end
@@ -54,11 +55,7 @@ module RBUI
54
55
  def codeblock
55
56
  div(**attrs) do
56
57
  div(class: "after:content-none") do
57
- pre do
58
- unsafe_raw FORMATTER.format(
59
- lexer.lex(@code)
60
- )
61
- end
58
+ pre { raw(safe(FORMATTER.format(lexer.lex(@code)))) }
62
59
  end
63
60
  end
64
61
  end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class Collapsible < Base
5
5
  def initialize(open: false, **attrs)
6
6
  @open = open
@@ -16,8 +16,8 @@ module RBUI
16
16
  def default_attrs
17
17
  {
18
18
  data: {
19
- controller: "rbui--collapsible",
20
- rbui__collapsible_open_value: @open
19
+ controller: "ruby-ui--collapsible",
20
+ ruby_ui__collapsible_open_value: @open
21
21
  }
22
22
  }
23
23
  end
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class CollapsibleContent < 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__collapsible_target: "content"},
13
+ data: {ruby_ui__collapsible_target: "content"},
14
14
  class: "overflow-y-hidden"
15
15
  }
16
16
  end
@@ -0,0 +1,47 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ // Connects to data-controller="accordion"
4
+ export default class extends Controller {
5
+ static targets = ['content']
6
+ static values = {
7
+ open: {
8
+ type: Boolean,
9
+ default: false,
10
+ },
11
+ }
12
+
13
+ connect() {
14
+ // Set the initial state of the accordion
15
+ this.openValue ? this.open() : this.close()
16
+ }
17
+
18
+ // Toggle the 'open' value
19
+ toggle() {
20
+ this.openValue = !this.openValue
21
+ }
22
+
23
+ // Handle changes in the 'open' value
24
+ openValueChanged(isOpen, wasOpen) {
25
+ if (isOpen) {
26
+ this.open()
27
+ } else {
28
+ this.close()
29
+ }
30
+ }
31
+
32
+ // Open the accordion content
33
+ open() {
34
+ if (this.hasContentTarget) {
35
+ this.contentTarget.classList.remove('hidden')
36
+ this.openValue = true
37
+ }
38
+ }
39
+
40
+ // Close the accordion content
41
+ close() {
42
+ if (this.hasContentTarget) {
43
+ this.contentTarget.classList.add('hidden')
44
+ this.openValue = false
45
+ }
46
+ }
47
+ }
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RBUI
3
+ module RubyUI
4
4
  class CollapsibleTrigger < 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
- action: "click->rbui--collapsible#toggle"
14
+ action: "click->ruby-ui--collapsible#toggle"
15
15
  }
16
16
  }
17
17
  end
@@ -0,0 +1,26 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RubyUI
4
+ class Combobox < Base
5
+ def initialize(term: "items", **)
6
+ @term = term
7
+ super(**)
8
+ end
9
+
10
+ def view_template(&)
11
+ div(**attrs, &)
12
+ end
13
+
14
+ private
15
+
16
+ def default_attrs
17
+ {
18
+ role: "combobox",
19
+ data: {
20
+ controller: "ruby-ui--combobox",
21
+ ruby_ui__combobox_term_value: @term.to_s
22
+ }
23
+ }
24
+ end
25
+ end
26
+ end