lightning_ui_kit 0.1.1

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 (72) hide show
  1. checksums.yaml +7 -0
  2. data/MIT-LICENSE +20 -0
  3. data/README.md +33 -0
  4. data/Rakefile +5 -0
  5. data/app/assets/builds/lightning_ui.css +2710 -0
  6. data/app/assets/builds/lightning_ui.js +6 -0
  7. data/app/assets/builds/lightning_ui.js.map +7 -0
  8. data/app/assets/stylesheets/lightning_ui/application.css +1 -0
  9. data/app/components/lightning_ui/alert_component.html.erb +4 -0
  10. data/app/components/lightning_ui/alert_component.rb +16 -0
  11. data/app/components/lightning_ui/avatar_component.html.erb +20 -0
  12. data/app/components/lightning_ui/avatar_component.rb +20 -0
  13. data/app/components/lightning_ui/badge_component.html.erb +9 -0
  14. data/app/components/lightning_ui/badge_component.rb +43 -0
  15. data/app/components/lightning_ui/banner_component.html.erb +17 -0
  16. data/app/components/lightning_ui/banner_component.rb +35 -0
  17. data/app/components/lightning_ui/base_component.rb +9 -0
  18. data/app/components/lightning_ui/button_component.html.erb +13 -0
  19. data/app/components/lightning_ui/button_component.rb +75 -0
  20. data/app/components/lightning_ui/checkbox_component.html.erb +37 -0
  21. data/app/components/lightning_ui/checkbox_component.rb +14 -0
  22. data/app/components/lightning_ui/description_list/item_component.html.erb +13 -0
  23. data/app/components/lightning_ui/description_list/item_component.rb +7 -0
  24. data/app/components/lightning_ui/description_list_component.html.erb +5 -0
  25. data/app/components/lightning_ui/description_list_component.rb +5 -0
  26. data/app/components/lightning_ui/dropdown/item_component.rb +5 -0
  27. data/app/components/lightning_ui/dropdown_component.html.erb +35 -0
  28. data/app/components/lightning_ui/dropdown_component.rb +23 -0
  29. data/app/components/lightning_ui/input_component.html.erb +99 -0
  30. data/app/components/lightning_ui/input_component.rb +44 -0
  31. data/app/components/lightning_ui/link_component.html.erb +7 -0
  32. data/app/components/lightning_ui/link_component.rb +8 -0
  33. data/app/components/lightning_ui/modal_component.html.erb +27 -0
  34. data/app/components/lightning_ui/modal_component.rb +21 -0
  35. data/app/components/lightning_ui/pagination_component.html.erb +21 -0
  36. data/app/components/lightning_ui/pagination_component.rb +42 -0
  37. data/app/components/lightning_ui/select_component.html.erb +20 -0
  38. data/app/components/lightning_ui/select_component.rb +25 -0
  39. data/app/components/lightning_ui/sidebar_component.html.erb +1 -0
  40. data/app/components/lightning_ui/sidebar_component.rb +4 -0
  41. data/app/components/lightning_ui/skeleton_component.html.erb +10 -0
  42. data/app/components/lightning_ui/skeleton_component.rb +8 -0
  43. data/app/components/lightning_ui/spinner_component.html.erb +10 -0
  44. data/app/components/lightning_ui/spinner_component.rb +4 -0
  45. data/app/components/lightning_ui/switch_component.html.erb +38 -0
  46. data/app/components/lightning_ui/switch_component.rb +31 -0
  47. data/app/components/lightning_ui/table/action_component.rb +9 -0
  48. data/app/components/lightning_ui/table/column_component.rb +12 -0
  49. data/app/components/lightning_ui/table_component.html.erb +40 -0
  50. data/app/components/lightning_ui/table_component.rb +16 -0
  51. data/app/components/lightning_ui/text_component.html.erb +3 -0
  52. data/app/components/lightning_ui/text_component.rb +12 -0
  53. data/app/components/lightning_ui/textarea_component.html.erb +48 -0
  54. data/app/components/lightning_ui/textarea_component.rb +42 -0
  55. data/app/helpers/lightning_ui/application_helper.rb +7 -0
  56. data/app/helpers/lightning_ui/heroicon_helper.rb +14 -0
  57. data/app/javascript/lightning_ui/controllers/accordion_controller.js +51 -0
  58. data/app/javascript/lightning_ui/controllers/banner_controller.js +11 -0
  59. data/app/javascript/lightning_ui/controllers/checkbox_controller.js +18 -0
  60. data/app/javascript/lightning_ui/controllers/clipboard_controller.js +21 -0
  61. data/app/javascript/lightning_ui/controllers/dropdown_controller.js +22 -0
  62. data/app/javascript/lightning_ui/controllers/main_controller.js +17 -0
  63. data/app/javascript/lightning_ui/controllers/modal_controller.js +44 -0
  64. data/app/javascript/lightning_ui/controllers/reveal_controller.js +22 -0
  65. data/app/javascript/lightning_ui/controllers/switch_controller.js +18 -0
  66. data/app/javascript/lightning_ui/index.js +31 -0
  67. data/config/initializers/heroicons.rb +5 -0
  68. data/lib/lightning_ui/engine.rb +20 -0
  69. data/lib/lightning_ui/version.rb +3 -0
  70. data/lib/lightning_ui.rb +6 -0
  71. data/lib/tasks/lightning_ui_tasks.rake +4 -0
  72. metadata +200 -0
@@ -0,0 +1,20 @@
1
+ <%= tag.div data:, class:"[&>[data-slot=label]+[data-slot=control]]:mt-3 [&>[data-slot=label]+[data-slot=description]]:mt-1 [&>[data-slot=description]+[data-slot=control]]:mt-3 [&>[data-slot=control]+[data-slot=description]]:mt-3 [&>[data-slot=control]+[data-slot=error]]:mt-3 *:data-[slot=label]:font-medium" do %>
2
+ <% if @label %>
3
+ <label data-slot="label" class="text-base/6 text-zinc-950 select-none data-disabled:opacity-50 sm:text-sm/6">
4
+ <%= @label %>
5
+ </label>
6
+ <% end %>
7
+ <% if @description %>
8
+ <p data-slot="description" class="text-base/6 text-zinc-500 data-disabled:opacity-50 sm:text-sm/6">
9
+ <%= @description %>
10
+ </p>
11
+ <% end %>
12
+ <span data-slot="control" class="group relative block w-full before:absolute before:inset-px before:rounded-[calc(var(--radius-lg)-1px)] before:bg-white before:shadow-sm after:pointer-events-none after:absolute after:inset-0 after:rounded-lg after:ring-transparent after:ring-inset has-data-focus:after:ring-2 has-data-focus:after:ring-blue-500 has-data-disabled:opacity-50 has-data-disabled:before:bg-zinc-950/5 has-data-disabled:before:shadow-none">
13
+ <% if @form %>
14
+ <%= @form.select(@name, @options_for_select, {multiple: @multiple}, { class: "relative block w-full appearance-none rounded-lg py-[calc(--spacing(2.5)-1px)] sm:py-[calc(--spacing(1.5)-1px)] pr-[calc(--spacing(10)-1px)] pl-[calc(--spacing(3.5)-1px)] sm:pr-[calc(--spacing(9)-1px)] sm:pl-[calc(--spacing(3)-1px)] [&_optgroup]:font-semibold text-base/6 text-zinc-950 placeholder:text-zinc-500 sm:text-sm/6 border border-zinc-950/10 data-hover:border-zinc-950/20 bg-transparent focus:outline-hidden data-invalid:border-red-500 data-invalid:data-hover:border-red-500 data-disabled:border-zinc-950/20 data-disabled:opacity-100"}) %>
15
+ <% else %>
16
+ <%= select_tag(@name, @options_for_select, multiple: @multiple, class: "relative block w-full appearance-none rounded-lg py-[calc(--spacing(2.5)-1px)] sm:py-[calc(--spacing(1.5)-1px)] pr-[calc(--spacing(10)-1px)] pl-[calc(--spacing(3.5)-1px)] sm:pr-[calc(--spacing(9)-1px)] sm:pl-[calc(--spacing(3)-1px)] [&_optgroup]:font-semibold text-base/6 text-zinc-950 placeholder:text-zinc-500 sm:text-sm/6 border border-zinc-950/10 data-hover:border-zinc-950/20 bg-transparent focus:outline-hidden data-invalid:border-red-500 data-invalid:data-hover:border-red-500 data-disabled:border-zinc-950/20 data-disabled:opacity-100") %>
17
+ <% end %>
18
+ <span class="pointer-events-none absolute inset-y-0 right-0 flex items-center pr-2"><svg class="size-5 stroke-zinc-500 group-has-data-disabled:stroke-zinc-600 sm:size-4 forced-colors:stroke-[CanvasText]" viewBox="0 0 16 16" aria-hidden="true" fill="none"><path d="M5.75 10.75L8 13L10.25 10.75" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"></path><path d="M10.25 5.25L8 3L5.75 5.25" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"></path></svg></span>
19
+ </span>
20
+ <% end %>
@@ -0,0 +1,25 @@
1
+ # frozen_string_literal: true
2
+
3
+ class LightningUi::SelectComponent < LightningUi::BaseComponent
4
+ def initialize(name:, form: nil, label: nil, description: nil, disabled: false, options_for_select: [], multiple: false, **options)
5
+ @name = name
6
+ @form = form
7
+ @label = label
8
+ @multiple = multiple
9
+ @description = description
10
+ @disabled = disabled
11
+ @options_for_select = options_for_select
12
+ @options = options
13
+ end
14
+
15
+ def data
16
+ default_data = {
17
+ slot: "field",
18
+ action: "click->switch#toggle",
19
+ controller: "select",
20
+ disabled: @disabled
21
+ }
22
+
23
+ default_data.merge(@options[:data] || {})
24
+ end
25
+ end
@@ -0,0 +1 @@
1
+ <div>Add Sidebar template here</div>
@@ -0,0 +1,4 @@
1
+ # frozen_string_literal: true
2
+
3
+ class LightningUi::SidebarComponent < LightningUi::BaseComponent
4
+ end
@@ -0,0 +1,10 @@
1
+ <div role="status" class="animate-pulse">
2
+ <% if @with_title %>
3
+ <h3 class="h-3 bg-gray-300 rounded-full w-48 mb-4"></h3>
4
+ <% end %>
5
+ <div class="space-y-2.5">
6
+ <% @lines.times do %>
7
+ <p class="h-2 bg-gray-300 rounded-full"></p>
8
+ <% end %>
9
+ </div>
10
+ </div>
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ class LightningUi::SkeletonComponent < LightningUi::BaseComponent
4
+ def initialize(with_title: false, lines: 3)
5
+ @with_title = with_title
6
+ @lines = lines
7
+ end
8
+ end
@@ -0,0 +1,10 @@
1
+ <svg class="animate-spin border-zinc-500"
2
+ xmlns="http://www.w3.org/2000/svg" width="18" height="18"
3
+ viewBox="0 0 30 30" fill="none">
4
+ <path fill-rule="evenodd" clip-rule="evenodd"
5
+ d="M14.4304 0.0323927C14.0002 0.172039 13.7522 0.420298 13.6321 0.825659C13.5643 1.05258 13.5527 1.58013 13.5527 4.33426C13.5527 7.98251 13.5488 7.93596 13.9208 8.30835C14.4187 8.80487 15.5812 8.80487 16.0792 8.30835C16.4511 7.93596 16.4473 7.98251 16.4473 4.33426C16.4473 0.804324 16.4415 0.742259 16.1547 0.400903C15.9145 0.115792 15.6568 0.0207556 15.0775 0.00523939C14.8614 -0.00695113 14.6447 0.00214369 14.4304 0.0323927ZM24.4026 3.89593C24.1643 4.00648 23.7574 4.38857 21.816 6.33779C20.0316 8.12797 19.4872 8.70595 19.4 8.89602C19.3276 9.05362 19.2902 9.2251 19.2905 9.39857C19.2909 9.57205 19.3289 9.74338 19.4019 9.9007C19.5376 10.1916 19.9832 10.6494 20.2835 10.8065C20.5877 10.9655 21.0391 10.9636 21.3549 10.8026C21.6785 10.6358 26.1541 6.15547 26.3207 5.83157C26.4001 5.66482 26.4417 5.4826 26.4427 5.29788C26.4437 5.11317 26.404 4.93051 26.3265 4.7629C26.1121 4.40396 25.8092 4.10611 25.4469 3.89787C25.2865 3.81116 25.1072 3.7656 24.925 3.76526C24.7427 3.76492 24.5633 3.80982 24.4026 3.89593ZM22.2559 13.5858C21.9939 13.6581 21.7571 13.8017 21.5719 14.0009C21.3065 14.3597 21.2232 15.0889 21.3898 15.6029C21.5022 15.952 21.785 16.2352 22.1338 16.3477C22.3702 16.4253 22.7557 16.435 25.6406 16.435C28.4093 16.435 28.9189 16.4233 29.1475 16.3554C29.719 16.1848 29.998 15.7348 30 14.9842C30 14.2375 29.7152 13.7894 29.1242 13.611C28.9247 13.5509 28.2989 13.5392 25.6484 13.5431C23.8698 13.547 22.343 13.5664 22.2559 13.5858ZM20.3339 19.3811C20.0277 19.5207 19.5763 19.963 19.4194 20.2772C19.3393 20.4342 19.2957 20.6073 19.2916 20.7836C19.2876 20.9599 19.3233 21.1348 19.3961 21.2954C19.5395 21.6154 24.0422 26.1384 24.3968 26.3188C24.8986 26.5767 25.4139 26.4507 25.9332 25.9425C26.4447 25.4421 26.5803 24.899 26.3207 24.3928C26.1405 24.0379 21.6223 19.5304 21.3026 19.3869C21.1498 19.3207 20.9853 19.286 20.8188 19.285C20.6524 19.284 20.4874 19.3167 20.3339 19.3811ZM14.5059 21.342C14.2442 21.4158 14.0076 21.56 13.822 21.759C13.5546 22.1197 13.5527 22.1449 13.5527 25.6322C13.5527 28.4096 13.5643 28.9197 13.6321 29.1485C13.8026 29.7207 14.254 30 15 30C15.7459 30 16.1973 29.7207 16.3678 29.1485C16.488 28.7374 16.486 22.5018 16.364 22.143C16.3264 21.9983 16.2577 21.8636 16.1627 21.7484C16.0676 21.6331 15.9485 21.5401 15.8137 21.4758C15.6006 21.3594 15.4766 21.3303 15.1104 21.3167C14.9086 21.3046 14.706 21.313 14.5059 21.342Z"
6
+ fill="#9F9FA9" />
7
+ <path fill-rule="evenodd" clip-rule="evenodd"
8
+ d="M4.83402 3.86487C4.5279 4.00452 4.07647 4.44673 3.91953 4.76093C3.83948 4.918 3.79581 5.09111 3.79179 5.2674C3.78776 5.44369 3.82348 5.61861 3.89628 5.77918C4.03965 6.0992 8.54237 10.6222 8.89693 10.8026C9.39873 11.0605 9.91411 10.9344 10.4334 10.4263C10.9448 9.92589 11.0805 9.38282 10.8208 8.8766C10.6407 8.52167 6.12245 4.01421 5.80276 3.87069C5.65 3.80447 5.48545 3.76982 5.31899 3.76882C5.15252 3.76782 4.98757 3.80049 4.83402 3.86487ZM0.85443 13.6129C0.282873 13.7933 0 14.2491 0 14.9842C0.00193748 15.7348 0.280935 16.1847 0.852493 16.3554C1.08112 16.4233 1.59067 16.4349 4.35934 16.4349C7.24425 16.4349 7.62981 16.4252 7.86618 16.3477C8.03981 16.2905 8.19761 16.1934 8.32684 16.064C8.45607 15.9347 8.55313 15.7767 8.61018 15.6029C8.7768 15.0889 8.69349 14.3596 8.42805 13.9989C8.32166 13.8831 8.19656 13.7861 8.058 13.7118L7.79256 13.5644L4.4349 13.5528C1.73211 13.545 1.03268 13.5567 0.85443 13.6129ZM8.90274 19.4121C8.66443 19.5227 8.25756 19.9047 6.3162 21.854C4.53177 23.6442 3.98734 24.2221 3.90016 24.4122C3.82771 24.5698 3.79037 24.7413 3.7907 24.9148C3.79104 25.0882 3.82904 25.2596 3.90209 25.4169C4.03772 25.7078 4.48334 26.1655 4.78365 26.3226C5.08783 26.4817 5.53927 26.4797 5.85508 26.3188C6.17864 26.152 10.6542 21.6717 10.8208 21.3478C10.9002 21.181 10.9419 20.9988 10.9429 20.8141C10.9439 20.6294 10.9042 20.4467 10.8267 20.2791C10.6122 19.9201 10.3094 19.6223 9.94704 19.414C9.7867 19.3273 9.60738 19.2818 9.42513 19.2814C9.24289 19.2811 9.06341 19.326 8.90274 19.4121Z"
9
+ fill="#D1D5DB" />
10
+ </svg>
@@ -0,0 +1,4 @@
1
+ # frozen_string_literal: true
2
+
3
+ class LightningUi::SpinnerComponent < LightningUi::BaseComponent
4
+ end
@@ -0,0 +1,38 @@
1
+ <div
2
+ data-slot="field"
3
+ class="grid grid-cols-[1fr_auto] items-center gap-x-8 gap-y-1 sm:grid-cols-[1fr_auto] *:data-[slot=control]:col-start-2 *:data-[slot=control]:self-center *:data-[slot=label]:col-start-1 *:data-[slot=label]:row-start-1 *:data-[slot=label]:justify-self-start *:data-[slot=description]:col-start-1 *:data-[slot=description]:row-start-2 has-data-[slot=description]:**:data-[slot=label]:font-medium"
4
+ data-controller="lui-switch"
5
+ >
6
+ <% if @label %>
7
+ <label data-slot="label" class="text-base/6 text-zinc-950 select-none data-disabled:opacity-50 sm:text-sm/6">
8
+ <%= @label %>
9
+ </label>
10
+ <% end %>
11
+ <% if @description %>
12
+ <p data-slot="description" class="text-base/6 text-zinc-500 data-disabled:opacity-50 sm:text-sm/6">
13
+ <%= @description %>
14
+ </p>
15
+ <% end %>
16
+ <%= tag.button(
17
+ class: "group relative isolate inline-flex h-6 w-10 cursor-default rounded-full p-[3px] sm:h-5 sm:w-8 transition duration-0 ease-in-out data-changing:duration-200 forced-colors:outline forced-colors:[--switch-bg:Highlight] ] bg-zinc-200 ring-1 ring-black/5 ring-inset data-checked:bg-(--switch-bg) data-checked:ring-(--switch-bg-ring) ) ) focus:outline-hidden data-focus:outline-2 data-focus:outline-offset-2 data-focus:outline-blue-500 data-hover:ring-black/15 data-hover:data-checked:ring-(--switch-bg-ring) ) data-disabled:bg-zinc-200 data-disabled:opacity-50 data-disabled:data-checked:bg-zinc-200 data-disabled:data-checked:ring-black/5 [--switch-bg-ring:var(--color-zinc-950)]/90 [--switch-bg:var(--color-zinc-900)] ] [--switch-ring:var(--color-zinc-950)]/90 [--switch-shadow:var(--color-black)]/10 [--switch:white]",
18
+ aria: { checked: @enabled },
19
+ role: "switch",
20
+ type: "button",
21
+ tabindex: 0,
22
+ data:
23
+ ) do %>
24
+ <span
25
+ aria-hidden="true"
26
+ class="pointer-events-none relative inline-block size-[1.125rem] rounded-full sm:size-3.5 translate-x-0 transition duration-200 ease-in-out border border-transparent bg-white ring-1 shadow-sm ring-black/5 group-data-checked:bg-(--switch) group-data-checked:shadow-(--switch-shadow) group-data-checked:ring-(--switch-ring) group-data-checked:translate-x-4 sm:group-data-checked:translate-x-3 group-data-checked:group-data-disabled:bg-white group-data-checked:group-data-disabled:shadow-sm group-data-checked:group-data-disabled:ring-black/5">
27
+ </span>
28
+ <% if @form %>
29
+ <%= @form.hidden_field(@name, value: @enabled, data: { lui_switch_target: "field" }) %>
30
+ <% else %>
31
+ <%= hidden_field_tag(
32
+ @name,
33
+ @enabled,
34
+ data: { lui_switch_target: "field" }
35
+ ) %>
36
+ <% end %>
37
+ <% end %>
38
+ </div>
@@ -0,0 +1,31 @@
1
+ # frozen_string_literal: true
2
+
3
+ class LightningUi::SwitchComponent < LightningUi::BaseComponent
4
+ def initialize(name:, form: nil, label: nil, description: nil, enabled: false, disabled: false, **options)
5
+ @name = name
6
+ @form = form
7
+ @label = label
8
+ @description = description
9
+ @enabled = enabled
10
+ @disabled = disabled
11
+ @options = options
12
+ end
13
+
14
+ def data
15
+ default_data.merge(@options[:data] || {})
16
+ end
17
+
18
+ def default_data
19
+ {
20
+ slot: "control",
21
+ action: "click->lui-switch#toggle"
22
+ }.tap do |data|
23
+ if @disabled
24
+ data[:disabled] = "true"
25
+ end
26
+ if @enabled
27
+ data[:checked] = "true"
28
+ end
29
+ end
30
+ end
31
+ end
@@ -0,0 +1,9 @@
1
+ class LightningUi::Table::ActionComponent < LightningUi::BaseComponent
2
+ def initialize(&block)
3
+ @block = block
4
+ end
5
+
6
+ def call(row)
7
+ @block.call(row)
8
+ end
9
+ end
@@ -0,0 +1,12 @@
1
+ class LightningUi::Table::ColumnComponent < LightningUi::BaseComponent
2
+ attr_reader :title
3
+
4
+ def initialize(title, &block)
5
+ @title = title
6
+ @block = block
7
+ end
8
+
9
+ def call(row)
10
+ @block.call(row)
11
+ end
12
+ end
@@ -0,0 +1,40 @@
1
+ <div class="border-zinc-950/5 border rounded-lg">
2
+ <table class="min-w-full text-left text-sm/6 text-zinc-950 border-collapse border-inherit">
3
+ <thead class="text-zinc-500">
4
+ <tr class="">
5
+ <% columns.each do |column| %>
6
+ <th class="border-b border-b-zinc-950/10 px-4 py-2 font-medium"><%= column.title %></th>
7
+ <% end %>
8
+ <% if actions.present? %>
9
+ <th class="border-b border-b-zinc-950/10 px-4 py-2 font-medium"><%= @actions_title %></th>
10
+ <% end %>
11
+ </tr>
12
+ </thead>
13
+ <tbody>
14
+ <% if @data.empty? %>
15
+ <tr>
16
+ <td class="p-4 text-center" colspan="<%= columns.size + (actions.present? ? 1 : 0) %>">
17
+ <p class="font-medium"><%= @empty_message %></p>
18
+ </td>
19
+ </tr>
20
+ <% else %>
21
+ <% @data.each do |row| %>
22
+ <tr class="group">
23
+ <% columns.each do |column| %>
24
+ <td class="p-4 border-b border-zinc-950/5 group-last:border-b-0">
25
+ <%= column.call(row) %>
26
+ </td>
27
+ <% end %>
28
+ <% if actions.present? %>
29
+ <td class="p-4 flex gap-3 border-b border-zinc-950/5 group-last:border-b-0">
30
+ <% actions.each do |action| %>
31
+ <%= action.call(row) %>
32
+ <% end %>
33
+ </td>
34
+ <% end %>
35
+ </tr>
36
+ <% end %>
37
+ <% end %>
38
+ </tbody>
39
+ </table>
40
+ </div>
@@ -0,0 +1,16 @@
1
+ # frozen_string_literal: true
2
+
3
+ class LightningUi::TableComponent < LightningUi::BaseComponent
4
+ renders_many :columns, ->(title, &block) do
5
+ LightningUi::Table::ColumnComponent.new(title, &block)
6
+ end
7
+ renders_many :actions, ->(&block) do
8
+ LightningUi::Table::ActionComponent.new(&block)
9
+ end
10
+
11
+ def initialize(data:, actions_title: "Actions", empty_message: "No data available")
12
+ @data = data
13
+ @actions_title = actions_title
14
+ @empty_message = empty_message
15
+ end
16
+ end
@@ -0,0 +1,3 @@
1
+ <%= tag.p(class: classes) do %>
2
+ <%= content %>
3
+ <% end %>
@@ -0,0 +1,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ class LightningUi::TextComponent < LightningUi::BaseComponent
4
+ def initialize(size: :md, **options)
5
+ @size = size
6
+ @options = options
7
+ end
8
+
9
+ def classes
10
+ merge_classes(["text-zinc-600 text-#{@size}", @options[:class]].compact.join(" "))
11
+ end
12
+ end
@@ -0,0 +1,48 @@
1
+ <%= tag.div(
2
+ class: "[&>[data-slot=label]+[data-slot=control]]:mt-3 [&>[data-slot=label]+[data-slot=description]]:mt-1 [&>[data-slot=description]+[data-slot=control]]:mt-3 [&>[data-slot=control]+[data-slot=description]]:mt-3 [&>[data-slot=control]+[data-slot=error]]:mt-3 *:data-[slot=label]:font-medium",
3
+ data:
4
+ ) do %>
5
+ <% if @label %>
6
+ <%= tag.label(
7
+ @label,
8
+ class: "text-base/6 text-zinc-950 select-none data-disabled:opacity-50 sm:text-sm/6",
9
+ data: label_data
10
+ ) %>
11
+ <% end %>
12
+ <% if @description %>
13
+ <%= tag.p(
14
+ @description,
15
+ class: "text-base/6 text-zinc-500 data-disabled:opacity-50 sm:text-sm/6",
16
+ data: description_data
17
+ ) %>
18
+ <% end %>
19
+ <% if @form %>
20
+ <span data-slot="control" class="relative block w-full before:absolute before:inset-px before:rounded-[calc(var(--radius-lg)-1px)] before:bg-white before:shadow-sm dark:before:hidden after:pointer-events-none after:absolute after:inset-0 after:rounded-lg after:ring-transparent after:ring-inset sm:focus-within:after:ring-2 sm:focus-within:after:ring-blue-500 has-data-disabled:opacity-50 has-data-disabled:before:bg-zinc-950/5 has-data-disabled:before:shadow-none has-data-invalid:before:shadow-red-500/10">
21
+ <%= @form.text_area(
22
+ @name,
23
+ value: @value,
24
+ rows: @rows,
25
+ cols: @cols,
26
+ multiple: @multiple,
27
+ data: input_data,
28
+ class: "relative block w-full appearance-none rounded-lg px-[calc(--spacing(3.5)-1px)] py-[calc(--spacing(2.5)-1px)] sm:px-[calc(--spacing(3)-1px)] sm:py-[calc(--spacing(1.5)-1px)] text-base/6 text-zinc-950 placeholder:text-zinc-500 sm:text-sm/6 border border-zinc-950/10 data-hover:border-zinc-950/20 bg-transparent focus:outline-hidden data-invalid:border-red-500 data-invalid:data-hover:border-red-500 dark:data-invalid:data-hover:border-red-500 data-disabled:border-zinc-950/20",
29
+ disabled: @disabled,
30
+ autofocus: @autofocus
31
+ ) %>
32
+ </span>
33
+ <% else %>
34
+ <span data-slot="control" class="relative block w-full before:absolute before:inset-px before:rounded-[calc(var(--radius-lg)-1px)] before:bg-white before:shadow-sm dark:before:hidden after:pointer-events-none after:absolute after:inset-0 after:rounded-lg after:ring-transparent after:ring-inset sm:focus-within:after:ring-2 sm:focus-within:after:ring-blue-500 has-data-disabled:opacity-50 has-data-disabled:before:bg-zinc-950/5 has-data-disabled:before:shadow-none has-data-invalid:before:shadow-red-500/10">
35
+ <%= text_area_tag(
36
+ @name,
37
+ @value,
38
+ rows: @rows,
39
+ cols: @cols,
40
+ multiple: @multiple,
41
+ data: input_data,
42
+ class: "relative block w-full appearance-none rounded-lg px-[calc(--spacing(3.5)-1px)] py-[calc(--spacing(2.5)-1px)] sm:px-[calc(--spacing(3)-1px)] sm:py-[calc(--spacing(1.5)-1px)] text-base/6 text-zinc-950 placeholder:text-zinc-500 sm:text-sm/6 border border-zinc-950/10 data-hover:border-zinc-950/20 bg-transparent focus:outline-hidden data-invalid:border-red-500 data-invalid:data-hover:border-red-500 dark:data-invalid:data-hover:border-red-500 data-disabled:border-zinc-950/20",
43
+ disabled: @disabled,
44
+ autofocus: @autofocus
45
+ ) %>
46
+ </span>
47
+ <% end %>
48
+ <% end %>
@@ -0,0 +1,42 @@
1
+ # frozen_string_literal: true
2
+
3
+ class LightningUi::TextareaComponent < LightningUi::BaseComponent
4
+ def initialize(name:, value: nil, autofocus: false, label: nil, form: nil, type: :text, description: nil, disabled: false, multiple: false, rows: 3, cols: nil, **options)
5
+ @name = name
6
+ @value = value
7
+ @disabled = disabled
8
+ @autofocus = autofocus
9
+ @rows = rows
10
+ @multiple = multiple
11
+ @cols = cols
12
+ @label = label
13
+ @form = form
14
+ @type = type
15
+ @description = description
16
+ @options = options
17
+ end
18
+
19
+ def data
20
+ @options[:data] || {}
21
+ end
22
+
23
+ def input_data
24
+ @options[:input_data] || {}
25
+ end
26
+
27
+ def label_data
28
+ {slot: "label"}.merge(@options[:label_data] || {}).tap do |data|
29
+ if @disabled
30
+ data[:disabled] = "true"
31
+ end
32
+ end
33
+ end
34
+
35
+ def description_data
36
+ {slot: "description"}.merge(@options[:description_data] || {}).tap do |data|
37
+ if @disabled
38
+ data[:disabled] = "true"
39
+ end
40
+ end
41
+ end
42
+ end
@@ -0,0 +1,7 @@
1
+ module LightningUi
2
+ module ApplicationHelper
3
+ def lui_asset_path(file)
4
+ "/lightning_ui-assets/#{file}".gsub("//", "/")
5
+ end
6
+ end
7
+ end
@@ -0,0 +1,14 @@
1
+ # require "heroicon"
2
+
3
+ module LightningUi
4
+ module HeroiconHelper
5
+ def heroicon(name, variant: Heroicons.configuration.variant, options: {}, path_options: {})
6
+ raw Heroicons::Icon.render(
7
+ name: name,
8
+ variant: variant,
9
+ options: options,
10
+ path_options: path_options
11
+ )
12
+ end
13
+ end
14
+ end
@@ -0,0 +1,51 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+
3
+ export default class extends Controller {
4
+ static targets = ["content", "item"];
5
+
6
+ connect() {
7
+ this.openItem(this.itemTargets[0])
8
+ }
9
+
10
+ toggle(e) {
11
+ e.preventDefault();
12
+
13
+ this.itemTargets.forEach(item => {
14
+ if (item.contains(e.target)) {
15
+ if (this.isOpen(item)) {
16
+ this.closeItem(item);
17
+ } else {
18
+ this.openItem(item);
19
+ }
20
+ } else {
21
+ this.closeItem(item);
22
+ }
23
+ });
24
+ }
25
+
26
+ openItem(item) {
27
+ const content = item.querySelector("[data-accordion-target=content]");
28
+ content.classList.remove("grid-rows-[0fr]");
29
+ content.classList.add("grid-rows-[1fr]");
30
+ content.classList.remove("opacity-0");
31
+ content.classList.add("opacity-100");
32
+ const arrow = item.querySelector("[data-accordion-target=arrow]");
33
+ arrow.classList.add("rotate-180");
34
+ }
35
+
36
+ closeItem(item) {
37
+ const content = item.querySelector("[data-accordion-target=content]");
38
+ content.classList.remove("grid-rows-[1fr]");
39
+ content.classList.add("grid-rows-[0fr]");
40
+ content.classList.remove("opacity-100");
41
+ content.classList.add("opacity-0");
42
+ const arrow = item.querySelector("[data-accordion-target=arrow]");
43
+ arrow.classList.remove("rotate-180");
44
+ }
45
+
46
+ isOpen(item) {
47
+ const content = item.querySelector("[data-accordion-target=content]");
48
+ return content.classList.contains("grid-rows-[1fr]");
49
+ }
50
+ }
51
+
@@ -0,0 +1,11 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ export default class extends Controller {
4
+ close() {
5
+ this.element.classList.remove("opacity-100");
6
+ this.element.classList.add("opacity-0");
7
+ setTimeout(() => {
8
+ this.element.classList.add("hidden");
9
+ }, 300);
10
+ }
11
+ }
@@ -0,0 +1,18 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ export default class extends Controller {
4
+ static targets = ["field", "control"]
5
+
6
+ toggle(event) {
7
+ event.preventDefault();
8
+
9
+ const checkbox = this.controlTarget
10
+ if (checkbox.dataset.checked) {
11
+ this.fieldTarget.value = false;
12
+ delete checkbox.dataset.checked;
13
+ } else {
14
+ this.fieldTarget.value = true;
15
+ checkbox.dataset.checked = true;
16
+ }
17
+ }
18
+ }
@@ -0,0 +1,21 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ export default class extends Controller {
4
+ static targets = ["input"]
5
+
6
+ connect() {
7
+ if (this.hasInputTarget) {
8
+ this.inputTarget.disabled = true
9
+ }
10
+ }
11
+
12
+ copy(e) {
13
+ e.preventDefault()
14
+ if (!this.hasInputTarget) {
15
+ return
16
+ }
17
+
18
+ const text = this.inputTarget.innerHTML || this.inputTarget.value
19
+ navigator.clipboard.writeText(text)
20
+ }
21
+ }
@@ -0,0 +1,22 @@
1
+ import { Controller } from "@hotwired/stimulus";
2
+ import { useTransition } from "stimulus-use";
3
+
4
+ export default class Dropdown extends Controller {
5
+ static targets = ["menu"];
6
+
7
+ connect() {
8
+ useTransition(this, {
9
+ element: this.menuTarget,
10
+ });
11
+ }
12
+
13
+ toggle() {
14
+ this.toggleTransition();
15
+ }
16
+
17
+ hide(event) {
18
+ if (!this.element.contains(event.target) && !this.menuTarget.classList.contains("hidden")) {
19
+ this.leave();
20
+ }
21
+ }
22
+ }
@@ -0,0 +1,17 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ export default class extends Controller {
4
+ openModal() {
5
+ this.findElement("lui-modal").open()
6
+ }
7
+
8
+ closeModal() {
9
+ this.findElement("lui-modal").close()
10
+ }
11
+
12
+ findElement(type) {
13
+ const targetId = this.element.dataset.target.replace("#", "")
14
+ const target = document.getElementById(targetId)
15
+ return this.application.getControllerForElementAndIdentifier(target, type)
16
+ }
17
+ }
@@ -0,0 +1,44 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ export default class extends Controller {
4
+ static targets = ["dialog"]
5
+ static values = {
6
+ open: Boolean
7
+ }
8
+
9
+ connect() {
10
+ this.dialogTarget.addEventListener('click', this.onClick.bind(this))
11
+ if (this.openValue) {
12
+ this.open()
13
+ }
14
+ }
15
+
16
+ disconnect() {
17
+ this.dialogTarget.removeEventListener('click', this.onClick.bind(this))
18
+ }
19
+
20
+ open() {
21
+ this.dialogTarget.showModal()
22
+ }
23
+
24
+ close() {
25
+ this.dialogTarget.setAttribute("closing", "")
26
+
27
+ Promise.all(
28
+ this.dialogTarget.getAnimations().map((animation) => animation.finished),
29
+ ).then(() => {
30
+ this.dialogTarget.removeAttribute("closing")
31
+ this.dialogTarget.close()
32
+ })
33
+ }
34
+
35
+ onClick(event) {
36
+ if (event.target === this.dialogTarget) {
37
+ this.dialogTarget.close()
38
+ }
39
+ }
40
+
41
+ submitForm() {
42
+ this.dialogTarget.querySelector("form").requestSubmit()
43
+ }
44
+ }
@@ -0,0 +1,22 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ export default class extends Controller {
4
+ static targets = ["item"]
5
+ static values = { initialShow: Boolean }
6
+
7
+ connect() {
8
+ if (!this.initialShowValue) {
9
+ this.itemTargets.forEach(item => {
10
+ item.classList.add("hidden");
11
+ });
12
+ }
13
+ }
14
+
15
+ toggle(event) {
16
+ event.preventDefault();
17
+
18
+ this.itemTargets.forEach(item => {
19
+ item.classList.toggle("hidden");
20
+ });
21
+ }
22
+ }
@@ -0,0 +1,18 @@
1
+ import { Controller } from "@hotwired/stimulus"
2
+
3
+ export default class extends Controller {
4
+ static targets = ["field"]
5
+
6
+ toggle(event) {
7
+ event.preventDefault();
8
+
9
+ const button = event.target
10
+ if (button.dataset.checked) {
11
+ this.fieldTarget.value = false;
12
+ delete button.dataset.checked;
13
+ } else {
14
+ this.fieldTarget.value = true;
15
+ button.dataset.checked = true;
16
+ }
17
+ }
18
+ }