ruby_ui 1.0.0.beta1 → 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 +4 -40
- data/lib/generators/ruby_ui/dependencies.yml +74 -0
- data/lib/generators/ruby_ui/install/install_generator.rb +21 -22
- 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 +21 -0
- data/lib/ruby_ui/accordion/accordion_controller.js +97 -0
- data/lib/ruby_ui/alert/alert.rb +1 -1
- data/lib/ruby_ui/alert_dialog/alert_dialog_content.rb +1 -1
- data/lib/ruby_ui/alert_dialog/alert_dialog_controller.js +31 -0
- data/lib/ruby_ui/alert_dialog/alert_dialog_footer.rb +1 -1
- data/lib/ruby_ui/alert_dialog/alert_dialog_header.rb +1 -1
- data/lib/ruby_ui/breadcrumb/breadcrumb.rb +17 -0
- data/lib/ruby_ui/breadcrumb/breadcrumb_ellipsis.rb +39 -0
- data/lib/ruby_ui/breadcrumb/breadcrumb_item.rb +17 -0
- 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/ruby_ui/calendar/calendar_controller.js +249 -0
- data/lib/ruby_ui/calendar/calendar_input_controller.js +8 -0
- 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/ruby_ui/chart/chart_controller.js +103 -0
- data/lib/ruby_ui/checkbox/checkbox_group_controller.js +21 -0
- data/lib/ruby_ui/clipboard/clipboard_controller.js +54 -0
- data/lib/ruby_ui/collapsible/collapsible_controller.js +47 -0
- data/lib/ruby_ui/combobox/combobox.rb +8 -6
- data/lib/ruby_ui/combobox/combobox_checkbox.rb +25 -0
- data/lib/ruby_ui/combobox/combobox_controller.js +176 -0
- data/lib/ruby_ui/combobox/{combobox_empty.rb → combobox_empty_state.rb} +2 -2
- data/lib/ruby_ui/combobox/combobox_item.rb +9 -37
- data/lib/ruby_ui/combobox/combobox_list.rb +2 -11
- 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/ruby_ui/combobox/combobox_search_input.rb +21 -24
- data/lib/ruby_ui/combobox/combobox_toggle_all_checkbox.rb +25 -0
- data/lib/ruby_ui/combobox/combobox_trigger.rb +25 -20
- data/lib/ruby_ui/command/command_controller.js +136 -0
- data/lib/ruby_ui/context_menu/context_menu_controller.js +144 -0
- data/lib/ruby_ui/dialog/dialog_content.rb +2 -2
- data/lib/ruby_ui/dialog/dialog_controller.js +32 -0
- data/lib/ruby_ui/dialog/dialog_footer.rb +1 -1
- data/lib/ruby_ui/dialog/dialog_header.rb +1 -1
- data/lib/ruby_ui/dropdown_menu/dropdown_menu_controller.js +120 -0
- data/lib/ruby_ui/form/form_field_controller.js +61 -0
- data/lib/ruby_ui/hover_card/hover_card_controller.js +144 -0
- data/lib/ruby_ui/masked_input/masked_input_controller.js +9 -0
- data/lib/ruby_ui/popover/popover_controller.js +107 -0
- data/lib/ruby_ui/progress/progress.rb +37 -0
- data/lib/ruby_ui/radio_button/radio_button.rb +4 -1
- data/lib/ruby_ui/select/select_content.rb +1 -1
- data/lib/ruby_ui/select/select_controller.js +171 -0
- data/lib/ruby_ui/select/select_item_controller.js +11 -0
- data/lib/ruby_ui/select/select_value.rb +1 -1
- data/lib/ruby_ui/separator/separator.rb +38 -0
- data/lib/ruby_ui/sheet/sheet_content.rb +1 -1
- data/lib/ruby_ui/sheet/sheet_content_controller.js +7 -0
- data/lib/ruby_ui/sheet/sheet_controller.js +9 -0
- data/lib/ruby_ui/{combobox/combobox_separator.rb → skeleton/skeleton.rb} +4 -2
- data/lib/ruby_ui/switch/switch.rb +24 -0
- data/lib/ruby_ui/tabs/tabs_controller.js +45 -0
- data/lib/ruby_ui/theme_toggle/theme_toggle_controller.js +30 -0
- data/lib/ruby_ui/tooltip/tooltip_controller.js +37 -0
- data/lib/ruby_ui.rb +1 -1
- metadata +57 -11
- data/lib/ruby_ui/combobox/combobox_content.rb +0 -31
- data/lib/ruby_ui/combobox/combobox_group.rb +0 -38
- data/lib/ruby_ui/combobox/combobox_input.rb +0 -22
- data/lib/ruby_ui/combobox/combobox_value.rb +0 -27
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: bef04b912f756edd217e2294608bc08eef650d0e944145ef052b5a1760860682
|
4
|
+
data.tar.gz: '0792ce039b5817ce7fd9b6965c20672497ba6615ab555b2965110ce6a6817ed2'
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 7c515453889a668ab223f6099407cd32c70046bd91e8e1e814ee3759e865857b83d0de87fda35b67a157da6a045794f067b1871574eff82b3d313180a62c5aff
|
7
|
+
data.tar.gz: 2cb5b4988c27072c593bf6fa88b3f40785cb8ba19c6c9627082b56260e4793978757ddac1683216e74e1a91c0052ac2474cead61ecef67c8b4249f67988b7f9b
|
data/LICENSE.txt
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
The MIT License (MIT)
|
2
|
+
|
3
|
+
Copyright (c) 2024 RubyUI
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in
|
13
|
+
all copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
21
|
+
THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,85 @@
|
|
1
|
+
# RubyUI (former PhlexUI) 🚀
|
2
|
+
|
3
|
+
Beautifully designed components that you can copy and paste into your apps. Accessible. Customizable. Open Source.
|
4
|
+
|
5
|
+
This is NOT a component library. It's a collection of re-usable components that you can generate or copy and paste into your apps.
|
6
|
+
|
7
|
+
Pick the components you need. Copy and paste the code into your project and customize to your needs. The code is yours.
|
8
|
+
|
9
|
+
Use this as a reference to build your own component libraries.
|
10
|
+
|
11
|
+
### Key Features:
|
12
|
+
|
13
|
+
- **Built for Speed** ⚡: RubyUI leverages Phlex, which is up to 12x faster than traditional Rails ERB templates.
|
14
|
+
- **Stunning UI** 🎨: Design beautiful, streamlined, and customizable UIs that sell your app effortlessly.
|
15
|
+
- **Stay Organized** 📁: Keep your UI components well-organized and easy to manage.
|
16
|
+
- **Customer-Centric UX** 🧑💼: Create memorable app experiences for your users.
|
17
|
+
- **Completely Customizable** 🔧: Full control over the design of all components.
|
18
|
+
- **Minimal Dependencies** 🍃: Uses custom-built Stimulus.js controllers to keep your app lean.
|
19
|
+
- **Reuse with Ease** ♻️: Build components once and use them seamlessly across your project.
|
20
|
+
|
21
|
+
### How to Use:
|
22
|
+
|
23
|
+
1. **Find the perfect component** 🔍: Browse live-embedded components on our documentation page.
|
24
|
+
2. **Copy the snippet** 📋: Easily copy code snippets for quick implementation.
|
25
|
+
3. **Make it yours** 🎨: Customize components using Tailwind utility classes to fit your specific needs.
|
26
|
+
|
27
|
+
## Installation 🚀
|
28
|
+
|
29
|
+
### 1. Install the gem
|
30
|
+
|
31
|
+
```bash
|
32
|
+
bundle add ruby_ui --group development --require false
|
33
|
+
```
|
34
|
+
|
35
|
+
or add it to your Gemfile:
|
36
|
+
|
37
|
+
```ruby
|
38
|
+
gem "ruby_ui", group: :development, require: false
|
39
|
+
```
|
40
|
+
|
41
|
+
### 2. Run the installer:
|
42
|
+
|
43
|
+
```bash
|
44
|
+
bin/rails g ruby_ui:install
|
45
|
+
```
|
46
|
+
|
47
|
+
### 3. Done! 🎉
|
48
|
+
|
49
|
+
You can generate your components using `ruby_ui:component` generator.
|
50
|
+
|
51
|
+
```bash
|
52
|
+
bin/rails g ruby_ui:component Accordion
|
53
|
+
```
|
54
|
+
|
55
|
+
## Documentation 📖
|
56
|
+
|
57
|
+
Visit https://rubyui.com/docs/introduction to view the full documentation, including:
|
58
|
+
|
59
|
+
- Detailed component guides
|
60
|
+
- Themes
|
61
|
+
- Lookbook
|
62
|
+
- Getting started guide
|
63
|
+
|
64
|
+
## Speed Comparison 🏎️
|
65
|
+
|
66
|
+
RubyUI, powered by Phlex, outperforms alternative methods:
|
67
|
+
|
68
|
+
- Phlex: Baseline 🏁
|
69
|
+
- ViewComponent: ~1.5x slower 🚙
|
70
|
+
- ERB Templates: ~5x slower 🐢
|
71
|
+
|
72
|
+
See the original [view layers benchmark](https://github.com/KonnorRogers/view-layer-benchmarks) by @KonnorRogers and its [variations](https://github.com/KonnorRogers/view-layer-benchmarks/forks).
|
73
|
+
|
74
|
+
## Importmap notes:
|
75
|
+
|
76
|
+
If you run into importmap issues this stackoverflow question might help:
|
77
|
+
https://stackoverflow.com/questions/70548841/how-to-add-custom-js-file-to-new-rails-7-project/72855705
|
78
|
+
|
79
|
+
## License 📜
|
80
|
+
|
81
|
+
Licensed under the [MIT license](https://github.com/shadcn/ui/blob/main/LICENSE.md).
|
82
|
+
|
83
|
+
---
|
84
|
+
|
85
|
+
© 2024 RubyUI. All rights reserved. 🔒
|
@@ -15,26 +15,13 @@ module RubyUI
|
|
15
15
|
exit
|
16
16
|
end
|
17
17
|
|
18
|
-
say "Generating
|
19
|
-
end
|
20
|
-
|
21
|
-
def copy_main_component_file
|
22
|
-
main_component_file_path = File.join(component_folder_path, "#{component_folder_name}.rb")
|
23
|
-
|
24
|
-
# some components dont't have a main component, eg. Typography
|
25
|
-
return unless File.exist? main_component_file_path
|
26
|
-
|
27
|
-
say "Generating main component"
|
28
|
-
|
29
|
-
copy_file main_component_file_path, Rails.root.join("app/components/ruby_ui", "#{component_folder_name}.rb")
|
18
|
+
say "Generating #{component_name} files..."
|
30
19
|
end
|
31
20
|
|
32
21
|
def copy_related_component_files
|
33
|
-
|
34
|
-
|
35
|
-
say "Generating related components"
|
22
|
+
say "Generating components"
|
36
23
|
|
37
|
-
|
24
|
+
components_file_paths.each do |file_path|
|
38
25
|
component_file_name = file_path.split("/").last
|
39
26
|
copy_file file_path, Rails.root.join("app/components/ruby_ui", component_folder_name, component_file_name)
|
40
27
|
end
|
@@ -75,9 +62,7 @@ module RubyUI
|
|
75
62
|
|
76
63
|
def component_folder_path = File.join(self.class.source_root, component_folder_name)
|
77
64
|
|
78
|
-
def
|
79
|
-
|
80
|
-
def related_components_file_paths = Dir.glob(File.join(component_folder_path, "*.rb")) - [main_component_file_path]
|
65
|
+
def components_file_paths = Dir.glob(File.join(component_folder_path, "*.rb"))
|
81
66
|
|
82
67
|
def js_controller_file_paths = Dir.glob(File.join(component_folder_path, "*.js"))
|
83
68
|
|
@@ -99,27 +84,6 @@ module RubyUI
|
|
99
84
|
end
|
100
85
|
end
|
101
86
|
|
102
|
-
def pin_motion
|
103
|
-
say <<~TEXT
|
104
|
-
WARNING: Installing motion from CDN because `bin/importmap pin motion` doesn't download the correct file.
|
105
|
-
TEXT
|
106
|
-
|
107
|
-
inject_into_file Rails.root.join("config/importmap.rb"), <<~RUBY
|
108
|
-
pin "motion", to: "https://cdn.jsdelivr.net/npm/motion@11.11.17/+esm"\n
|
109
|
-
RUBY
|
110
|
-
end
|
111
|
-
|
112
|
-
def pin_tippy_js
|
113
|
-
say <<~TEXT
|
114
|
-
WARNING: Installing tippy.js from CDN because `bin/importmap pin tippy.js` doesn't download the correct file.
|
115
|
-
TEXT
|
116
|
-
|
117
|
-
inject_into_file Rails.root.join("config/importmap.rb"), <<~RUBY
|
118
|
-
pin "tippy.js", to: "https://cdn.jsdelivr.net/npm/tippy.js@6.3.7/+esm"
|
119
|
-
pin "@popperjs/core", to: "https://cdn.jsdelivr.net/npm/@popperjs/core@2.11.8/+esm"\n
|
120
|
-
RUBY
|
121
|
-
end
|
122
|
-
|
123
87
|
def dependencies
|
124
88
|
@dependencies ||= YAML.load_file(File.join(__dir__, "dependencies.yml")).freeze
|
125
89
|
|
@@ -0,0 +1,74 @@
|
|
1
|
+
accordion:
|
2
|
+
js_packages:
|
3
|
+
- "motion"
|
4
|
+
|
5
|
+
alert_dialog:
|
6
|
+
components:
|
7
|
+
- "Button"
|
8
|
+
|
9
|
+
calendar:
|
10
|
+
js_packages:
|
11
|
+
- "mustache"
|
12
|
+
|
13
|
+
carousel:
|
14
|
+
js_packages:
|
15
|
+
- "embla-carousel"
|
16
|
+
|
17
|
+
chart:
|
18
|
+
js_packages:
|
19
|
+
- "chart.js"
|
20
|
+
|
21
|
+
clipboard:
|
22
|
+
js_packages:
|
23
|
+
- "@floating-ui/dom"
|
24
|
+
|
25
|
+
codeblock:
|
26
|
+
components:
|
27
|
+
- "Button"
|
28
|
+
- "Clipboard"
|
29
|
+
|
30
|
+
gems:
|
31
|
+
- "rouge"
|
32
|
+
|
33
|
+
combobox:
|
34
|
+
js_packages:
|
35
|
+
- "@floating-ui/dom"
|
36
|
+
|
37
|
+
command:
|
38
|
+
js_packages:
|
39
|
+
- "fuse.js"
|
40
|
+
|
41
|
+
context_menu:
|
42
|
+
js_packages:
|
43
|
+
- "tippy.js"
|
44
|
+
|
45
|
+
dropdown_menu:
|
46
|
+
js_packages:
|
47
|
+
- "@floating-ui/dom"
|
48
|
+
|
49
|
+
hover_card:
|
50
|
+
js_packages:
|
51
|
+
- "tippy.js"
|
52
|
+
|
53
|
+
masked_input:
|
54
|
+
components:
|
55
|
+
- "Input"
|
56
|
+
|
57
|
+
js_packages:
|
58
|
+
- "maska"
|
59
|
+
|
60
|
+
pagination:
|
61
|
+
components:
|
62
|
+
- "Button"
|
63
|
+
|
64
|
+
popover:
|
65
|
+
js_packages:
|
66
|
+
- "@floating-ui/dom"
|
67
|
+
|
68
|
+
select:
|
69
|
+
js_packages:
|
70
|
+
- "@floating-ui/dom"
|
71
|
+
|
72
|
+
tooltip:
|
73
|
+
js_packages:
|
74
|
+
- "@floating-ui/dom"
|
@@ -14,17 +14,12 @@ module RubyUI
|
|
14
14
|
say "Checking for phlex-rails"
|
15
15
|
|
16
16
|
if gem_installed?("phlex-rails")
|
17
|
-
|
18
|
-
say "You need to upgrade to phlex-rails 2 to use RubyUI", :red
|
19
|
-
exit
|
20
|
-
else
|
21
|
-
say "phlex-rails is already installed", :green
|
22
|
-
end
|
17
|
+
say "phlex-rails is already installed", :green
|
23
18
|
else
|
24
19
|
say "Adding phlex-rails to Gemfile"
|
25
|
-
run %(bundle add phlex-rails
|
20
|
+
run %(bundle add phlex-rails)
|
26
21
|
|
27
|
-
say "
|
22
|
+
say "Generating phlex-rails structure"
|
28
23
|
run "bin/rails generate phlex:install"
|
29
24
|
end
|
30
25
|
end
|
@@ -53,26 +48,26 @@ module RubyUI
|
|
53
48
|
end
|
54
49
|
|
55
50
|
def add_tailwind_css
|
56
|
-
say "Adding
|
57
|
-
template "application.tailwind.css.erb", Rails.root.join("app/assets/stylesheets/application.tailwind.css")
|
58
|
-
end
|
59
|
-
|
60
|
-
def add_tailwind_config
|
61
|
-
say "Adding RubyUI config to tailwind config"
|
51
|
+
say "Adding Tailwind css"
|
62
52
|
|
63
|
-
if
|
64
|
-
|
65
|
-
elsif File.exist?(Rails.root.join("config/tailwind.config.js")) # tailwindcss-rails gem
|
66
|
-
template "tailwind.config.js.tailwindcss-rails.erb", Rails.root.join("config/tailwind.config.js")
|
53
|
+
css_path = if using_importmap?
|
54
|
+
Rails.root.join("app/assets/tailwind/application.css")
|
67
55
|
else
|
68
|
-
|
56
|
+
Rails.root.join("app/assets/stylesheets/application.tailwind.css")
|
69
57
|
end
|
58
|
+
|
59
|
+
template "tailwind.css.erb", css_path
|
70
60
|
end
|
71
61
|
|
72
|
-
def
|
73
|
-
say "Installing
|
62
|
+
def install_tailwind_plugins
|
63
|
+
say "Installing tw-animate-css plugin"
|
64
|
+
install_js_package("tw-animate-css")
|
74
65
|
|
75
|
-
|
66
|
+
say "Installing @tailwindcss/forms plugin"
|
67
|
+
install_js_package("@tailwindcss/forms")
|
68
|
+
|
69
|
+
say "Installing @tailwindcss/typography plugin"
|
70
|
+
install_js_package("@tailwindcss/typography")
|
76
71
|
end
|
77
72
|
|
78
73
|
def add_ruby_ui_base
|
@@ -85,6 +80,10 @@ module RubyUI
|
|
85
80
|
def gem_installed?(name)
|
86
81
|
Gem::Specification.find_all_by_name(name).any?
|
87
82
|
end
|
83
|
+
|
84
|
+
def using_tailwindcss_rails_gem?
|
85
|
+
File.exist?(Rails.root.join("app/assets/tailwind/application.css"))
|
86
|
+
end
|
88
87
|
end
|
89
88
|
end
|
90
89
|
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module RubyUI
|
4
|
+
extend Phlex::Kit
|
5
|
+
end
|
6
|
+
|
7
|
+
# Allow using RubyUI instead RubyUi
|
8
|
+
Rails.autoloaders.main.inflector.inflect(
|
9
|
+
"ruby_ui" => "RubyUI"
|
10
|
+
)
|
11
|
+
|
12
|
+
# Allow using RubyUI::ComponentName instead Components::RubyUI::ComponentName
|
13
|
+
Rails.autoloaders.main.push_dir(
|
14
|
+
"#{Rails.root}/app/components/ruby_ui", namespace: RubyUI
|
15
|
+
)
|
16
|
+
|
17
|
+
# Allow using RubyUI::ComponentName instead RubyUI::ComponentName::ComponentName
|
18
|
+
Rails.autoloaders.main.collapse(Rails.root.join("app/components/ruby_ui/*"))
|
@@ -0,0 +1,156 @@
|
|
1
|
+
@import "tailwindcss";
|
2
|
+
|
3
|
+
@plugin "@tailwindcss/forms";
|
4
|
+
@plugin "@tailwindcss/typography";
|
5
|
+
|
6
|
+
<% if using_importmap? %>
|
7
|
+
@import "../../../vendor/javascript/tw-animate-css.js";
|
8
|
+
<% else %>
|
9
|
+
@import "tw-animate-css";
|
10
|
+
<% end %>
|
11
|
+
|
12
|
+
@custom-variant dark (&:is(.dark *));
|
13
|
+
|
14
|
+
:root {
|
15
|
+
--background: oklch(1 0 0);
|
16
|
+
--foreground: oklch(0.145 0 0);
|
17
|
+
--card: oklch(1 0 0);
|
18
|
+
--card-foreground: oklch(0.145 0 0);
|
19
|
+
--popover: oklch(1 0 0);
|
20
|
+
--popover-foreground: oklch(0.145 0 0);
|
21
|
+
--primary: oklch(0.205 0 0);
|
22
|
+
--primary-foreground: oklch(0.985 0 0);
|
23
|
+
--secondary: oklch(0.97 0 0);
|
24
|
+
--secondary-foreground: oklch(0.205 0 0);
|
25
|
+
--muted: oklch(0.97 0 0);
|
26
|
+
--muted-foreground: oklch(0.556 0 0);
|
27
|
+
--accent: oklch(0.97 0 0);
|
28
|
+
--accent-foreground: oklch(0.205 0 0);
|
29
|
+
--destructive: oklch(0.577 0.245 27.325);
|
30
|
+
--destructive-foreground: oklch(0.577 0.245 27.325);
|
31
|
+
--border: oklch(0.922 0 0);
|
32
|
+
--input: oklch(0.922 0 0);
|
33
|
+
--ring: oklch(0.708 0 0);
|
34
|
+
--chart-1: oklch(0.646 0.222 41.116);
|
35
|
+
--chart-2: oklch(0.6 0.118 184.704);
|
36
|
+
--chart-3: oklch(0.398 0.07 227.392);
|
37
|
+
--chart-4: oklch(0.828 0.189 84.429);
|
38
|
+
--chart-5: oklch(0.769 0.188 70.08);
|
39
|
+
--radius: 0.625rem;
|
40
|
+
--sidebar: oklch(0.985 0 0);
|
41
|
+
--sidebar-foreground: oklch(0.145 0 0);
|
42
|
+
--sidebar-primary: oklch(0.205 0 0);
|
43
|
+
--sidebar-primary-foreground: oklch(0.985 0 0);
|
44
|
+
--sidebar-accent: oklch(0.97 0 0);
|
45
|
+
--sidebar-accent-foreground: oklch(0.205 0 0);
|
46
|
+
--sidebar-border: oklch(0.922 0 0);
|
47
|
+
--sidebar-ring: oklch(0.708 0 0);
|
48
|
+
|
49
|
+
/* ruby_ui specific */
|
50
|
+
--warning: hsl(38 92% 50%);
|
51
|
+
--warning-foreground: hsl(0 0% 100%);
|
52
|
+
--success: hsl(87 100% 37%);
|
53
|
+
--success-foreground: hsl(0 0% 100%);
|
54
|
+
}
|
55
|
+
|
56
|
+
.dark {
|
57
|
+
--background: oklch(0.145 0 0);
|
58
|
+
--foreground: oklch(0.985 0 0);
|
59
|
+
--card: oklch(0.145 0 0);
|
60
|
+
--card-foreground: oklch(0.985 0 0);
|
61
|
+
--popover: oklch(0.145 0 0);
|
62
|
+
--popover-foreground: oklch(0.985 0 0);
|
63
|
+
--primary: oklch(0.985 0 0);
|
64
|
+
--primary-foreground: oklch(0.205 0 0);
|
65
|
+
--secondary: oklch(0.269 0 0);
|
66
|
+
--secondary-foreground: oklch(0.985 0 0);
|
67
|
+
--muted: oklch(0.269 0 0);
|
68
|
+
--muted-foreground: oklch(0.708 0 0);
|
69
|
+
--accent: oklch(0.269 0 0);
|
70
|
+
--accent-foreground: oklch(0.985 0 0);
|
71
|
+
--destructive: oklch(0.396 0.141 25.723);
|
72
|
+
--destructive-foreground: oklch(0.637 0.237 25.331);
|
73
|
+
--border: oklch(0.269 0 0);
|
74
|
+
--input: oklch(0.269 0 0);
|
75
|
+
--ring: oklch(0.439 0 0);
|
76
|
+
--chart-1: oklch(0.488 0.243 264.376);
|
77
|
+
--chart-2: oklch(0.696 0.17 162.48);
|
78
|
+
--chart-3: oklch(0.769 0.188 70.08);
|
79
|
+
--chart-4: oklch(0.627 0.265 303.9);
|
80
|
+
--chart-5: oklch(0.645 0.246 16.439);
|
81
|
+
--sidebar: oklch(0.205 0 0);
|
82
|
+
--sidebar-foreground: oklch(0.985 0 0);
|
83
|
+
--sidebar-primary: oklch(0.488 0.243 264.376);
|
84
|
+
--sidebar-primary-foreground: oklch(0.985 0 0);
|
85
|
+
--sidebar-accent: oklch(0.269 0 0);
|
86
|
+
--sidebar-accent-foreground: oklch(0.985 0 0);
|
87
|
+
--sidebar-border: oklch(0.269 0 0);
|
88
|
+
--sidebar-ring: oklch(0.439 0 0);
|
89
|
+
|
90
|
+
/* ruby_ui specific */
|
91
|
+
--warning: hsl(38 92% 50%);
|
92
|
+
--warning-foreground: hsl(0 0% 100%);
|
93
|
+
--success: hsl(84 81% 44%);
|
94
|
+
--success-foreground: hsl(0 0% 100%);
|
95
|
+
}
|
96
|
+
|
97
|
+
@theme inline {
|
98
|
+
--color-background: var(--background);
|
99
|
+
--color-foreground: var(--foreground);
|
100
|
+
--color-card: var(--card);
|
101
|
+
--color-card-foreground: var(--card-foreground);
|
102
|
+
--color-popover: var(--popover);
|
103
|
+
--color-popover-foreground: var(--popover-foreground);
|
104
|
+
--color-primary: var(--primary);
|
105
|
+
--color-primary-foreground: var(--primary-foreground);
|
106
|
+
--color-secondary: var(--secondary);
|
107
|
+
--color-secondary-foreground: var(--secondary-foreground);
|
108
|
+
--color-muted: var(--muted);
|
109
|
+
--color-muted-foreground: var(--muted-foreground);
|
110
|
+
--color-accent: var(--accent);
|
111
|
+
--color-accent-foreground: var(--accent-foreground);
|
112
|
+
--color-destructive: var(--destructive);
|
113
|
+
--color-destructive-foreground: var(--destructive-foreground);
|
114
|
+
--color-border: var(--border);
|
115
|
+
--color-input: var(--input);
|
116
|
+
--color-ring: var(--ring);
|
117
|
+
--color-chart-1: var(--chart-1);
|
118
|
+
--color-chart-2: var(--chart-2);
|
119
|
+
--color-chart-3: var(--chart-3);
|
120
|
+
--color-chart-4: var(--chart-4);
|
121
|
+
--color-chart-5: var(--chart-5);
|
122
|
+
--radius-sm: calc(var(--radius) - 4px);
|
123
|
+
--radius-md: calc(var(--radius) - 2px);
|
124
|
+
--radius-lg: var(--radius);
|
125
|
+
--radius-xl: calc(var(--radius) + 4px);
|
126
|
+
--color-sidebar: var(--sidebar);
|
127
|
+
--color-sidebar-foreground: var(--sidebar-foreground);
|
128
|
+
--color-sidebar-primary: var(--sidebar-primary);
|
129
|
+
--color-sidebar-primary-foreground: var(--sidebar-primary-foreground);
|
130
|
+
--color-sidebar-accent: var(--sidebar-accent);
|
131
|
+
--color-sidebar-accent-foreground: var(--sidebar-accent-foreground);
|
132
|
+
--color-sidebar-border: var(--sidebar-border);
|
133
|
+
--color-sidebar-ring: var(--sidebar-ring);
|
134
|
+
|
135
|
+
/* ruby_ui specific */
|
136
|
+
--color-warning: var(--warning);
|
137
|
+
--color-warning-foreground: var(--warning-foreground);
|
138
|
+
--color-success: var(--success);
|
139
|
+
--color-success-foreground: var(--success-foreground);
|
140
|
+
}
|
141
|
+
|
142
|
+
/* Container settings */
|
143
|
+
@utility container {
|
144
|
+
margin-inline: auto;
|
145
|
+
padding-inline: 2rem;
|
146
|
+
max-width: 1400px;
|
147
|
+
}
|
148
|
+
|
149
|
+
@layer base {
|
150
|
+
* {
|
151
|
+
@apply border-border outline-ring/50;
|
152
|
+
}
|
153
|
+
body {
|
154
|
+
@apply bg-background text-foreground;
|
155
|
+
}
|
156
|
+
}
|
@@ -31,6 +31,27 @@ module RubyUI
|
|
31
31
|
def using_npm? = File.exist?(Rails.root.join("package-lock.json"))
|
32
32
|
|
33
33
|
def using_yarn? = File.exist?(Rails.root.join("yarn.lock"))
|
34
|
+
|
35
|
+
def pin_motion
|
36
|
+
say <<~TEXT
|
37
|
+
WARNING: Installing motion from CDN because `bin/importmap pin motion` doesn't download the correct file.
|
38
|
+
TEXT
|
39
|
+
|
40
|
+
inject_into_file Rails.root.join("config/importmap.rb"), <<~RUBY
|
41
|
+
pin "motion", to: "https://cdn.jsdelivr.net/npm/motion@11.11.17/+esm"\n
|
42
|
+
RUBY
|
43
|
+
end
|
44
|
+
|
45
|
+
def pin_tippy_js
|
46
|
+
say <<~TEXT
|
47
|
+
WARNING: Installing tippy.js from CDN because `bin/importmap pin tippy.js` doesn't download the correct file.
|
48
|
+
TEXT
|
49
|
+
|
50
|
+
inject_into_file Rails.root.join("config/importmap.rb"), <<~RUBY
|
51
|
+
pin "tippy.js", to: "https://cdn.jsdelivr.net/npm/tippy.js@6.3.7/+esm"
|
52
|
+
pin "@popperjs/core", to: "https://cdn.jsdelivr.net/npm/@popperjs/core@2.11.8/+esm"\n
|
53
|
+
RUBY
|
54
|
+
end
|
34
55
|
end
|
35
56
|
end
|
36
57
|
end
|
@@ -0,0 +1,97 @@
|
|
1
|
+
import { Controller } from "@hotwired/stimulus";
|
2
|
+
import { animate } from "motion";
|
3
|
+
|
4
|
+
// Connects to data-controller="ruby-ui--accordion"
|
5
|
+
export default class extends Controller {
|
6
|
+
static targets = ["icon", "content"];
|
7
|
+
static values = {
|
8
|
+
open: {
|
9
|
+
type: Boolean,
|
10
|
+
default: false,
|
11
|
+
},
|
12
|
+
animationDuration: {
|
13
|
+
type: Number,
|
14
|
+
default: 0.15, // Default animation duration (in seconds)
|
15
|
+
},
|
16
|
+
animationEasing: {
|
17
|
+
type: String,
|
18
|
+
default: "ease-in-out", // Default animation easing
|
19
|
+
},
|
20
|
+
rotateIcon: {
|
21
|
+
type: Number,
|
22
|
+
default: 180, // Default icon rotation (in degrees)
|
23
|
+
},
|
24
|
+
};
|
25
|
+
|
26
|
+
connect() {
|
27
|
+
// Set the initial state of the accordion
|
28
|
+
let originalAnimationDuration = this.animationDurationValue;
|
29
|
+
this.animationDurationValue = 0;
|
30
|
+
this.openValue ? this.open() : this.close();
|
31
|
+
this.animationDurationValue = originalAnimationDuration;
|
32
|
+
}
|
33
|
+
|
34
|
+
// Toggle the 'open' value
|
35
|
+
toggle() {
|
36
|
+
this.openValue = !this.openValue;
|
37
|
+
}
|
38
|
+
|
39
|
+
// Handle changes in the 'open' value
|
40
|
+
openValueChanged(isOpen, wasOpen) {
|
41
|
+
if (isOpen) {
|
42
|
+
this.open();
|
43
|
+
} else {
|
44
|
+
this.close();
|
45
|
+
}
|
46
|
+
}
|
47
|
+
|
48
|
+
// Open the accordion content
|
49
|
+
open() {
|
50
|
+
if (this.hasContentTarget) {
|
51
|
+
this.revealContent();
|
52
|
+
this.hasIconTarget && this.rotateIcon();
|
53
|
+
this.openValue = true;
|
54
|
+
}
|
55
|
+
}
|
56
|
+
|
57
|
+
// Close the accordion content
|
58
|
+
close() {
|
59
|
+
if (this.hasContentTarget) {
|
60
|
+
this.hideContent();
|
61
|
+
this.hasIconTarget && this.rotateIcon();
|
62
|
+
this.openValue = false;
|
63
|
+
}
|
64
|
+
}
|
65
|
+
|
66
|
+
// Reveal the accordion content with animation
|
67
|
+
revealContent() {
|
68
|
+
const contentHeight = this.contentTarget.scrollHeight;
|
69
|
+
animate(
|
70
|
+
this.contentTarget,
|
71
|
+
{ height: `${contentHeight}px` },
|
72
|
+
{
|
73
|
+
duration: this.animationDurationValue,
|
74
|
+
easing: this.animationEasingValue,
|
75
|
+
},
|
76
|
+
);
|
77
|
+
}
|
78
|
+
|
79
|
+
// Hide the accordion content with animation
|
80
|
+
hideContent() {
|
81
|
+
animate(
|
82
|
+
this.contentTarget,
|
83
|
+
{ height: 0 },
|
84
|
+
{
|
85
|
+
duration: this.animationDurationValue,
|
86
|
+
easing: this.animationEasingValue,
|
87
|
+
},
|
88
|
+
);
|
89
|
+
}
|
90
|
+
|
91
|
+
// Rotate the accordion icon 180deg using animate function
|
92
|
+
rotateIcon() {
|
93
|
+
animate(this.iconTarget, {
|
94
|
+
rotate: `${this.openValue ? this.rotateIconValue : 0}deg`,
|
95
|
+
});
|
96
|
+
}
|
97
|
+
}
|
data/lib/ruby_ui/alert/alert.rb
CHANGED
@@ -27,7 +27,7 @@ module RubyUI
|
|
27
27
|
end
|
28
28
|
|
29
29
|
def default_attrs
|
30
|
-
base_classes = "backdrop-blur relative w-full ring-1 ring-inset rounded-lg px-4 py-4 text-sm [&>svg+div]:translate-y-[-3px] [&>svg]:absolute [&>svg]:
|
30
|
+
base_classes = "backdrop-blur relative w-full ring-1 ring-inset rounded-lg px-4 py-4 text-sm [&>svg+div]:translate-y-[-3px] [&>svg]:absolute [&>svg]:start-4 [&>svg]:top-4 [&>svg~*]:ps-8"
|
31
31
|
{
|
32
32
|
class: [base_classes, colors]
|
33
33
|
}
|
@@ -26,7 +26,7 @@ module RubyUI
|
|
26
26
|
div(
|
27
27
|
role: "alertdialog",
|
28
28
|
data_state: "open",
|
29
|
-
class: "fixed left-[50%] top-[50%] z-50
|
29
|
+
class: "flex flex-col fixed left-[50%] top-[50%] z-50 w-full max-w-lg max-h-screen overflow-y-auto translate-x-[-50%] translate-y-[-50%] gap-4 border bg-background p-6 shadow-lg duration-200 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%] data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%] sm:rounded-lg md:w-full",
|
30
30
|
style: "pointer-events:auto",
|
31
31
|
&
|
32
32
|
)
|
@@ -0,0 +1,31 @@
|
|
1
|
+
import { Controller } from "@hotwired/stimulus";
|
2
|
+
|
3
|
+
// Connects to data-controller="ruby-ui--alert-dialog"
|
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
|
+
if (this.openValue) {
|
15
|
+
this.open();
|
16
|
+
}
|
17
|
+
}
|
18
|
+
|
19
|
+
open() {
|
20
|
+
document.body.insertAdjacentHTML("beforeend", this.contentTarget.innerHTML);
|
21
|
+
// prevent scroll on body
|
22
|
+
document.body.classList.add("overflow-hidden");
|
23
|
+
}
|
24
|
+
|
25
|
+
dismiss(e) {
|
26
|
+
// allow scroll on body
|
27
|
+
document.body.classList.remove("overflow-hidden");
|
28
|
+
// remove the element
|
29
|
+
this.element.remove();
|
30
|
+
}
|
31
|
+
}
|