Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
14 changes: 14 additions & 0 deletions docs/app/views/docs/tabs.rb
Original file line number Diff line number Diff line change
Expand Up @@ -102,6 +102,20 @@ def view_template
RUBY
end

render Docs::VisualCodeExample.new(title: "As navigation links", context: self) do
<<~RUBY
# Render triggers as `<a>` instead of `<button>` for tabs that navigate
# (e.g. server-rendered tabs driven by a query param). This avoids
# nesting interactive elements and keeps the markup valid and accessible.
Tabs(default_value: "account", class: 'w-96') do
TabsList do
TabsTrigger(value: "account", as: :a, href: "?tab=account") { "Account" }
TabsTrigger(value: "password", as: :a, href: "?tab=password") { "Password" }
end
end
RUBY
end

render Docs::VisualCodeExample.new(title: "Change default value", context: self) do
<<~RUBY
Tabs(default: "password", class: 'w-96') do
Expand Down
14 changes: 10 additions & 4 deletions gem/lib/ruby_ui/tabs/tabs_trigger.rb
Original file line number Diff line number Diff line change
Expand Up @@ -2,20 +2,24 @@

module RubyUI
class TabsTrigger < Base
def initialize(value:, **attrs)
def initialize(value:, as: :button, **attrs)
@value = value
@as = as
super(**attrs)
end

def view_template(&)
button(**attrs, &)
if @as == :a
a(**attrs, &)
else
button(**attrs, &)
end
end

private

def default_attrs
{
type: :button,
base = {
data: {
ruby_ui__tabs_target: "trigger",
action: "click->ruby-ui--tabs#show",
Expand All @@ -29,6 +33,8 @@ def default_attrs
"focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2"
]
}
base[:type] = :button unless @as == :a
base
end
end
end
38 changes: 38 additions & 0 deletions gem/test/ruby_ui/tabs_trigger_test.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
# frozen_string_literal: true

require "test_helper"

class RubyUI::TabsTriggerTest < ComponentTest
def test_renders_as_button_by_default
output = phlex { RubyUI.TabsTrigger(value: "account") { "Account" } }

assert_match(/\A<button/, output)
assert_match(/type="button"/, output)
assert_match(/Account/, output)
refute_match(/<a[\s>]/, output)
end

def test_renders_as_anchor_with_href_when_as_a
output = phlex { RubyUI.TabsTrigger(value: "account", as: :a, href: "/account") { "Account" } }

assert_match(/\A<a/, output)
assert_match(/href="\/account"/, output)
assert_match(/Account/, output)
refute_match(/<button/, output)
refute_match(/type="button"/, output)
end

def test_unexpected_as_value_still_renders_a_typed_button
output = phlex { RubyUI.TabsTrigger(value: "account", as: :buton) { "Account" } }

assert_match(/\A<button/, output)
assert_match(/type="button"/, output)
end

def test_keeps_trigger_data_attributes_when_rendered_as_anchor
output = phlex { RubyUI.TabsTrigger(value: "account", as: :a, href: "/account") { "Account" } }

assert_match(/data-ruby-ui--tabs-target="trigger"/, output)
assert_match(/data-value="account"/, output)
end
end
2 changes: 1 addition & 1 deletion mcp/data/registry.json
Original file line number Diff line number Diff line change
Expand Up @@ -2301,7 +2301,7 @@
},
{
"path": "tabs_trigger.rb",
"content": "# frozen_string_literal: true\n\nmodule RubyUI\n class TabsTrigger < Base\n def initialize(value:, **attrs)\n @value = value\n super(**attrs)\n end\n\n def view_template(&)\n button(**attrs, &)\n end\n\n private\n\n def default_attrs\n {\n type: :button,\n data: {\n ruby_ui__tabs_target: \"trigger\",\n action: \"click->ruby-ui--tabs#show\",\n value: @value\n },\n class: [\n \"inline-flex items-center justify-center whitespace-nowrap rounded-md px-3 py-1 text-sm font-medium ring-offset-background transition-all\",\n \"disabled:pointer-events-none disabled:opacity-50\",\n \"aria-disabled:pointer-events-none aria-disabled:opacity-50 aria-disabled:cursor-not-allowed\",\n \"data-[state=active]:bg-background data-[state=active]:text-foreground data-[state=active]:shadow\",\n \"focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2\"\n ]\n }\n end\n end\nend\n"
"content": "# frozen_string_literal: true\n\nmodule RubyUI\n class TabsTrigger < Base\n def initialize(value:, as: :button, **attrs)\n @value = value\n @as = as\n super(**attrs)\n end\n\n def view_template(&)\n if @as == :a\n a(**attrs, &)\n else\n button(**attrs, &)\n end\n end\n\n private\n\n def default_attrs\n base = {\n data: {\n ruby_ui__tabs_target: \"trigger\",\n action: \"click->ruby-ui--tabs#show\",\n value: @value\n },\n class: [\n \"inline-flex items-center justify-center whitespace-nowrap rounded-md px-3 py-1 text-sm font-medium ring-offset-background transition-all\",\n \"disabled:pointer-events-none disabled:opacity-50\",\n \"aria-disabled:pointer-events-none aria-disabled:opacity-50 aria-disabled:cursor-not-allowed\",\n \"data-[state=active]:bg-background data-[state=active]:text-foreground data-[state=active]:shadow\",\n \"focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2\"\n ]\n }\n base[:type] = :button unless @as == :a\n base\n end\n end\nend\n"
}
],
"dependencies": {
Expand Down
Loading