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.
- checksums.yaml +4 -4
- data/LICENSE.txt +21 -0
- data/README.md +100 -0
- data/lib/relay_ui/engine.rb +11 -14
- data/lib/relay_ui/version.rb +2 -2
- data/lib/relay_ui.rb +1 -1
- data/lib/rui/badges/base.rb +9 -0
- data/lib/rui/badges/blue.rb +7 -0
- data/lib/rui/badges/gray.rb +7 -0
- data/lib/rui/badges/green.rb +7 -0
- data/lib/rui/badges/purple.rb +7 -0
- data/lib/rui/badges/red.rb +7 -0
- data/lib/rui/badges/yellow.rb +7 -0
- data/lib/rui/badges.rb +3 -0
- data/lib/rui/base.rb +8 -0
- data/lib/rui/buttons/base.rb +29 -0
- data/lib/rui/buttons/destructive.rb +7 -0
- data/lib/rui/buttons/ghost.rb +7 -0
- data/lib/rui/buttons/outline.rb +7 -0
- data/lib/rui/buttons/primary.rb +7 -0
- data/lib/rui/buttons/secondary.rb +7 -0
- data/lib/rui/buttons.rb +3 -0
- data/lib/rui/card.rb +92 -0
- data/lib/rui/flash.rb +51 -0
- data/lib/rui/forms/checkbox.rb +25 -0
- data/lib/rui/forms/email.rb +19 -0
- data/lib/rui/forms/field_group.rb +5 -0
- data/lib/rui/forms/label.rb +7 -0
- data/lib/rui/forms/password.rb +18 -0
- data/lib/rui/forms/phone.rb +18 -0
- data/lib/rui/forms/radio.rb +22 -0
- data/lib/rui/forms/select.rb +24 -0
- data/lib/rui/forms/text.rb +17 -0
- data/lib/rui/forms/textarea.rb +18 -0
- data/lib/rui/forms.rb +3 -0
- data/lib/rui/helpers.rb +3 -0
- data/lib/rui/icon.rb +13 -0
- data/lib/rui/layout/body.rb +7 -0
- data/lib/rui/layout/main.rb +5 -0
- data/lib/rui/layout/page.rb +5 -0
- data/lib/rui/layout.rb +3 -0
- data/lib/rui/links/base.rb +30 -0
- data/lib/rui/links/destructive.rb +7 -0
- data/lib/rui/links/ghost.rb +7 -0
- data/lib/rui/links/outline.rb +7 -0
- data/lib/rui/links/primary.rb +7 -0
- data/lib/rui/links/secondary.rb +7 -0
- data/lib/rui/links.rb +3 -0
- data/lib/rui/markdown/generator.rb +36 -0
- data/lib/rui/markdown/safe.rb +9 -0
- data/lib/rui/markdown/safe_renderer.rb +15 -0
- data/lib/rui/markdown/unsafe.rb +9 -0
- data/lib/rui/markdown/unsafe_renderer.rb +42 -0
- data/lib/rui/markdown.rb +3 -0
- data/lib/rui/navigation/menu_button.rb +7 -0
- data/lib/rui/navigation/sidebar.rb +43 -0
- data/lib/rui/navigation/top.rb +19 -0
- data/lib/rui/navigation.rb +3 -0
- data/{app/components → lib/rui}/slideout.rb +9 -9
- data/lib/rui/table.rb +54 -0
- data/lib/rui/text.rb +21 -0
- metadata +147 -37
- data/Rakefile +0 -64
- data/app/assets/javascripts/relay_ui/index.js +0 -5
- data/app/assets/stylesheets/relay_ui/application.css +0 -1
- data/app/components/badges/base.rb +0 -5
- data/app/components/badges/blue.rb +0 -5
- data/app/components/badges/gray.rb +0 -5
- data/app/components/badges/green.rb +0 -5
- data/app/components/badges/purple.rb +0 -5
- data/app/components/badges/red.rb +0 -5
- data/app/components/badges/yellow.rb +0 -5
- data/app/components/base.rb +0 -15
- data/app/components/buttons/base.rb +0 -30
- data/app/components/buttons/destructive.rb +0 -5
- data/app/components/buttons/ghost.rb +0 -5
- data/app/components/buttons/link.rb +0 -9
- data/app/components/buttons/outline.rb +0 -5
- data/app/components/buttons/primary.rb +0 -5
- data/app/components/buttons/secondary.rb +0 -5
- data/app/components/code_block.rb +0 -13
- data/app/components/headings.rb +0 -37
- data/app/components/icon.rb +0 -9
- data/app/components/lists/base.rb +0 -13
- data/app/components/lists/ordered.rb +0 -5
- data/app/components/lists/unordered.rb +0 -5
- data/app/components/navigation.rb +0 -49
- data/app/components/text.rb +0 -19
- data/package.json +0 -16
- data/tailwind.config.js +0 -10
- data/vendor/assets/javascripts/relay_ui/dist/relay_ui.js +0 -2471
- 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:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 7158b2530047c016d6353aba1125ad761104ae2f553656c3d28c40cf6fc74f4a
|
4
|
+
data.tar.gz: 917a96410664ca64d92e4f41c86d7af74ebcbdc27511dbcf167fc74bd45bec26
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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).
|
data/lib/relay_ui/engine.rb
CHANGED
@@ -1,19 +1,16 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
4
|
-
|
5
|
-
|
3
|
+
module RUI
|
4
|
+
if defined?(Rails)
|
5
|
+
class Engine < ::Rails::Engine
|
6
|
+
isolate_namespace RUI
|
6
7
|
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
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
|
data/lib/relay_ui/version.rb
CHANGED
data/lib/relay_ui.rb
CHANGED
data/lib/rui/badges.rb
ADDED
data/lib/rui/base.rb
ADDED
@@ -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
|
data/lib/rui/buttons.rb
ADDED
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,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
|