relay_ui 0.1.1 → 0.3.0

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 (92) hide show
  1. checksums.yaml +4 -4
  2. data/LICENSE.txt +21 -0
  3. data/README.md +100 -0
  4. data/lib/relay_ui/engine.rb +11 -14
  5. data/lib/relay_ui/version.rb +2 -2
  6. data/lib/relay_ui.rb +1 -1
  7. data/lib/rui/badges/base.rb +9 -0
  8. data/lib/rui/badges/blue.rb +7 -0
  9. data/lib/rui/badges/gray.rb +7 -0
  10. data/lib/rui/badges/green.rb +7 -0
  11. data/lib/rui/badges/purple.rb +7 -0
  12. data/lib/rui/badges/red.rb +7 -0
  13. data/lib/rui/badges/yellow.rb +7 -0
  14. data/lib/rui/badges.rb +3 -0
  15. data/lib/rui/base.rb +8 -0
  16. data/lib/rui/buttons/base.rb +29 -0
  17. data/lib/rui/buttons/destructive.rb +7 -0
  18. data/lib/rui/buttons/ghost.rb +7 -0
  19. data/lib/rui/buttons/outline.rb +7 -0
  20. data/lib/rui/buttons/primary.rb +7 -0
  21. data/lib/rui/buttons/secondary.rb +7 -0
  22. data/lib/rui/buttons.rb +3 -0
  23. data/lib/rui/card.rb +92 -0
  24. data/lib/rui/flash.rb +51 -0
  25. data/lib/rui/forms/checkbox.rb +25 -0
  26. data/lib/rui/forms/email.rb +19 -0
  27. data/lib/rui/forms/field_group.rb +5 -0
  28. data/lib/rui/forms/label.rb +7 -0
  29. data/lib/rui/forms/password.rb +18 -0
  30. data/lib/rui/forms/phone.rb +18 -0
  31. data/lib/rui/forms/radio.rb +22 -0
  32. data/lib/rui/forms/select.rb +24 -0
  33. data/lib/rui/forms/text.rb +17 -0
  34. data/lib/rui/forms/textarea.rb +18 -0
  35. data/lib/rui/forms.rb +3 -0
  36. data/lib/rui/helpers.rb +3 -0
  37. data/lib/rui/icon.rb +13 -0
  38. data/lib/rui/layout/body.rb +7 -0
  39. data/lib/rui/layout/main.rb +5 -0
  40. data/lib/rui/layout/page.rb +5 -0
  41. data/lib/rui/layout.rb +3 -0
  42. data/lib/rui/links/base.rb +30 -0
  43. data/lib/rui/links/destructive.rb +7 -0
  44. data/lib/rui/links/ghost.rb +7 -0
  45. data/lib/rui/links/outline.rb +7 -0
  46. data/lib/rui/links/primary.rb +7 -0
  47. data/lib/rui/links/secondary.rb +7 -0
  48. data/lib/rui/links.rb +3 -0
  49. data/lib/rui/markdown/generator.rb +36 -0
  50. data/lib/rui/markdown/safe.rb +9 -0
  51. data/lib/rui/markdown/safe_renderer.rb +15 -0
  52. data/lib/rui/markdown/unsafe.rb +9 -0
  53. data/lib/rui/markdown/unsafe_renderer.rb +42 -0
  54. data/lib/rui/markdown.rb +3 -0
  55. data/lib/rui/navigation/menu_button.rb +7 -0
  56. data/lib/rui/navigation/sidebar.rb +43 -0
  57. data/lib/rui/navigation/top.rb +19 -0
  58. data/lib/rui/navigation.rb +3 -0
  59. data/{app/components → lib/rui}/slideout.rb +9 -9
  60. data/lib/rui/table.rb +54 -0
  61. data/lib/rui/text.rb +21 -0
  62. metadata +147 -37
  63. data/Rakefile +0 -64
  64. data/app/assets/javascripts/relay_ui/index.js +0 -5
  65. data/app/assets/stylesheets/relay_ui/application.css +0 -1
  66. data/app/components/badges/base.rb +0 -5
  67. data/app/components/badges/blue.rb +0 -5
  68. data/app/components/badges/gray.rb +0 -5
  69. data/app/components/badges/green.rb +0 -5
  70. data/app/components/badges/purple.rb +0 -5
  71. data/app/components/badges/red.rb +0 -5
  72. data/app/components/badges/yellow.rb +0 -5
  73. data/app/components/base.rb +0 -15
  74. data/app/components/buttons/base.rb +0 -30
  75. data/app/components/buttons/destructive.rb +0 -5
  76. data/app/components/buttons/ghost.rb +0 -5
  77. data/app/components/buttons/link.rb +0 -9
  78. data/app/components/buttons/outline.rb +0 -5
  79. data/app/components/buttons/primary.rb +0 -5
  80. data/app/components/buttons/secondary.rb +0 -5
  81. data/app/components/code_block.rb +0 -13
  82. data/app/components/headings.rb +0 -37
  83. data/app/components/icon.rb +0 -9
  84. data/app/components/lists/base.rb +0 -13
  85. data/app/components/lists/ordered.rb +0 -5
  86. data/app/components/lists/unordered.rb +0 -5
  87. data/app/components/navigation.rb +0 -49
  88. data/app/components/text.rb +0 -19
  89. data/package.json +0 -16
  90. data/tailwind.config.js +0 -10
  91. data/vendor/assets/javascripts/relay_ui/dist/relay_ui.js +0 -2471
  92. data/vendor/assets/stylesheets/relay_ui/relay_ui.css +0 -2528
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 48ac05153d26910dcdc24eebad0761c52fa937a8f126a78f2e64d25a8f436505
4
- data.tar.gz: 907342ba056ebb313354187fb6c34328c3029a818c7e024ad86915fe079dad2d
3
+ metadata.gz: 7158b2530047c016d6353aba1125ad761104ae2f553656c3d28c40cf6fc74f4a
4
+ data.tar.gz: 917a96410664ca64d92e4f41c86d7af74ebcbdc27511dbcf167fc74bd45bec26
5
5
  SHA512:
6
- metadata.gz: cd29c8eac7f421ba53e512403c940b6b71a586c3311507e005eac4b88f7fdc2b5919c116e4c78e4df210c41cf8fa8e05db808e3e5df25f272a067f34f74b17d6
7
- data.tar.gz: 40da871a68f19836353ba5a92faae889e3196d851daedc5be69f141cf2c6465dccb63613a9bfc770e2ec415ced6fce4be244e7456f2e038c174d2046f5333042
6
+ metadata.gz: cad6bf23e5f39a374115580a70469e382cfdb3578f0c553a04394b73632902484ba479755430eed9ad5af782a7ec633f57fe8d97ea5e75404a7609463221ebf9
7
+ data.tar.gz: 9f9bc6189f61911c6691b73ba0e6822fd586d339c859714d492db5b5587d3e2fcc2ea5e6c20ae0b18fff65f92872124999798c66200a84308f119e5ce7aeeab1
data/LICENSE.txt ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2025 LogicRelay LLC
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,100 @@
1
+ # RelayUI
2
+
3
+ > [!WARNING]
4
+ > This project is extremely new, and we're still working out the kinks. While it is just being used for the first time in new production apps, we're not yet ready to recommend it quite yet. It's highly likely that the API will change frequently in ways that will be extremely disruptive to your codebase. Use at your own risk for now, and we'll update this warning when we're ready to recommend it. Look for a 1.0.0 release sometime in Q2 2025.
5
+
6
+ RelayUI is a set of highly prescriptive UI elements written in Phlex and designed for Ruby and Rails applications. We know what you're thinking: "Another UI kit? Really?" But hear us out...
7
+
8
+ We generally see two types of UI kits available today:
9
+ - **Highly flexible and customizable UI kits** - These are great for building consumer-facing apps where you need to have a unique look and feel, but they can create a lot of overhead for developers who just want to get business value out the door quickly. Many of them also rely on code generation, and we wanted a solution that could be updated via bundler.
10
+ - **Paid UI kits** - These are great for businesses that want to get something out the door quickly and don't mind paying for it, but we like open-source software and we think that there's a place for a free, easy-to-use UI kit.
11
+
12
+ This is why RelayUI was created. It's purposefully inflexible, meant to provide a baseline set of functionality that covers 80-90% of use cases so developers can skip decisions like what icons to use or how to style buttons and focus on the stuff that matters.
13
+
14
+ RelayUI is supported by [LogicRelay](https://www.logicrelay.com), a software development agency that specializes in rapid creation of business software. We created RelayUI to help us build software faster and more consistently, and we're sharing it with the world in the hopes that it will help others do the same.
15
+
16
+ ## Why Phlex?
17
+
18
+ [Phlex](https://github.com/phlex-ruby/phlex) is a Ruby-based DSL for building HTML views and components. After experimenting with [ViewComponents](https://viewcomponent.org/) and using the gem in some projects, we found the differentiation between views and presenter logic to be cumbersome at scale. Phlex provides the power of ruby in your views without the need for a separate presenter layer, and it's fast. Really fast.
19
+
20
+ Our Phlex components [can be rendered in plain-old `erb` files](https://www.phlex.fun/components/rendering.html#rendering-phlex-components-in-erb), but we think you'll find that the Phlex DSL is so easy to use that you'll want to use it everywhere.
21
+
22
+ ## The RUI Namespace
23
+
24
+ All of RelayUI's components are housed in the `RUI::` namespace. This turns your IDE's autocomplete into a powerful tool for building UIs quickly. Just type `RUI::` and see what's available!
25
+
26
+ With this in mind, we prefer pulling basic variants up to the model level. Whereas many UI kits may specify variants via parameters (eg: `Component.new(variant: :primary)`), we prefer to give major variants class-level importance. So, we'll opt for patterns like `RUI::Buttons::Primary` and `RUI::Buttons::Secondary` instead.
27
+
28
+ ## Using TailwindCSS
29
+
30
+ RelayUI uses [TailwindCSS v4](https://tailwindcss.com/) for styling under the hood. One of the challenges we aimed to solve is how to include the styles Tailwind provides in a way that doesn't collide with any other CSS styles or frameworks being used. For example, we wanted to make sure RelayUI still worked well in projects that used Tailwind v3, or even Bootstrap.
31
+
32
+ For that reason, we've decided to prefix our CSS classes with `rui:`. This way, RelayUI is able to come out of the box with all of the styles you need to make our components work, but you can choose to use any CSS framework you want and not risk any CSS class conflicts or collisions.
33
+ STRING
34
+
35
+ # Installation
36
+
37
+ ### First, add the gem to your `Gemfile`:
38
+
39
+ ```ruby
40
+ gem 'relay_ui'
41
+ ```
42
+
43
+ ### Then install your gems:
44
+
45
+ ```shell
46
+ bundle install
47
+ ```
48
+
49
+ ### Or install it directly:
50
+
51
+ ```shell
52
+ gem install relay_ui
53
+ ```
54
+
55
+ ### Include the gem's stylesheet in your application layout:
56
+
57
+ ```ruby
58
+ stylesheet_link_tag "relay_ui/relay_ui", media: "all"
59
+ ```
60
+
61
+ That's it! All of the basic functionality of the UI kit is now available to you. For certain components that require additional elements (like stimulus controllers), you'll need to include those separately. They will be documented in the component's usage instructions.
62
+
63
+ # Usage
64
+
65
+ The usage docs can be found at https://www.relayui.com
66
+
67
+ # Contributing
68
+
69
+ **If you're interested in contributing on this project we'd love to chat with you on [our Discord server](https://discord.gg/wPyvK87KFe).**
70
+
71
+ There's a lot of work to be done on this project, and we're always looking for help. Here's some details on how we're going about building this kit:
72
+
73
+ - We have a [public roadmap](https://github.com/orgs/logicrelay/projects/14/views/3) that we're using to track features that are slated for the next release and features that haven't been prioritized yet.
74
+ - We're using [Github Projects](https://github.com/orgs/logicrelay/projects/14/views/1) to track WIP features and bugs.
75
+ - We'll generally spec out a milestone in a JIT fashion as the previous milestone is released.
76
+
77
+ ## Prerequisites
78
+
79
+ There are a couple of things you'll need to do to get your environment set up for contributing to this project:
80
+
81
+ - **mise** - We use [`mise`](https://mise.jdx.dev/getting-started.html) to manage our development environment and provide easy access to the tools we need to work on the project.
82
+ - **bundler config** - You'll need to set the local path to the root directory of the project by running the following command in the root of the repo: `bundle config local.relay_ui /path/to/relay_ui`. This will allow you to work on the UI kit locally and see changes in real time.
83
+ - **LiveReload** - [LiveReload](https://addons.mozilla.org/en-US/firefox/addon/livereload/) is not a must have, but it's a nice to have. It will automatically refresh your browser when you're working on the UI kit locally.
84
+
85
+ ## Tooling
86
+
87
+ There are three processes to run locally when developing...
88
+ - `mise run watch` starts the watch process that generates a combined CSS file using TailwindCSS.
89
+ - `mise run guard` runs `guard` and `guard-livereload` (optional)
90
+ - `mise run server` starts the local server.
91
+
92
+ ## License
93
+
94
+ The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
95
+
96
+ [Maintained by the team @ LogicRelay](https://www.logicrelay.com/)
97
+
98
+ ## Code of Conduct
99
+
100
+ Everyone interacting in the RelayUi project's codebases, issue trackers, chat rooms and mailing lists is expected to follow the [code of conduct](https://github.com/[USERNAME]/relay_ui/blob/main/CODE_OF_CONDUCT.md).
@@ -1,19 +1,16 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RelayUi
4
- class Engine < ::Rails::Engine
5
- isolate_namespace RelayUi
3
+ module RUI
4
+ if defined?(Rails)
5
+ class Engine < ::Rails::Engine
6
+ isolate_namespace RUI
6
7
 
7
- initializer "relay_ui.assets.precompile" do |app|
8
- app.config.assets.paths << root.join("vendor", "assets", "stylesheets")
9
- app.config.assets.paths << root.join("vendor", "assets", "javascripts")
10
- end
11
-
12
- initializer "relay_ui.autoload.components" do
13
- Rails.autoloaders.main.push_dir(
14
- "#{Gem::Specification.find_by_name('relay_ui').gem_dir}/app/components",
15
- namespace: RelayUi
16
- )
8
+ initializer "relay_ui.autoload.components" do
9
+ Rails.autoloaders.main.push_dir(
10
+ "#{Gem::Specification.find_by_name('relay_ui').gem_dir}/lib/rui",
11
+ namespace: RUI
12
+ )
13
+ end
17
14
  end
18
15
  end
19
- end
16
+ end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module RelayUi
4
- VERSION = "0.1.1"
3
+ module RUI
4
+ VERSION = "0.3.0"
5
5
  end
data/lib/relay_ui.rb CHANGED
@@ -3,7 +3,7 @@
3
3
  require_relative "relay_ui/version"
4
4
  require_relative "relay_ui/engine"
5
5
 
6
- module RelayUi
6
+ module RUI
7
7
  class Error < StandardError; end
8
8
  # Your code goes here...
9
9
  end
@@ -0,0 +1,9 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Badges::Base < RUI::Base
4
+ def view_template
5
+ div do
6
+ span(class: "whitespace-nowrap px-3 py-0.5 text-xs rounded-full text-center font-medium #{variant_classes}") { yield }
7
+ end
8
+ end
9
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Badges::Blue < RUI::Badges::Base
4
+ private
5
+
6
+ def variant_classes = "bg-blue-100 text-blue-700"
7
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Badges::Gray < RUI::Badges::Base
4
+ private
5
+
6
+ def variant_classes = "bg-gray-100 text-gray-700"
7
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Badges::Green < RUI::Badges::Base
4
+ private
5
+
6
+ def variant_classes = "bg-green-100 text-green-700"
7
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Badges::Purple < RUI::Badges::Base
4
+ private
5
+
6
+ def variant_classes = "bg-purple-100 text-purple-700"
7
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Badges::Red < RUI::Badges::Base
4
+ private
5
+
6
+ def variant_classes = "bg-red-100 text-red-700"
7
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Badges::Yellow < RUI::Badges::Base
4
+ private
5
+
6
+ def variant_classes = "bg-yellow-100 text-yellow-700"
7
+ end
data/lib/rui/badges.rb ADDED
@@ -0,0 +1,3 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RUI::Badges; end
data/lib/rui/base.rb ADDED
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Base < Phlex::HTML
4
+ def initialize(**attrs, &)
5
+ @attrs = attrs
6
+ super(&)
7
+ end
8
+ end
@@ -0,0 +1,29 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Buttons::Base < RUI::Base
4
+ def initialize(icon: nil, **attrs)
5
+ @icon = icon
6
+ @attrs = attrs
7
+ end
8
+
9
+ def view_template
10
+ button(class: classes, **@attrs) do
11
+ div(class: "flex flex-row items-center gap-2") do
12
+ if @icon
13
+ div(class: "size-4 my-1") do
14
+ render RUI::Icon.new(@icon)
15
+ end
16
+ end
17
+ span(class: "text-nowrap") { yield } if block_given?
18
+ end
19
+ end
20
+ end
21
+
22
+ private
23
+
24
+ def classes
25
+ "#{base_classes} #{variant_classes}"
26
+ end
27
+
28
+ def base_classes = "inline-block px-2 py-1 hover:cursor-pointer rounded-md transition duration-200 ease-in-out"
29
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Buttons::Destructive < RUI::Buttons::Base
4
+ private
5
+
6
+ def variant_classes = "bg-red-700 hover:bg-red-900 text-white border border-red-700 hover:border-red-900"
7
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Buttons::Ghost < RUI::Buttons::Base
4
+ private
5
+
6
+ def variant_classes = "hover:bg-zinc-100"
7
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Buttons::Outline < RUI::Buttons::Base
4
+ private
5
+
6
+ def variant_classes = "border rounded border-zinc-300 hover:border-zinc-700 text-zinc-700"
7
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Buttons::Primary < RUI::Buttons::Base
4
+ private
5
+
6
+ def variant_classes = "bg-blue-700 hover:bg-blue-900 text-white border border-blue-700 hover:border-blue-900"
7
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Buttons::Secondary < RUI::Buttons::Base
4
+ private
5
+
6
+ def variant_classes = "bg-zinc-700 hover:bg-zinc-900 text-white border border-zinc-700 hover:border-zinc-900"
7
+ end
@@ -0,0 +1,3 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RUI::Buttons; end
data/lib/rui/card.rb ADDED
@@ -0,0 +1,92 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Card < RUI::Base
4
+ def view_template(&)
5
+ div(role: "region", aria: { label: "Card" }, **@attrs) do
6
+ div(
7
+ class: "#{padding(capture(&))} flex flex-col gap-4 border border-zinc-300 rounded-lg overflow-hidden",
8
+ data: { component: "card" },
9
+ &
10
+ )
11
+ end
12
+ end
13
+
14
+ def header(&)
15
+ render Header.new(&)
16
+ end
17
+
18
+ def image(&)
19
+ img(src: capture(&))
20
+ end
21
+
22
+ def body(&)
23
+ div(class: "px-4", &)
24
+ end
25
+
26
+ def footer(&)
27
+ render Footer.new(&)
28
+ end
29
+
30
+ class Header < RUI::Base
31
+ def view_template(&)
32
+ header(class: "flex flex-row justify-between items-center gap-4 px-4", &)
33
+ end
34
+
35
+ def thumbnail(&)
36
+ img(src: capture(&), class: "inline-block rounded-full size-10")
37
+ end
38
+
39
+ def titles(&)
40
+ render Titles.new(&)
41
+ end
42
+
43
+ def actions(&)
44
+ div(class: "flex flex-row gap-2 items-center", &)
45
+ end
46
+
47
+ class Titles < RUI::Base
48
+ def view_template(&)
49
+ div(class: "grow shrink basis-0 flex flex-col", &)
50
+ end
51
+
52
+ def title(&)
53
+ h2(class: "text-xl font-semibold", &)
54
+ end
55
+
56
+ def subtitle(&)
57
+ h3(class: "text-lg", &)
58
+ end
59
+ end
60
+ end
61
+
62
+ class Footer < RUI::Base
63
+ def view_template(&)
64
+ footer(class: "flex flex-row justify-end items-center gap-4 px-4", &)
65
+ end
66
+
67
+ def actions(&)
68
+ render Actions.new(&)
69
+ end
70
+
71
+ class Actions < RUI::Base
72
+ def view_template(&)
73
+ div(class: "flex flex-row gap-2 items-center", &)
74
+ end
75
+
76
+ def action(href: "#", **kwargs, &)
77
+ a(href:, class: "text-blue-700 hover:underline uppercase text-sm font-semibold", **kwargs, &)
78
+ end
79
+ end
80
+ end
81
+
82
+ private
83
+
84
+ def padding(content)
85
+ return if content.empty?
86
+ padding = []
87
+ doc = Nokogiri::HTML::DocumentFragment.parse(content)
88
+ padding << "pt-4" if doc.children.first.name != "img"
89
+ padding << "pb-4" if doc.children.last.name != "img"
90
+ padding.join(" ")
91
+ end
92
+ end
data/lib/rui/flash.rb ADDED
@@ -0,0 +1,51 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Flash < RUI::Base
4
+ def initialize(variant = "alert")
5
+ @variant = variant
6
+ end
7
+
8
+ def view_template
9
+ div(role: "alert", data: { controller: "flash" }, class: "lg:max-w-lg border p-3 lg:px-5 lg:py-3 rounded-lg opacity-0 transition-opacity duration-100 ease-in-out " + classes) do
10
+ div(class: "flex flex-row items-center justify-between gap-3") do
11
+ div(class: "w-6") { render RUI::Icon.new(icon) }
12
+ p(class: "w-full") { yield }
13
+ button(class: "w-6 hover:cursor-pointer", data: { action: "flash#close" }) do
14
+ render RUI::Icon.new("x")
15
+ end
16
+ end
17
+ end
18
+ end
19
+
20
+ private
21
+
22
+ def classes = {
23
+ "notice" => "bg-blue-50 text-blue-900 border-blue-500",
24
+ "success" => "bg-green-50 text-green-900 border-green-500",
25
+ "alert" => "bg-yellow-50 text-yellow-900 border-yellow-500",
26
+ "failure" => "bg-red-50 text-red-900 border-red-500"
27
+ }.fetch @variant, "bg-zinc-50 text-zinc-900 border-zinc-500"
28
+
29
+ def icon = {
30
+ "notice" => "info-circle",
31
+ "success" => "circle-check",
32
+ "alert" => "alert-triangle",
33
+ "failure" => "flag"
34
+ }.fetch @variant, "alert-circle-filled"
35
+
36
+ class Wrapper < RUI::Base
37
+ def view_template(&)
38
+ div(class: "fixed left-0 lg:left-auto bottom-0 right-0 m-5 lg:m-10 flex flex-col gap-3", &)
39
+ end
40
+ end
41
+
42
+ module Helper
43
+ def success(message)
44
+ flash[:success] = message
45
+ end
46
+
47
+ def failure(message)
48
+ flash[:failure] = message
49
+ end
50
+ end
51
+ end
@@ -0,0 +1,25 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Forms::Checkbox < RUI::Base
4
+ def initialize(**kwargs)
5
+ @kwargs = kwargs
6
+ end
7
+
8
+ def view_template(&)
9
+ div(class: "flex flex-row gap-3 items-center") do
10
+ input(
11
+ type: "hidden",
12
+ value: "0",
13
+ autocomplete: "off",
14
+ **@kwargs
15
+ )
16
+ input(
17
+ type: "checkbox",
18
+ value: "1",
19
+ class: "appearance-none size-4 border border-zinc-300 rounded hover:border-zinc-500 hover:cursor-pointer checked:bg-blue-700",
20
+ **@kwargs
21
+ )
22
+ render RUI::Forms::Label.new(&) if block_given?
23
+ end
24
+ end
25
+ end
@@ -0,0 +1,19 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Forms::Email < RUI::Base
4
+ def initialize(**kwargs)
5
+ @kwargs = kwargs
6
+ end
7
+
8
+ def view_template(&)
9
+ render RUI::Forms::FieldGroup.new do
10
+ render RUI::Forms::Label.new(&) if block_given?
11
+ input(
12
+ type: "email",
13
+ class: "border border-zinc-300 hover:border-zinc-400 rounded px-2 py-1",
14
+ **@kwargs
15
+ )
16
+ end
17
+ div
18
+ end
19
+ end
@@ -0,0 +1,5 @@
1
+ class RUI::Forms::FieldGroup < RUI::Base
2
+ def view_template(&)
3
+ div(class: "flex flex-col gap-1", &)
4
+ end
5
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Forms::Label < RUI::Base
4
+ def view_template(&)
5
+ label(class: "text-sm font-semibold", &)
6
+ end
7
+ end
@@ -0,0 +1,18 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Forms::Password < RUI::Base
4
+ def initialize(**kwargs)
5
+ @kwargs = kwargs
6
+ end
7
+
8
+ def view_template(&)
9
+ render RUI::Forms::FieldGroup.new do
10
+ render RUI::Forms::Label.new(&) if block_given?
11
+ input(
12
+ type: "password",
13
+ class: "border border-zinc-300 hover:border-zinc-400 rounded px-2 py-1",
14
+ **@kwargs
15
+ )
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,18 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Forms::Phone < RUI::Base
4
+ def initialize(**kwargs)
5
+ @kwargs = kwargs
6
+ end
7
+
8
+ def view_template(&)
9
+ render RUI::Forms::FieldGroup.new do
10
+ render RUI::Forms::Label.new(&) if block_given?
11
+ input(
12
+ type: :tel,
13
+ class: "border border-zinc-300 hover:border-zinc-400 rounded px-2 py-1",
14
+ **@kwargs
15
+ )
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,22 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Forms::Radio < RUI::Base
4
+ def initialize(label:, &)
5
+ @label = label
6
+ super(&)
7
+ end
8
+
9
+ def view_template
10
+ render RUI::Forms::FieldGroup.new do
11
+ render RUI::Forms::Label.new { @label }
12
+ yield
13
+ end
14
+ end
15
+
16
+ def option(**attrs, &)
17
+ div(class: "flex flex-row gap-3 items-center") do
18
+ input(class: "hover:cursor-pointer", type: :radio, **attrs)
19
+ label(&)
20
+ end
21
+ end
22
+ end
@@ -0,0 +1,24 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Forms::Select < RUI::Base
4
+ def initialize(label:, **attrs, &)
5
+ @label = label
6
+ @attrs = attrs
7
+ super(&)
8
+ end
9
+
10
+ def view_template(&)
11
+ render RUI::Forms::FieldGroup.new do
12
+ render RUI::Forms::Label.new { @label }
13
+ select(
14
+ class: "border border-zinc-300 hover:border-zinc-400 rounded px-2 py-1.5 hover:cursor-pointer",
15
+ **@attrs,
16
+ &
17
+ )
18
+ end
19
+ end
20
+
21
+ def option(**attrs, &)
22
+ tag(:option, **attrs, &)
23
+ end
24
+ end
@@ -0,0 +1,17 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RUI::Forms::Text < RUI::Base
4
+ def initialize(**kwargs)
5
+ @kwargs = kwargs
6
+ end
7
+
8
+ def view_template(&)
9
+ render RUI::Forms::FieldGroup.new do
10
+ render RUI::Forms::Label.new(&) if block_given?
11
+ input(
12
+ class: "border border-zinc-300 hover:border-zinc-400 rounded px-2 py-1",
13
+ **@kwargs
14
+ )
15
+ end
16
+ end
17
+ end