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.
- checksums.yaml +4 -4
- data/LICENSE.txt +21 -0
- data/README.md +85 -0
- data/lib/generators/ruby_ui/component_generator.rb +94 -0
- data/lib/generators/ruby_ui/dependencies.yml +74 -0
- data/lib/generators/ruby_ui/install/install_generator.rb +89 -0
- data/lib/generators/ruby_ui/install/templates/ruby_ui.rb.erb +18 -0
- data/lib/generators/ruby_ui/install/templates/tailwind.css.erb +156 -0
- data/lib/generators/ruby_ui/javascript_utils.rb +57 -0
- data/lib/{rbui → ruby_ui}/accordion/accordion.rb +1 -1
- data/lib/{rbui → ruby_ui}/accordion/accordion_content.rb +2 -2
- data/lib/ruby_ui/accordion/accordion_controller.js +97 -0
- data/lib/{rbui → ruby_ui}/accordion/accordion_default_content.rb +1 -1
- data/lib/{rbui → ruby_ui}/accordion/accordion_default_trigger.rb +3 -3
- data/lib/{rbui → ruby_ui}/accordion/accordion_icon.rb +2 -2
- data/lib/{rbui → ruby_ui}/accordion/accordion_item.rb +4 -4
- data/lib/{rbui → ruby_ui}/accordion/accordion_trigger.rb +3 -2
- data/lib/{rbui → ruby_ui}/alert/alert.rb +3 -3
- data/lib/{rbui → ruby_ui}/alert/alert_description.rb +1 -1
- data/lib/{rbui → ruby_ui}/alert/alert_title.rb +1 -1
- data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog.rb +3 -3
- data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_action.rb +2 -2
- data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_cancel.rb +3 -3
- data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_content.rb +5 -5
- data/lib/ruby_ui/alert_dialog/alert_dialog_controller.js +31 -0
- data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_description.rb +1 -1
- data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_footer.rb +2 -2
- data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_header.rb +2 -2
- data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_title.rb +1 -1
- data/lib/{rbui → ruby_ui}/alert_dialog/alert_dialog_trigger.rb +2 -2
- data/lib/{rbui → ruby_ui}/aspect_ratio/aspect_ratio.rb +1 -1
- data/lib/{rbui → ruby_ui}/avatar/avatar.rb +2 -2
- data/lib/{rbui → ruby_ui}/avatar/avatar_fallback.rb +1 -1
- data/lib/{rbui → ruby_ui}/avatar/avatar_image.rb +1 -1
- data/lib/{rbui → ruby_ui}/badge/badge.rb +2 -2
- data/lib/{rbui → ruby_ui}/base.rb +1 -8
- data/lib/ruby_ui/breadcrumb/breadcrumb.rb +17 -0
- data/lib/ruby_ui/breadcrumb/breadcrumb_ellipsis.rb +39 -0
- data/lib/{rbui/typography/typography_list_item.rb → ruby_ui/breadcrumb/breadcrumb_item.rb} +3 -3
- data/lib/ruby_ui/breadcrumb/breadcrumb_link.rb +22 -0
- data/lib/ruby_ui/breadcrumb/breadcrumb_list.rb +17 -0
- data/lib/ruby_ui/breadcrumb/breadcrumb_page.rb +19 -0
- data/lib/ruby_ui/breadcrumb/breadcrumb_separator.rb +38 -0
- data/lib/{rbui → ruby_ui}/button/button.rb +13 -13
- data/lib/ruby_ui/calendar/calendar.rb +39 -0
- data/lib/{rbui → ruby_ui}/calendar/calendar_body.rb +2 -2
- data/lib/ruby_ui/calendar/calendar_controller.js +249 -0
- data/lib/{rbui → ruby_ui}/calendar/calendar_days.rb +14 -14
- data/lib/{rbui → ruby_ui}/calendar/calendar_header.rb +1 -1
- data/lib/ruby_ui/calendar/calendar_input_controller.js +8 -0
- data/lib/{rbui → ruby_ui}/calendar/calendar_next.rb +2 -2
- data/lib/{rbui → ruby_ui}/calendar/calendar_prev.rb +2 -2
- data/lib/{rbui → ruby_ui}/calendar/calendar_title.rb +2 -2
- data/lib/{rbui → ruby_ui}/calendar/calendar_weekdays.rb +2 -2
- data/lib/{rbui → ruby_ui}/card/card.rb +1 -1
- data/lib/{rbui → ruby_ui}/card/card_content.rb +1 -1
- data/lib/{rbui → ruby_ui}/card/card_description.rb +1 -1
- data/lib/{rbui → ruby_ui}/card/card_footer.rb +1 -1
- data/lib/{rbui → ruby_ui}/card/card_header.rb +1 -1
- data/lib/{rbui → ruby_ui}/card/card_title.rb +1 -1
- data/lib/ruby_ui/carousel/carousel.rb +44 -0
- data/lib/ruby_ui/carousel/carousel_content.rb +23 -0
- data/lib/ruby_ui/carousel/carousel_controller.js +60 -0
- data/lib/ruby_ui/carousel/carousel_item.rb +23 -0
- data/lib/ruby_ui/carousel/carousel_next.rb +48 -0
- data/lib/ruby_ui/carousel/carousel_previous.rb +49 -0
- data/lib/{rbui → ruby_ui}/chart/chart.rb +3 -3
- data/lib/ruby_ui/chart/chart_controller.js +103 -0
- data/lib/{rbui → ruby_ui}/checkbox/checkbox.rb +4 -4
- data/lib/{rbui → ruby_ui}/checkbox/checkbox_group.rb +2 -2
- data/lib/ruby_ui/checkbox/checkbox_group_controller.js +21 -0
- data/lib/{rbui → ruby_ui}/clipboard/clipboard.rb +6 -6
- data/lib/ruby_ui/clipboard/clipboard_controller.js +54 -0
- data/lib/{rbui → ruby_ui}/clipboard/clipboard_popover.rb +2 -2
- data/lib/{rbui → ruby_ui}/clipboard/clipboard_source.rb +2 -2
- data/lib/{rbui → ruby_ui}/clipboard/clipboard_trigger.rb +3 -3
- data/lib/{rbui → ruby_ui}/codeblock/codeblock.rb +7 -10
- data/lib/{rbui → ruby_ui}/collapsible/collapsible.rb +3 -3
- data/lib/{rbui → ruby_ui}/collapsible/collapsible_content.rb +2 -2
- data/lib/ruby_ui/collapsible/collapsible_controller.js +47 -0
- data/lib/{rbui → ruby_ui}/collapsible/collapsible_trigger.rb +2 -2
- data/lib/ruby_ui/combobox/combobox.rb +26 -0
- data/lib/ruby_ui/combobox/combobox_checkbox.rb +25 -0
- data/lib/ruby_ui/combobox/combobox_controller.js +176 -0
- data/lib/{rbui/combobox/combobox_empty.rb → ruby_ui/combobox/combobox_empty_state.rb} +3 -3
- data/lib/ruby_ui/combobox/combobox_item.rb +25 -0
- data/lib/ruby_ui/combobox/combobox_list.rb +18 -0
- data/lib/ruby_ui/combobox/combobox_list_group.rb +20 -0
- data/lib/ruby_ui/combobox/combobox_popover.rb +30 -0
- data/lib/ruby_ui/combobox/combobox_radio.rb +26 -0
- data/lib/{rbui → ruby_ui}/combobox/combobox_search_input.rb +22 -25
- data/lib/ruby_ui/combobox/combobox_toggle_all_checkbox.rb +25 -0
- data/lib/{rbui → ruby_ui}/combobox/combobox_trigger.rb +26 -21
- data/lib/{rbui → ruby_ui}/command/command.rb +1 -1
- data/lib/ruby_ui/command/command_controller.js +136 -0
- data/lib/{rbui → ruby_ui}/command/command_dialog.rb +2 -2
- data/lib/{rbui → ruby_ui}/command/command_dialog_content.rb +6 -6
- data/lib/{rbui → ruby_ui}/command/command_dialog_trigger.rb +3 -3
- data/lib/{rbui → ruby_ui}/command/command_empty.rb +2 -2
- data/lib/{rbui → ruby_ui}/command/command_group.rb +2 -2
- data/lib/{rbui → ruby_ui}/command/command_input.rb +3 -3
- data/lib/{rbui → ruby_ui}/command/command_item.rb +2 -2
- data/lib/{rbui → ruby_ui}/command/command_list.rb +1 -1
- data/lib/{rbui → ruby_ui}/context_menu/context_menu.rb +2 -2
- data/lib/{rbui → ruby_ui}/context_menu/context_menu_content.rb +2 -2
- data/lib/ruby_ui/context_menu/context_menu_controller.js +144 -0
- data/lib/{rbui → ruby_ui}/context_menu/context_menu_item.rb +3 -3
- data/lib/{rbui → ruby_ui}/context_menu/context_menu_label.rb +2 -2
- data/lib/{rbui → ruby_ui}/context_menu/context_menu_separator.rb +1 -1
- data/lib/{rbui → ruby_ui}/context_menu/context_menu_trigger.rb +3 -3
- data/lib/{rbui → ruby_ui}/dialog/dialog.rb +3 -3
- data/lib/{rbui → ruby_ui}/dialog/dialog_content.rb +9 -9
- data/lib/ruby_ui/dialog/dialog_controller.js +32 -0
- data/lib/{rbui → ruby_ui}/dialog/dialog_description.rb +1 -1
- data/lib/{rbui → ruby_ui}/dialog/dialog_footer.rb +2 -2
- data/lib/{rbui → ruby_ui}/dialog/dialog_header.rb +2 -2
- data/lib/{rbui → ruby_ui}/dialog/dialog_middle.rb +1 -1
- data/lib/{rbui → ruby_ui}/dialog/dialog_title.rb +1 -1
- data/lib/{rbui → ruby_ui}/dialog/dialog_trigger.rb +2 -2
- data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu.rb +4 -4
- data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu_content.rb +2 -2
- data/lib/ruby_ui/dropdown_menu/dropdown_menu_controller.js +120 -0
- data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu_item.rb +3 -3
- data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu_label.rb +1 -1
- data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu_separator.rb +1 -1
- data/lib/{rbui → ruby_ui}/dropdown_menu/dropdown_menu_trigger.rb +2 -2
- data/lib/{rbui → ruby_ui}/form/form.rb +1 -1
- data/lib/{rbui → ruby_ui}/form/form_field.rb +2 -2
- data/lib/ruby_ui/form/form_field_controller.js +61 -0
- data/lib/{rbui → ruby_ui}/form/form_field_error.rb +2 -2
- data/lib/{rbui → ruby_ui}/form/form_field_hint.rb +1 -1
- data/lib/{rbui → ruby_ui}/form/form_field_label.rb +1 -1
- data/lib/{rbui → ruby_ui}/hover_card/hover_card.rb +3 -3
- data/lib/{rbui → ruby_ui}/hover_card/hover_card_content.rb +2 -2
- data/lib/ruby_ui/hover_card/hover_card_controller.js +144 -0
- data/lib/{rbui → ruby_ui}/hover_card/hover_card_trigger.rb +2 -2
- data/lib/{rbui → ruby_ui}/input/input.rb +3 -3
- data/lib/{rbui → ruby_ui}/link/link.rb +13 -13
- data/lib/ruby_ui/masked_input/masked_input.rb +15 -0
- data/lib/ruby_ui/masked_input/masked_input_controller.js +9 -0
- data/lib/{rbui → ruby_ui}/pagination/pagination.rb +1 -1
- data/lib/{rbui → ruby_ui}/pagination/pagination_content.rb +1 -1
- data/lib/{rbui → ruby_ui}/pagination/pagination_ellipsis.rb +1 -1
- data/lib/{rbui → ruby_ui}/pagination/pagination_item.rb +4 -4
- data/lib/{rbui → ruby_ui}/popover/popover.rb +4 -4
- data/lib/{rbui → ruby_ui}/popover/popover_content.rb +2 -2
- data/lib/ruby_ui/popover/popover_controller.js +107 -0
- data/lib/{rbui → ruby_ui}/popover/popover_trigger.rb +2 -2
- data/lib/ruby_ui/progress/progress.rb +37 -0
- data/lib/ruby_ui/radio_button/radio_button.rb +25 -0
- data/lib/{rbui → ruby_ui}/select/select.rb +5 -5
- data/lib/{rbui → ruby_ui}/select/select_content.rb +3 -3
- data/lib/ruby_ui/select/select_controller.js +171 -0
- data/lib/{rbui → ruby_ui}/select/select_group.rb +1 -1
- data/lib/{rbui → ruby_ui}/select/select_input.rb +4 -4
- data/lib/{rbui → ruby_ui}/select/select_item.rb +4 -4
- data/lib/ruby_ui/select/select_item_controller.js +11 -0
- data/lib/{rbui → ruby_ui}/select/select_label.rb +1 -1
- data/lib/{rbui → ruby_ui}/select/select_trigger.rb +3 -3
- data/lib/{rbui → ruby_ui}/select/select_value.rb +3 -3
- data/lib/ruby_ui/separator/separator.rb +38 -0
- data/lib/{rbui → ruby_ui}/sheet/sheet.rb +2 -2
- data/lib/{rbui → ruby_ui}/sheet/sheet_content.rb +8 -8
- data/lib/ruby_ui/sheet/sheet_content_controller.js +7 -0
- data/lib/ruby_ui/sheet/sheet_controller.js +9 -0
- data/lib/{rbui → ruby_ui}/sheet/sheet_description.rb +1 -1
- data/lib/{rbui → ruby_ui}/sheet/sheet_footer.rb +1 -1
- data/lib/{rbui → ruby_ui}/sheet/sheet_header.rb +1 -1
- data/lib/{rbui → ruby_ui}/sheet/sheet_middle.rb +1 -1
- data/lib/{rbui → ruby_ui}/sheet/sheet_title.rb +1 -1
- data/lib/{rbui → ruby_ui}/sheet/sheet_trigger.rb +2 -2
- data/lib/{rbui → ruby_ui}/shortcut_key/shortcut_key.rb +1 -1
- data/lib/ruby_ui/skeleton/skeleton.rb +17 -0
- data/lib/ruby_ui/switch/switch.rb +24 -0
- data/lib/{rbui → ruby_ui}/table/table.rb +1 -1
- data/lib/{rbui → ruby_ui}/table/table_body.rb +1 -1
- data/lib/{rbui → ruby_ui}/table/table_caption.rb +1 -1
- data/lib/{rbui → ruby_ui}/table/table_cell.rb +1 -1
- data/lib/{rbui → ruby_ui}/table/table_footer.rb +1 -1
- data/lib/{rbui → ruby_ui}/table/table_head.rb +1 -1
- data/lib/{rbui → ruby_ui}/table/table_header.rb +1 -1
- data/lib/{rbui → ruby_ui}/table/table_row.rb +1 -1
- data/lib/{rbui → ruby_ui}/tabs/tabs.rb +3 -3
- data/lib/{rbui → ruby_ui}/tabs/tabs_content.rb +2 -2
- data/lib/ruby_ui/tabs/tabs_controller.js +45 -0
- data/lib/{rbui → ruby_ui}/tabs/tabs_list.rb +1 -1
- data/lib/{rbui → ruby_ui}/tabs/tabs_trigger.rb +3 -3
- data/lib/{rbui → ruby_ui}/textarea/textarea.rb +3 -3
- data/lib/{rbui → ruby_ui}/theme_toggle/theme_toggle.rb +4 -4
- data/lib/ruby_ui/theme_toggle/theme_toggle_controller.js +30 -0
- data/lib/{rbui → ruby_ui}/tooltip/tooltip.rb +3 -3
- data/lib/{rbui → ruby_ui}/tooltip/tooltip_content.rb +3 -3
- data/lib/ruby_ui/tooltip/tooltip_controller.js +37 -0
- data/lib/{rbui → ruby_ui}/tooltip/tooltip_trigger.rb +2 -2
- data/lib/ruby_ui/typography/heading.rb +60 -0
- data/lib/{rbui/typography/typography_inline_code.rb → ruby_ui/typography/inline_code.rb} +2 -2
- data/lib/{rbui/typography/typography_inline_link.rb → ruby_ui/typography/inline_link.rb} +2 -2
- data/lib/ruby_ui/typography/text.rb +53 -0
- data/lib/{rbui → ruby_ui}/typography/typography_blockquote.rb +1 -1
- data/lib/ruby_ui.rb +5 -1
- metadata +208 -173
- data/lib/generators/rbui/base_generator.rb +0 -17
- data/lib/generators/rbui/component_generator.rb +0 -137
- data/lib/generators/rbui/install/install_generator.rb +0 -194
- data/lib/rbui/calendar/calendar.rb +0 -39
- data/lib/rbui/combobox/combobox.rb +0 -24
- data/lib/rbui/combobox/combobox_content.rb +0 -31
- data/lib/rbui/combobox/combobox_group.rb +0 -38
- data/lib/rbui/combobox/combobox_input.rb +0 -22
- data/lib/rbui/combobox/combobox_item.rb +0 -53
- data/lib/rbui/combobox/combobox_list.rb +0 -29
- data/lib/rbui/combobox/combobox_separator.rb +0 -15
- data/lib/rbui/combobox/combobox_value.rb +0 -27
- data/lib/rbui/radio_button/radio_button.rb +0 -22
- data/lib/rbui/railtie.rb +0 -52
- data/lib/rbui/typography/typography_h1.rb +0 -17
- data/lib/rbui/typography/typography_h2.rb +0 -17
- data/lib/rbui/typography/typography_h3.rb +0 -17
- data/lib/rbui/typography/typography_h4.rb +0 -17
- data/lib/rbui/typography/typography_large.rb +0 -17
- data/lib/rbui/typography/typography_lead.rb +0 -17
- data/lib/rbui/typography/typography_list.rb +0 -47
- data/lib/rbui/typography/typography_muted.rb +0 -17
- data/lib/rbui/typography/typography_p.rb +0 -17
- data/lib/rbui/typography/typography_small.rb +0 -17
- data/lib/rbui/version.rb +0 -5
- 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
|
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->
|
22
|
-
|
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
|
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: {
|
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
|
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: "
|
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
|
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
|
-
|
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
|
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: "
|
22
|
-
|
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
|
3
|
+
module RubyUI
|
4
4
|
class HoverCardContent < Base
|
5
5
|
def view_template(&block)
|
6
|
-
|
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
|
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
|
-
|
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
|
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
|
-
|
20
|
-
action: "input->
|
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
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
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:
|
23
|
-
|
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
|
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: "
|
20
|
-
|
21
|
-
|
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
|