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.
Files changed (78) hide show
  1. checksums.yaml +4 -4
  2. data/LICENSE.txt +21 -0
  3. data/README.md +85 -0
  4. data/lib/generators/ruby_ui/component_generator.rb +4 -40
  5. data/lib/generators/ruby_ui/dependencies.yml +74 -0
  6. data/lib/generators/ruby_ui/install/install_generator.rb +21 -22
  7. data/lib/generators/ruby_ui/install/templates/ruby_ui.rb.erb +18 -0
  8. data/lib/generators/ruby_ui/install/templates/tailwind.css.erb +156 -0
  9. data/lib/generators/ruby_ui/javascript_utils.rb +21 -0
  10. data/lib/ruby_ui/accordion/accordion_controller.js +97 -0
  11. data/lib/ruby_ui/alert/alert.rb +1 -1
  12. data/lib/ruby_ui/alert_dialog/alert_dialog_content.rb +1 -1
  13. data/lib/ruby_ui/alert_dialog/alert_dialog_controller.js +31 -0
  14. data/lib/ruby_ui/alert_dialog/alert_dialog_footer.rb +1 -1
  15. data/lib/ruby_ui/alert_dialog/alert_dialog_header.rb +1 -1
  16. data/lib/ruby_ui/breadcrumb/breadcrumb.rb +17 -0
  17. data/lib/ruby_ui/breadcrumb/breadcrumb_ellipsis.rb +39 -0
  18. data/lib/ruby_ui/breadcrumb/breadcrumb_item.rb +17 -0
  19. data/lib/ruby_ui/breadcrumb/breadcrumb_link.rb +22 -0
  20. data/lib/ruby_ui/breadcrumb/breadcrumb_list.rb +17 -0
  21. data/lib/ruby_ui/breadcrumb/breadcrumb_page.rb +19 -0
  22. data/lib/ruby_ui/breadcrumb/breadcrumb_separator.rb +38 -0
  23. data/lib/ruby_ui/calendar/calendar_controller.js +249 -0
  24. data/lib/ruby_ui/calendar/calendar_input_controller.js +8 -0
  25. data/lib/ruby_ui/carousel/carousel.rb +44 -0
  26. data/lib/ruby_ui/carousel/carousel_content.rb +23 -0
  27. data/lib/ruby_ui/carousel/carousel_controller.js +60 -0
  28. data/lib/ruby_ui/carousel/carousel_item.rb +23 -0
  29. data/lib/ruby_ui/carousel/carousel_next.rb +48 -0
  30. data/lib/ruby_ui/carousel/carousel_previous.rb +49 -0
  31. data/lib/ruby_ui/chart/chart_controller.js +103 -0
  32. data/lib/ruby_ui/checkbox/checkbox_group_controller.js +21 -0
  33. data/lib/ruby_ui/clipboard/clipboard_controller.js +54 -0
  34. data/lib/ruby_ui/collapsible/collapsible_controller.js +47 -0
  35. data/lib/ruby_ui/combobox/combobox.rb +8 -6
  36. data/lib/ruby_ui/combobox/combobox_checkbox.rb +25 -0
  37. data/lib/ruby_ui/combobox/combobox_controller.js +176 -0
  38. data/lib/ruby_ui/combobox/{combobox_empty.rb → combobox_empty_state.rb} +2 -2
  39. data/lib/ruby_ui/combobox/combobox_item.rb +9 -37
  40. data/lib/ruby_ui/combobox/combobox_list.rb +2 -11
  41. data/lib/ruby_ui/combobox/combobox_list_group.rb +20 -0
  42. data/lib/ruby_ui/combobox/combobox_popover.rb +30 -0
  43. data/lib/ruby_ui/combobox/combobox_radio.rb +26 -0
  44. data/lib/ruby_ui/combobox/combobox_search_input.rb +21 -24
  45. data/lib/ruby_ui/combobox/combobox_toggle_all_checkbox.rb +25 -0
  46. data/lib/ruby_ui/combobox/combobox_trigger.rb +25 -20
  47. data/lib/ruby_ui/command/command_controller.js +136 -0
  48. data/lib/ruby_ui/context_menu/context_menu_controller.js +144 -0
  49. data/lib/ruby_ui/dialog/dialog_content.rb +2 -2
  50. data/lib/ruby_ui/dialog/dialog_controller.js +32 -0
  51. data/lib/ruby_ui/dialog/dialog_footer.rb +1 -1
  52. data/lib/ruby_ui/dialog/dialog_header.rb +1 -1
  53. data/lib/ruby_ui/dropdown_menu/dropdown_menu_controller.js +120 -0
  54. data/lib/ruby_ui/form/form_field_controller.js +61 -0
  55. data/lib/ruby_ui/hover_card/hover_card_controller.js +144 -0
  56. data/lib/ruby_ui/masked_input/masked_input_controller.js +9 -0
  57. data/lib/ruby_ui/popover/popover_controller.js +107 -0
  58. data/lib/ruby_ui/progress/progress.rb +37 -0
  59. data/lib/ruby_ui/radio_button/radio_button.rb +4 -1
  60. data/lib/ruby_ui/select/select_content.rb +1 -1
  61. data/lib/ruby_ui/select/select_controller.js +171 -0
  62. data/lib/ruby_ui/select/select_item_controller.js +11 -0
  63. data/lib/ruby_ui/select/select_value.rb +1 -1
  64. data/lib/ruby_ui/separator/separator.rb +38 -0
  65. data/lib/ruby_ui/sheet/sheet_content.rb +1 -1
  66. data/lib/ruby_ui/sheet/sheet_content_controller.js +7 -0
  67. data/lib/ruby_ui/sheet/sheet_controller.js +9 -0
  68. data/lib/ruby_ui/{combobox/combobox_separator.rb → skeleton/skeleton.rb} +4 -2
  69. data/lib/ruby_ui/switch/switch.rb +24 -0
  70. data/lib/ruby_ui/tabs/tabs_controller.js +45 -0
  71. data/lib/ruby_ui/theme_toggle/theme_toggle_controller.js +30 -0
  72. data/lib/ruby_ui/tooltip/tooltip_controller.js +37 -0
  73. data/lib/ruby_ui.rb +1 -1
  74. metadata +57 -11
  75. data/lib/ruby_ui/combobox/combobox_content.rb +0 -31
  76. data/lib/ruby_ui/combobox/combobox_group.rb +0 -38
  77. data/lib/ruby_ui/combobox/combobox_input.rb +0 -22
  78. 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: 518fa1b05606796f9ae57a444aa55f5a5f0fcad5280c1e0ed75fa4778dbe978f
4
- data.tar.gz: 58c2a5717af15d0326258044a7ffc862d06cb3f3c3cc47f54a269b465bca2bcf
3
+ metadata.gz: bef04b912f756edd217e2294608bc08eef650d0e944145ef052b5a1760860682
4
+ data.tar.gz: '0792ce039b5817ce7fd9b6965c20672497ba6615ab555b2965110ce6a6817ed2'
5
5
  SHA512:
6
- metadata.gz: c17770afaf73e476ba5d2a52bd60d4f4ab00c80e84f7390e9870099cd85a827f81992a6a8d1dd2fd246b34785b6fb2640a95d5f7ad5b08b697a710eb29c89db5
7
- data.tar.gz: d70e491efc576db70d0fa7ea559ce16472a7fdd4f2a7a3a770ec15bc62b25da2ec12d12932baed3a966b9d44f2f0cb8704235f3bb016397b5e483cfe30cb1645
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 component files"
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
- return if related_components_file_paths.empty?
34
-
35
- say "Generating related components"
22
+ say "Generating components"
36
23
 
37
- related_components_file_paths.each do |file_path|
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 main_component_file_path = File.join(component_folder_path, "#{component_folder_name}.rb")
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
- if Gem::Specification.find_by_name("phlex-rails").version < "2.0.0.beta2"
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 --github="phlex-ruby/phlex-rails")
20
+ run %(bundle add phlex-rails)
26
21
 
27
- say "Running phlex-rails structure"
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 RubyUI styles to application css"
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 File.exist?(Rails.root.join("tailwind.config.js")) # tailwindcss js package
64
- template "tailwind.config.js.js-package.erb", Rails.root.join("tailwind.config.js")
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
- say "Cannot find tailwind.config.js. You will need to install tailwind config manually", :red
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 install_tailwind_animate
73
- say "Installing tailwindcss-animate plugin"
62
+ def install_tailwind_plugins
63
+ say "Installing tw-animate-css plugin"
64
+ install_js_package("tw-animate-css")
74
65
 
75
- install_js_package("tailwindcss-animate")
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
+ }
@@ -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]:left-4 [&>svg]:top-4 [&>svg~*]:pl-8"
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 grid w-full max-w-lg 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",
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
+ }