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
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module RubyUI
|
4
4
|
class PopoverContent < Base
|
5
5
|
def view_template(&)
|
6
6
|
div(**attrs, &)
|
@@ -11,7 +11,7 @@ module RBUI
|
|
11
11
|
def default_attrs
|
12
12
|
{
|
13
13
|
data: {
|
14
|
-
|
14
|
+
ruby_ui__popover_target: "content"
|
15
15
|
},
|
16
16
|
class: [
|
17
17
|
"hidden z-50 rounded-md border bg-background p-1 text-foreground shadow-md outline-none",
|
@@ -0,0 +1,107 @@
|
|
1
|
+
import { Controller } from "@hotwired/stimulus";
|
2
|
+
import {
|
3
|
+
computePosition,
|
4
|
+
flip,
|
5
|
+
shift,
|
6
|
+
offset,
|
7
|
+
autoUpdate,
|
8
|
+
} from "@floating-ui/dom";
|
9
|
+
|
10
|
+
export default class extends Controller {
|
11
|
+
static targets = ["trigger", "content"];
|
12
|
+
static values = {
|
13
|
+
open: { type: Boolean, default: false },
|
14
|
+
options: { type: Object, default: {} },
|
15
|
+
trigger: { type: String, default: "hover" },
|
16
|
+
};
|
17
|
+
|
18
|
+
connect() {
|
19
|
+
this.closeTimeout = null;
|
20
|
+
this.cleanup = null;
|
21
|
+
this.addEventListeners();
|
22
|
+
}
|
23
|
+
|
24
|
+
disconnect() {
|
25
|
+
this.removeEventListeners();
|
26
|
+
if (this.cleanup) {
|
27
|
+
this.cleanup();
|
28
|
+
}
|
29
|
+
}
|
30
|
+
|
31
|
+
addEventListeners() {
|
32
|
+
if (this.triggerValue === "hover") {
|
33
|
+
this.triggerTarget.addEventListener("mouseenter", this.handleMouseEnter);
|
34
|
+
this.triggerTarget.addEventListener("mouseleave", this.handleMouseLeave);
|
35
|
+
this.contentTarget.addEventListener("mouseenter", this.handleMouseEnter);
|
36
|
+
this.contentTarget.addEventListener("mouseleave", this.handleMouseLeave);
|
37
|
+
} else if (this.triggerValue === "click") {
|
38
|
+
this.triggerTarget.addEventListener("click", this.handleClick);
|
39
|
+
document.addEventListener("click", this.handleOutsideClick);
|
40
|
+
}
|
41
|
+
}
|
42
|
+
|
43
|
+
removeEventListeners() {
|
44
|
+
this.triggerTarget.removeEventListener("mouseenter", this.handleMouseEnter);
|
45
|
+
this.triggerTarget.removeEventListener("mouseleave", this.handleMouseLeave);
|
46
|
+
this.contentTarget.removeEventListener("mouseenter", this.handleMouseEnter);
|
47
|
+
this.contentTarget.removeEventListener("mouseleave", this.handleMouseLeave);
|
48
|
+
this.triggerTarget.removeEventListener("click", this.handleClick);
|
49
|
+
document.removeEventListener("click", this.handleOutsideClick);
|
50
|
+
}
|
51
|
+
|
52
|
+
handleMouseEnter = () => {
|
53
|
+
clearTimeout(this.closeTimeout);
|
54
|
+
this.openValue = true;
|
55
|
+
this.showPopover();
|
56
|
+
};
|
57
|
+
|
58
|
+
handleMouseLeave = () => {
|
59
|
+
this.closeTimeout = setTimeout(() => {
|
60
|
+
this.openValue = false;
|
61
|
+
this.hidePopover();
|
62
|
+
}, 100);
|
63
|
+
};
|
64
|
+
|
65
|
+
handleClick = (event) => {
|
66
|
+
event.stopPropagation();
|
67
|
+
this.openValue = !this.openValue;
|
68
|
+
this.openValue ? this.showPopover() : this.hidePopover();
|
69
|
+
};
|
70
|
+
|
71
|
+
handleOutsideClick = (event) => {
|
72
|
+
if (!this.element.contains(event.target) && this.openValue) {
|
73
|
+
this.openValue = false;
|
74
|
+
this.hidePopover();
|
75
|
+
}
|
76
|
+
};
|
77
|
+
|
78
|
+
showPopover() {
|
79
|
+
this.contentTarget.classList.remove("hidden");
|
80
|
+
this.updatePosition();
|
81
|
+
}
|
82
|
+
|
83
|
+
hidePopover() {
|
84
|
+
this.contentTarget.classList.add("hidden");
|
85
|
+
if (this.cleanup) {
|
86
|
+
this.cleanup();
|
87
|
+
}
|
88
|
+
}
|
89
|
+
|
90
|
+
updatePosition() {
|
91
|
+
if (this.cleanup) {
|
92
|
+
this.cleanup();
|
93
|
+
}
|
94
|
+
|
95
|
+
this.cleanup = autoUpdate(this.triggerTarget, this.contentTarget, () => {
|
96
|
+
computePosition(this.triggerTarget, this.contentTarget, {
|
97
|
+
placement: this.optionsValue.placement || "bottom",
|
98
|
+
middleware: [flip(), shift(), offset(8)],
|
99
|
+
}).then(({ x, y }) => {
|
100
|
+
Object.assign(this.contentTarget.style, {
|
101
|
+
left: `${x}px`,
|
102
|
+
top: `${y}px`,
|
103
|
+
});
|
104
|
+
});
|
105
|
+
});
|
106
|
+
}
|
107
|
+
}
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module RubyUI
|
4
4
|
class PopoverTrigger < Base
|
5
5
|
def view_template(&)
|
6
6
|
div(**attrs, &)
|
@@ -11,7 +11,7 @@ module RBUI
|
|
11
11
|
def default_attrs
|
12
12
|
{
|
13
13
|
data: {
|
14
|
-
|
14
|
+
ruby_ui__popover_target: "trigger"
|
15
15
|
},
|
16
16
|
class: "inline-block"
|
17
17
|
}
|
@@ -0,0 +1,37 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module RubyUI
|
4
|
+
class Progress < Base
|
5
|
+
def initialize(value: 0, **attrs)
|
6
|
+
@value = value.to_f.clamp(0, 100)
|
7
|
+
|
8
|
+
super(**attrs)
|
9
|
+
end
|
10
|
+
|
11
|
+
def view_template
|
12
|
+
div(**attrs) do
|
13
|
+
div(**indicator_attrs)
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
private
|
18
|
+
|
19
|
+
def default_attrs
|
20
|
+
{
|
21
|
+
role: "progressbar",
|
22
|
+
aria_valuenow: @value,
|
23
|
+
aria_valuemin: 0,
|
24
|
+
aria_valuemax: 100,
|
25
|
+
aria_valuetext: "#{@value}%",
|
26
|
+
class: "relative h-2 overflow-hidden rounded-full bg-primary/20 [&>*]:bg-primary"
|
27
|
+
}
|
28
|
+
end
|
29
|
+
|
30
|
+
def indicator_attrs
|
31
|
+
{
|
32
|
+
class: "h-full w-full flex-1",
|
33
|
+
style: "transform: translateX(-#{100 - @value}%)"
|
34
|
+
}
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
@@ -0,0 +1,25 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module RubyUI
|
4
|
+
class RadioButton < Base
|
5
|
+
def view_template
|
6
|
+
input(**attrs)
|
7
|
+
end
|
8
|
+
|
9
|
+
private
|
10
|
+
|
11
|
+
def default_attrs
|
12
|
+
{
|
13
|
+
type: "radio",
|
14
|
+
data: {
|
15
|
+
ruby_ui__form_field_target: "input",
|
16
|
+
action: "change->ruby-ui--form-field#onInput invalid->ruby-ui--form-field#onInvalid"
|
17
|
+
},
|
18
|
+
class: [
|
19
|
+
"h-4 w-4 p-0 border-primary rounded-full flex-none",
|
20
|
+
"disabled:cursor-not-allowed disabled:opacity-50"
|
21
|
+
]
|
22
|
+
}
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module RubyUI
|
4
4
|
class Select < Base
|
5
5
|
def view_template(&)
|
6
6
|
div(**attrs, &)
|
@@ -11,10 +11,10 @@ module RBUI
|
|
11
11
|
def default_attrs
|
12
12
|
{
|
13
13
|
data: {
|
14
|
-
controller: "
|
15
|
-
|
16
|
-
action: "click@window->
|
17
|
-
|
14
|
+
controller: "ruby-ui--select",
|
15
|
+
ruby_ui__select_open_value: "false",
|
16
|
+
action: "click@window->ruby-ui--select#clickOutside",
|
17
|
+
ruby_ui__select_ruby_ui__select_item_outlet: ".item"
|
18
18
|
},
|
19
19
|
class: "group/select w-full relative"
|
20
20
|
}
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module RubyUI
|
4
4
|
class SelectContent < Base
|
5
5
|
def initialize(**attrs)
|
6
6
|
@id = "content#{SecureRandom.hex(4)}"
|
@@ -10,7 +10,7 @@ module RBUI
|
|
10
10
|
def view_template(&block)
|
11
11
|
div(**attrs) do
|
12
12
|
div(
|
13
|
-
class: "max-h-96
|
13
|
+
class: "max-h-96 w-full text-wrap overflow-auto rounded-md border bg-background p-1 text-foreground shadow-md animate-out group-data-[ruby-ui--select-open-value=true]/select:animate-in fade-out-0 group-data-[ruby-ui--select-open-value=true]/select:fade-in-0 zoom-out-95 group-data-[ruby-ui--select-open-value=true]/select:zoom-in-95 slide-in-from-top-2", &block
|
14
14
|
)
|
15
15
|
end
|
16
16
|
end
|
@@ -23,7 +23,7 @@ module RBUI
|
|
23
23
|
role: "listbox",
|
24
24
|
tabindex: "-1",
|
25
25
|
data: {
|
26
|
-
|
26
|
+
ruby_ui__select_target: "content"
|
27
27
|
},
|
28
28
|
class: "hidden w-full absolute top-0 left-0 z-50"
|
29
29
|
}
|
@@ -0,0 +1,171 @@
|
|
1
|
+
import { Controller } from "@hotwired/stimulus";
|
2
|
+
import { computePosition, autoUpdate, offset, flip } from "@floating-ui/dom";
|
3
|
+
|
4
|
+
export default class extends Controller {
|
5
|
+
static targets = ["trigger", "content", "input", "value", "item"];
|
6
|
+
static values = { open: Boolean };
|
7
|
+
static outlets = ["ruby-ui--select-item"];
|
8
|
+
|
9
|
+
constructor(...args) {
|
10
|
+
super(...args);
|
11
|
+
this.cleanup;
|
12
|
+
}
|
13
|
+
|
14
|
+
connect() {
|
15
|
+
this.setFloatingElement();
|
16
|
+
this.generateItemsIds();
|
17
|
+
}
|
18
|
+
|
19
|
+
disconnect() {
|
20
|
+
this.cleanup();
|
21
|
+
}
|
22
|
+
|
23
|
+
selectItem(event) {
|
24
|
+
event.preventDefault();
|
25
|
+
|
26
|
+
this.rubyUiSelectItemOutlets.forEach((item) =>
|
27
|
+
item.handleSelectItem(event),
|
28
|
+
);
|
29
|
+
|
30
|
+
const oldValue = this.inputTarget.value;
|
31
|
+
const newValue = event.target.dataset.value;
|
32
|
+
|
33
|
+
this.inputTarget.value = newValue;
|
34
|
+
this.valueTarget.innerText = event.target.innerText;
|
35
|
+
|
36
|
+
this.dispatchOnChange(oldValue, newValue);
|
37
|
+
this.closeContent();
|
38
|
+
}
|
39
|
+
|
40
|
+
onClick() {
|
41
|
+
this.toogleContent();
|
42
|
+
|
43
|
+
if (this.openValue) {
|
44
|
+
this.setFocusAndCurrent();
|
45
|
+
} else {
|
46
|
+
this.resetCurrent();
|
47
|
+
}
|
48
|
+
}
|
49
|
+
|
50
|
+
handleKeyDown(event) {
|
51
|
+
event.preventDefault();
|
52
|
+
|
53
|
+
const currentIndex = this.itemTargets.findIndex(
|
54
|
+
(item) => item.getAttribute("aria-current") === "true",
|
55
|
+
);
|
56
|
+
|
57
|
+
if (currentIndex + 1 < this.itemTargets.length) {
|
58
|
+
this.itemTargets[currentIndex].removeAttribute("aria-current");
|
59
|
+
this.setAriaCurrentAndActiveDescendant(currentIndex + 1);
|
60
|
+
}
|
61
|
+
}
|
62
|
+
|
63
|
+
handleKeyUp(event) {
|
64
|
+
event.preventDefault();
|
65
|
+
|
66
|
+
const currentIndex = this.itemTargets.findIndex(
|
67
|
+
(item) => item.getAttribute("aria-current") === "true",
|
68
|
+
);
|
69
|
+
|
70
|
+
if (currentIndex > 0) {
|
71
|
+
this.itemTargets[currentIndex].removeAttribute("aria-current");
|
72
|
+
this.setAriaCurrentAndActiveDescendant(currentIndex - 1);
|
73
|
+
}
|
74
|
+
}
|
75
|
+
|
76
|
+
handleEsc(event) {
|
77
|
+
event.preventDefault();
|
78
|
+
this.closeContent();
|
79
|
+
}
|
80
|
+
|
81
|
+
setFocusAndCurrent() {
|
82
|
+
const selectedItem = this.itemTargets.find(
|
83
|
+
(item) => item.getAttribute("aria-selected") === "true",
|
84
|
+
);
|
85
|
+
|
86
|
+
if (selectedItem) {
|
87
|
+
selectedItem.focus({ preventScroll: true });
|
88
|
+
selectedItem.setAttribute("aria-current", "true");
|
89
|
+
this.triggerTarget.setAttribute(
|
90
|
+
"aria-activedescendant",
|
91
|
+
selectedItem.getAttribute("id"),
|
92
|
+
);
|
93
|
+
} else {
|
94
|
+
this.itemTarget.focus({ preventScroll: true });
|
95
|
+
this.itemTarget.setAttribute("aria-current", "true");
|
96
|
+
this.triggerTarget.setAttribute(
|
97
|
+
"aria-activedescendant",
|
98
|
+
this.itemTarget.getAttribute("id"),
|
99
|
+
);
|
100
|
+
}
|
101
|
+
}
|
102
|
+
|
103
|
+
resetCurrent() {
|
104
|
+
this.itemTargets.forEach((item) => item.removeAttribute("aria-current"));
|
105
|
+
}
|
106
|
+
|
107
|
+
clickOutside(event) {
|
108
|
+
if (!this.openValue) return;
|
109
|
+
if (this.element.contains(event.target)) return;
|
110
|
+
|
111
|
+
event.preventDefault();
|
112
|
+
this.toogleContent();
|
113
|
+
}
|
114
|
+
|
115
|
+
toogleContent() {
|
116
|
+
this.openValue = !this.openValue;
|
117
|
+
this.contentTarget.classList.toggle("hidden");
|
118
|
+
this.triggerTarget.setAttribute("aria-expanded", this.openValue);
|
119
|
+
}
|
120
|
+
|
121
|
+
setFloatingElement() {
|
122
|
+
this.cleanup = autoUpdate(this.triggerTarget, this.contentTarget, () => {
|
123
|
+
computePosition(this.triggerTarget, this.contentTarget, {
|
124
|
+
middleware: [offset(4), flip()],
|
125
|
+
}).then(({ x, y }) => {
|
126
|
+
Object.assign(this.contentTarget.style, {
|
127
|
+
left: `${x}px`,
|
128
|
+
top: `${y}px`,
|
129
|
+
});
|
130
|
+
});
|
131
|
+
});
|
132
|
+
}
|
133
|
+
|
134
|
+
generateItemsIds() {
|
135
|
+
const contentId = this.contentTarget.getAttribute("id");
|
136
|
+
this.triggerTarget.setAttribute("aria-controls", contentId);
|
137
|
+
|
138
|
+
this.itemTargets.forEach((item, index) => {
|
139
|
+
item.id = `${contentId}-${index}`;
|
140
|
+
});
|
141
|
+
}
|
142
|
+
|
143
|
+
setAriaCurrentAndActiveDescendant(currentIndex) {
|
144
|
+
const currentItem = this.itemTargets[currentIndex];
|
145
|
+
currentItem.focus({ preventScroll: true });
|
146
|
+
currentItem.setAttribute("aria-current", "true");
|
147
|
+
this.triggerTarget.setAttribute(
|
148
|
+
"aria-activedescendant",
|
149
|
+
currentItem.getAttribute("id"),
|
150
|
+
);
|
151
|
+
}
|
152
|
+
|
153
|
+
closeContent() {
|
154
|
+
this.toogleContent();
|
155
|
+
this.resetCurrent();
|
156
|
+
|
157
|
+
this.triggerTarget.setAttribute("aria-activedescendant", true);
|
158
|
+
this.triggerTarget.focus({ preventScroll: true });
|
159
|
+
}
|
160
|
+
|
161
|
+
dispatchOnChange(oldValue, newValue) {
|
162
|
+
if (oldValue === newValue) return;
|
163
|
+
|
164
|
+
const event = new InputEvent("change", {
|
165
|
+
bubbles: true,
|
166
|
+
cancelable: true,
|
167
|
+
});
|
168
|
+
|
169
|
+
this.inputTarget.dispatchEvent(event);
|
170
|
+
}
|
171
|
+
}
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module RubyUI
|
4
4
|
class SelectInput < Base
|
5
5
|
def view_template
|
6
6
|
input(**attrs)
|
@@ -12,9 +12,9 @@ module RBUI
|
|
12
12
|
{
|
13
13
|
class: "hidden",
|
14
14
|
data: {
|
15
|
-
|
16
|
-
|
17
|
-
action: "change->
|
15
|
+
ruby_ui__select_target: "input",
|
16
|
+
ruby_ui__form_field_target: "input",
|
17
|
+
action: "change->ruby-ui--form-field#onChange invalid->ruby-ui--form-field#onInvalid"
|
18
18
|
}
|
19
19
|
}
|
20
20
|
end
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module RubyUI
|
4
4
|
class SelectItem < Base
|
5
5
|
def initialize(value: nil, **attrs)
|
6
6
|
@value = value
|
@@ -39,9 +39,9 @@ module RBUI
|
|
39
39
|
tabindex: "0",
|
40
40
|
class: "item group relative flex cursor-pointer select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground focus:bg-accent focus:text-accent-foreground aria-selected:bg-accent aria-selected:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
|
41
41
|
data: {
|
42
|
-
controller: "
|
43
|
-
action: "click->
|
44
|
-
|
42
|
+
controller: "ruby-ui--select-item",
|
43
|
+
action: "click->ruby-ui--select#selectItem keydown.enter->ruby-ui--select#selectItem keydown.down->ruby-ui--select#handleKeyDown keydown.up->ruby-ui--select#handleKeyUp keydown.esc->ruby-ui--select#handleEsc",
|
44
|
+
ruby_ui__select_target: "item"
|
45
45
|
},
|
46
46
|
data_value: @value,
|
47
47
|
data_orientation: "vertical",
|
@@ -0,0 +1,11 @@
|
|
1
|
+
import { Controller } from "@hotwired/stimulus";
|
2
|
+
export default class extends Controller {
|
3
|
+
|
4
|
+
handleSelectItem({ target }) {
|
5
|
+
if (this.element.dataset.value == target.dataset.value) {
|
6
|
+
this.element.setAttribute("aria-selected", true);
|
7
|
+
} else {
|
8
|
+
this.element.removeAttribute("aria-selected");
|
9
|
+
}
|
10
|
+
}
|
11
|
+
}
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module RubyUI
|
4
4
|
class SelectTrigger < Base
|
5
5
|
def view_template(&block)
|
6
6
|
button(**attrs) do
|
@@ -34,8 +34,8 @@ module RBUI
|
|
34
34
|
def default_attrs
|
35
35
|
{
|
36
36
|
data: {
|
37
|
-
action: "
|
38
|
-
|
37
|
+
action: "ruby-ui--select#onClick",
|
38
|
+
ruby_ui__select_target: "trigger"
|
39
39
|
},
|
40
40
|
type: "button",
|
41
41
|
role: "combobox",
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module RubyUI
|
4
4
|
class SelectValue < Base
|
5
5
|
def initialize(placeholder: nil, **attrs)
|
6
6
|
@placeholder = placeholder
|
@@ -18,9 +18,9 @@ module RBUI
|
|
18
18
|
def default_attrs
|
19
19
|
{
|
20
20
|
data: {
|
21
|
-
|
21
|
+
ruby_ui__select_target: "value"
|
22
22
|
},
|
23
|
-
class: "pointer-events-none"
|
23
|
+
class: "truncate pointer-events-none"
|
24
24
|
}
|
25
25
|
end
|
26
26
|
end
|
@@ -0,0 +1,38 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module RubyUI
|
4
|
+
class Separator < Base
|
5
|
+
ORIENTATIONS = %i[horizontal vertical].freeze
|
6
|
+
|
7
|
+
def initialize(as: :div, orientation: :horizontal, decorative: true, **attrs)
|
8
|
+
raise ArgumentError, "Invalid orientation: #{orientation}" unless ORIENTATIONS.include?(orientation.to_sym)
|
9
|
+
|
10
|
+
@as = as.to_sym
|
11
|
+
@orientation = orientation.to_sym
|
12
|
+
@decorative = decorative
|
13
|
+
super(**attrs)
|
14
|
+
end
|
15
|
+
|
16
|
+
def view_template(&)
|
17
|
+
tag(@as, **attrs, &)
|
18
|
+
end
|
19
|
+
|
20
|
+
private
|
21
|
+
|
22
|
+
def default_attrs
|
23
|
+
{
|
24
|
+
role: (@decorative ? "none" : "separator"),
|
25
|
+
class: [
|
26
|
+
"shrink-0 bg-border",
|
27
|
+
orientation_classes
|
28
|
+
]
|
29
|
+
}
|
30
|
+
end
|
31
|
+
|
32
|
+
def orientation_classes
|
33
|
+
return "h-[1px] w-full" if @orientation == :horizontal
|
34
|
+
|
35
|
+
"h-full w-[1px]"
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module RubyUI
|
4
4
|
class Sheet < Base
|
5
5
|
def view_template(&)
|
6
6
|
div(**attrs, &)
|
@@ -10,7 +10,7 @@ module RBUI
|
|
10
10
|
|
11
11
|
def default_attrs
|
12
12
|
{
|
13
|
-
data: {controller: "
|
13
|
+
data: {controller: "ruby-ui--sheet"}
|
14
14
|
}
|
15
15
|
end
|
16
16
|
end
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module RubyUI
|
4
4
|
class SheetContent < Base
|
5
5
|
SIDE_CLASS = {
|
6
6
|
top: "inset-x-0 top-0 border-b data-[state=closed]:slide-out-to-top data-[state=open]:slide-in-from-top",
|
@@ -16,8 +16,8 @@ module RBUI
|
|
16
16
|
end
|
17
17
|
|
18
18
|
def view_template(&block)
|
19
|
-
|
20
|
-
div(data: {controller: "
|
19
|
+
template(data: {ruby_ui__sheet_target: "content"}) do
|
20
|
+
div(data: {controller: "ruby-ui--sheet-content"}) do
|
21
21
|
backdrop
|
22
22
|
div(**attrs) do
|
23
23
|
block&.call
|
@@ -32,18 +32,18 @@ module RBUI
|
|
32
32
|
def default_attrs
|
33
33
|
{
|
34
34
|
data_state: "open", # For animate in
|
35
|
-
class:
|
35
|
+
class: [
|
36
36
|
"fixed pointer-events-auto z-50 gap-4 bg-background p-6 shadow-lg transition ease-in-out data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:duration-300 data-[state=open]:duration-500",
|
37
37
|
@side_classes
|
38
|
-
|
38
|
+
]
|
39
39
|
}
|
40
40
|
end
|
41
41
|
|
42
42
|
def close_button
|
43
43
|
button(
|
44
44
|
type: "button",
|
45
|
-
class: "absolute
|
46
|
-
data_action: "click->
|
45
|
+
class: "absolute end-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-accent data-[state=open]:text-muted-foreground",
|
46
|
+
data_action: "click->ruby-ui--sheet-content#close"
|
47
47
|
) do
|
48
48
|
svg(
|
49
49
|
width: "15",
|
@@ -68,7 +68,7 @@ module RBUI
|
|
68
68
|
def backdrop
|
69
69
|
div(
|
70
70
|
data_state: "open",
|
71
|
-
data_action: "click->
|
71
|
+
data_action: "click->ruby-ui--sheet-content#close",
|
72
72
|
class:
|
73
73
|
"fixed pointer-events-auto inset-0 z-50 bg-background/80 backdrop-blur-sm data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0"
|
74
74
|
)
|