View Source PyroComponents.Components.Core (PyroComponents v0.1.1)
A core set of functional .heex
components for building web apps. It is similar to (and often API-compatible with) Phoenix's generated core_components.ex
.
Compared to Phoenix's generated components, Pyro's implementation adds:
- Maintenance/bugfixes/new features, since it's a library
- A powerful override system for customization
- A special
:css_classes
type that utilizes the configured CSS merge utility - The button component implements both button and anchor tags (button-styled links!)
- Inputs
autofocus
prop to enable a hook for reliable focus on mounthidden
input type with a slot for custom content
- A rich flash experience
- Auto-remove after (configurable) timeout
- Progress bar for auto-removed flash messages
- Define which flashes are included in which trays (supports multiple trays)
- Slightly cleaner, more semantic markup
- Extra components
There are more complex components outside the Core
module, be sure to check those out as well.
Overridable Component Attributes
You can customize the components in this module by configuring overrides.
The components in this module support the following overridable attributes:
:icon_name
:string
(required):class
:css_classes
:icon_class
:css_classes
:copy
:boolean
(required):copy_label
:string
(required):copy_message
:string
:class
:css_classes
:copy_class
:css_classes
:clear_on_escape
:boolean
(required):get_tz_options
:any
(required):class
:css_classes
:input_class
:css_classes
:input_check_label_class
:css_classes
:input_datetime_zoned_wrapper_class
:css_classes
:description_class
:css_classes
:class
:css_classes
:class
:css_classes
:dd_class
:css_classes
:dt_class
:css_classes
:class
:css_classes
:action_class
:css_classes
:action_td_class
:css_classes
:action_wrapper_class
:css_classes
:tbody_class
:css_classes
:td_class
:css_classes
:th_action_class
:css_classes
:th_label_class
:css_classes
:thead_class
:css_classes
:tr_class
:css_classes
:color
:string
(required):size
:string
(required):class
:css_classes
:class
:css_classes
:title_class
:css_classes
:subtitle_class
:css_classes
:actions_class
:css_classes
:replace
:boolean
(required):class
:css_classes
:color
:string
(required):size
:string
(required):variant
:string
(required):class
:css_classes
:icon_class
:css_classes
:ping_class
:css_classes
:ping_animation_class
:css_classes
:scheme
:atom
(required):label_system
:string
:label_dark
:string
:label_light
:string
:icon_system
:string
:icon_dark
:string
:icon_light
:string
:class
:css_classes
:ttl
:integer
(required):color
:string
(required):shape
:string
(required):size
:string
(required):variant
:string
(required):message
:string
(required):class
:css_classes
:icon_class
:css_classes
:autoshow
:boolean
(required):close
:boolean
(required):close_icon_name
:string
(required):icon_name
:string
(required):hide_js
:any
(required):show_js
:any
(required):title
:string
:ttl
:integer
(required):kind
:string
(required):style_for_kind
:string
:class
:css_classes
:control_class
:css_classes
:close_button_class
:css_classes
:close_icon_class
:css_classes
:message_class
:css_classes
:progress_class
:css_classes
:title_class
:css_classes
:title_icon_class
:css_classes
:include_kinds
:list
(required):class
:css_classes
:class
:css_classes
:show_js
:any
(required):hide_js
:any
(required):class
:css_classes
:class
:css_classes
:class
:css_classes
:actions_class
:css_classes
:close_icon_name
:string
(required):origin
:string
(required):max_width
:string
(required):hide_js
:any
(required):class
:css_classes
:overlay_class
:css_classes
:wrapper_class
:css_classes
:header_class
:css_classes
:header_inner_class
:css_classes
:header_title_class
:css_classes
:header_close_button_class
:css_classes
:content_class
:css_classes
:close_icon_class
:css_classes
:size
:string
(required):class
:css_classes
:vertical_offset
:string
(required):horizontal_offset
:string
(required):icon_name
:string
(required):class
:css_classes
:tooltip_class
:css_classes
:tooltip_text_class
:css_classes
:icon_class
:css_classes
Summary
Components
Renders a link. This basically wraps Phoenix.Component.link/1
with some overridable attributes, in particular class
for consistent, DRY link default styling.
Renders a button.
Renders a code block.
A color scheme switch component.
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.id
(:string
) (required)value
(:string
) (required) - text to copy.label
(:string
) - button label, defaults to value. Defaults tonil
.disabled
(:boolean
) - Defaults tofalse
.icon_name
(:string
) - the name of the icon to display (nil for none); seeicon/1
for details. Defaults tonil
.ttl
(:integer
) - how long to show the flash message after copying (overridable, required).color
(:string
) - the color of the button (overridable, required).shape
(:string
) - shape of the button (overridable, required).size
(:string
) - the size of the button (overridable, required).variant
(:string
) - style of button (overridable, required).message
(:string
) - message to display after copying (overridable, required).class
(:any
) - (overridable,:css_classes
).icon_class
(:any
) - (overridable,:css_classes
).- Global attributes are accepted.
Generates a generic error message.
Renders flash notices.
Shows the flash group with titles and content.
Renders a header with title and optional subtitle/actions.
Renders an icon.
Renders an input with label and error messages.
Renders a label.
Renders a description list.
Renders a modal.
Renders a navigation link, taking into account whether the URI is the current page.
A progress element. Styling the progress element is tricky, so this wraps it with some nice conveniences.
Renders a simple form.
A slide-over component.
A simple spinner component.
Renders a simple table with generic styling.
A tooltip component.
Functions
Javascript to manage switching color schemes with color_scheme_switcher/1
.
Components
Renders a link. This basically wraps Phoenix.Component.link/1
with some overridable attributes, in particular class
for consistent, DRY link default styling.
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.navigate
(:string
) - Navigates from a LiveView to a new LiveView. The browser page is kept, but a new LiveView process is mounted and its content on the page is reloaded. It is only possible to navigate between LiveViews declared under the same routerPhoenix.LiveView.Router.live_session/3
. Otherwise, a full browser redirect is used.Defaults to
nil
.patch
(:string
) - Patches the current LiveView. Thehandle_params
callback of the current LiveView will be invoked and the minimum content will be sent over the wire, as any other LiveView diff.Defaults to
nil
.href
(:any
) - Uses traditional browser navigation to the new location. This means the whole page is reloaded on the browser.Defaults to
nil
.method
(:string
) - The HTTP method to use with the link. This is intended for usage outside of LiveView and therefore only works with thehref={...}
attribute. It has no effect onpatch
andnavigate
instructions. In case the method is notget
, the link is generated inside the form which sets the proper information. In order to submit the form, JavaScript must be enabled in the browser.Defaults to
"get"
.csrf_token
(:any
) - a boolean or custom token to use for links with an HTTP method other thanget
. Defaults totrue
.replace
(:boolean
) - when using:patch
or:navigate
, should the browser's history be replaced withpushState
? (overridable, required).class
(:any
) - merge/override default classes of thecode
element (overridable,:css_classes
).Global attributes are accepted. additional HTML attributes added to the
a
tag.
Slots
inner_block
(required) - the content rendered inside of thea
tag.
Renders a button.
Supports:
- Any button type
- Any anchor type
- LivePatch
- LiveRedirect
- External href links
Examples
<.button>Send!</.button>
<.button phx-click="go" class="ml-2">Send!</.button>
<.button navigate={~p"/home"}>Home</.button>
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.confirm
(:string
) - text to display in a confirm dialog before emitting click event. Defaults tonil
.csrf_token
(:string
) - Defaults tonil
.disabled
(:boolean
) - Defaults tofalse
.href
(:any
)icon_name
(:string
) - the name of the icon to display (nil for none); seeicon/1
for details. Defaults tonil
.loading
(:boolean
) - display a loading spinner. Defaults tofalse
.method
(:string
) - Defaults to"get"
.navigate
(:string
)patch
(:string
)ping
(:boolean
) - show a ping indicator. Defaults tofalse
.replace
(:boolean
) - Defaults tofalse
.type
(:string
) - type of the button. Defaults to"button"
.color
(:string
) - the color of the button (overridable, required).size
(:string
) - the size of the button (overridable, required).variant
(:string
) - style of button (overridable, required).class
(:any
) - (overridable,:css_classes
).icon_class
(:any
) - (overridable,:css_classes
).ping_class
(:any
) - (overridable,:css_classes
).ping_animation_class
(:any
) - (overridable,:css_classes
).- Global attributes are accepted. Supports all globals plus:
["download", "hreflang", "referrerpolicy", "rel", "target", "form", "name", "value"]
.
Slots
inner_block
(required) - the content of the button.
Renders a code block.
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.source
(:string
) (required) - the code snippet.id
(:string
) (required)copy
(:boolean
) - (overridable, required).copy_label
(:string
) - (overridable, required).copy_message
(:string
) - (overridable).language
(:string
) - language of the code snippet. Defaults to"elixir"
.class
(:any
) - merge/override default classes of thecode
element (overridable,:css_classes
).copy_class
(:any
) - (overridable,:css_classes
).
A color scheme switch component.
- Toggle through light, dark, and system color schemes
- Optionally show labels for each scheme
- Optionally override the default labels for each scheme with
label_system
,label_dark
, andlabel_light
- Optionally override the default icons for each scheme with
icon_system
,icon_dark
, andicon_light
Note:
This requires several things to work:
darkMode: 'class'
in your Tailwind configcolor_scheme_switcher_js/1
added to the page's<head>
beforeapp.js
- Pyro's
PyroColorSchemeHook
hook added to your hooks inapp.js
Examples
<.color_scheme_switcher />
<.color_scheme_switcher scheme="light" />
<.color_scheme_switcher scheme="dark" />
<.color_scheme_switcher scheme="system" />
<.color_scheme_switcher show_labels />
<.color_scheme_switcher label_system="System" label_dark="Dunkel" label_light="Hell" />
<.color_scheme_switcher icon_system="hero-computer-desktop-solid" icon_dark="hero-moon-solid" icon_light="hero-sun-solid" />
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.scheme
(:atom
) - the scheme used to initialize the color scheme switcher (overridable, required).label_system
(:string
) - the label for the system scheme (overridable).label_dark
(:string
) - the label for the dark scheme (overridable).label_light
(:string
) - the label for the light scheme (overridable).icon_system
(:string
) - the icon for the system scheme (overridable).icon_dark
(:string
) - the icon for the dark scheme (overridable).icon_light
(:string
) - the icon for the light scheme (overridable).show_labels
(:boolean
) - show or hide labels. Defaults tofalse
.class
(:any
) - (overridable,:css_classes
).
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.id
(:string
) (required)value
(:string
) (required) - text to copy.label
(:string
) - button label, defaults to value. Defaults tonil
.disabled
(:boolean
) - Defaults tofalse
.icon_name
(:string
) - the name of the icon to display (nil for none); seeicon/1
for details. Defaults tonil
.ttl
(:integer
) - how long to show the flash message after copying (overridable, required).color
(:string
) - the color of the button (overridable, required).shape
(:string
) - shape of the button (overridable, required).size
(:string
) - the size of the button (overridable, required).variant
(:string
) - style of button (overridable, required).message
(:string
) - message to display after copying (overridable, required).class
(:any
) - (overridable,:css_classes
).icon_class
(:any
) - (overridable,:css_classes
).- Global attributes are accepted.
Generates a generic error message.
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.icon_name
(:string
) - the name of the icon; seeicon/1
for details (overridable, required).class
(:any
) - (overridable,:css_classes
).icon_class
(:any
) - (overridable,:css_classes
).
Slots
inner_block
(required)
Renders flash notices.
Examples
<.flash kind={:info} flash={@flash} />
<.flash kind={:info} phx-mounted={show("#flash")}>Welcome Back!</.flash>
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.flash
(:map
) - the map of flash messages to display. Defaults to%{}
.autoshow
(:boolean
) - whether to auto show the flash on mount (overridable, required).close
(:boolean
) - whether the flash can be closed (overridable, required).close_icon_name
(:string
) - (overridable, required).icon_name
(:string
) - (overridable, required).hide_js
(:any
) - (overridable, required).show_js
(:any
) - (overridable, required).title
(:string
) - (overridable).ttl
(:integer
) - (overridable, required).kind
(:string
) - used for styling and flash lookup (overridable, required).style_for_kind
(:string
) - used for styling a flash with a different kind (overridable).class
(:any
) - (overridable,:css_classes
).control_class
(:any
) - (overridable,:css_classes
).close_button_class
(:any
) - (overridable,:css_classes
).close_icon_class
(:any
) - (overridable,:css_classes
).message_class
(:any
) - (overridable,:css_classes
).progress_class
(:any
) - (overridable,:css_classes
).title_class
(:any
) - (overridable,:css_classes
).title_icon_class
(:any
) - (overridable,:css_classes
).- Global attributes are accepted. the arbitrary HTML attributes to add to the flash container.
Slots
inner_block
- the optional inner block to render the flash message.
Shows the flash group with titles and content.
Examples
<.flash_group flash={@flash} />
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.flash
(:map
) (required) - the map of flash messages to display.include_kinds
(:list
) - the kinds of flashes to display (overridable, required).class
(:any
) - (overridable,:css_classes
).- Global attributes are accepted. the arbitrary HTML attributes to add to the flash tray.
Renders a header with title and optional subtitle/actions.
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.class
(:any
) - (overridable,:css_classes
).title_class
(:any
) - (overridable,:css_classes
).subtitle_class
(:any
) - (overridable,:css_classes
).actions_class
(:any
) - (overridable,:css_classes
).
Slots
inner_block
(required)subtitle
actions
Renders an icon.
Tip
See the Heroicons website to preview/search the available icons.
Additionally, there are long-term plans to add more icon libraries.
Examples
<.icon name="hero-x-mark-solid" />
<.icon name="hero-arrow-path" class="ml-1 w-3 h-3 animate-spin" />
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.name
(:string
) (required) - the icon name.class
(:any
) - (overridable,:css_classes
).- Global attributes are accepted. the arbitrary HTML attributes for the svg container. Supports all globals plus:
["fill", "stroke", "stroke-width"]
.
Renders an input with label and error messages.
A Phoenix.HTML.FormField
may be passed as argument,
which is used to retrieve the input name, id, and values.
Otherwise all attributes may be passed explicitly.
Types
This function accepts all HTML input types, considering that:
You may also set
type="select"
to render a<select>
tagtype="checkbox"
is used exclusively to render boolean valuesFor live file uploads, see
Phoenix.Component.live_file_input/1
See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input for more information.
Examples
<.input field={@form[:email]} type="email" />
<.input name="my-input" errors={["oh no!"]} />
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.autofocus
(:boolean
) - enable autofocus hook to reliably focus input on mount. Defaults tofalse
.checked
(:boolean
) - the checked flag for checkbox inputs.description
(:string
) - Defaults tonil
.errors
(:list
) - Defaults to[]
.field
(Phoenix.HTML.FormField
) - a form field struct retrieved from the form, for example: @form[:email].id
(:any
) - Defaults tonil
.label
(:string
) - Defaults tonil
.multiple
(:boolean
) - the multiple flag for select inputs. Defaults tofalse
.name
(:any
)options
(:list
) - the options to pass to Phoenix.HTML.Form.options_for_select/2.prompt
(:string
) - the prompt for select inputs. Defaults tonil
.type
(:string
) - Defaults to"text"
.value
(:any
)clear_on_escape
(:boolean
) - clear input value on pressing Escape (overridable, required).tz
(:string
) - timezone. Defaults to"Etc/UTC"
.get_tz_options
(:any
) - a 0 arity function that returns a list of valid timezones (overridable, required).tz_options
(:list
) - Defaults tonil
.class
(:any
) - class of the field container element (overridable,:css_classes
).input_class
(:any
) - class of the input element (overridable,:css_classes
).input_check_label_class
(:any
) - class of the label element for a check input (overridable,:css_classes
).input_datetime_zoned_wrapper_class
(:any
) - class of the input wrapper element for a datetime zoned input (overridable,:css_classes
).description_class
(:any
) - class of the field description (overridable,:css_classes
).- Global attributes are accepted. Supports all globals plus:
["accept", "autocomplete", "capture", "cols", "disabled", "form", "list", "max", "maxlength", "min", "minlength", "pattern", "placeholder", "readonly", "required", "rows", "size", "step"]
.
Slots
inner_block
Renders a label.
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.for
(:string
) - Defaults tonil
.class
(:any
) - (overridable,:css_classes
).
Slots
inner_block
(required)
Renders a description list.
Examples
<.list>
<:item title="Title"><%= @post.title %></:item>
<:item title="Views"><%= @post.views %></:item>
</.list>
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.class
(:any
) - (overridable,:css_classes
).dd_class
(:any
) - (overridable,:css_classes
).dt_class
(:any
) - (overridable,:css_classes
).
Slots
item
(required) - Accepts attributes:title
(:string
) (required)
Renders a modal.
Examples
<.modal id="confirm-modal">
Are you sure?
<:confirm>OK</:confirm>
<:cancel>Cancel</:cancel>
</.modal>
JS commands may be passed to the :on_cancel
and on_confirm
attributes
for the caller to react to each button press, for example:
<.modal id="confirm" on_confirm={JS.push("delete")} on_cancel={JS.navigate(~p"/posts")}>
Are you sure you?
<:confirm>OK</:confirm>
<:cancel>Cancel</:cancel>
</.modal>
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.id
(:string
) (required)show
(:boolean
) - Defaults tofalse
.on_cancel
(Phoenix.LiveView.JS
) - Defaults to%Phoenix.LiveView.JS{ops: []}
.on_confirm
(Phoenix.LiveView.JS
) - Defaults to%Phoenix.LiveView.JS{ops: []}
.show_js
(:any
) - (overridable, required).hide_js
(:any
) - (overridable, required).class
(:any
) - (overridable,:css_classes
).
Slots
inner_block
(required)title
subtitle
confirm
cancel
A progress element. Styling the progress element is tricky, so this wraps it with some nice conveniences.
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.max
(:integer
) - Defaults to100
.value
(:integer
) - Defaults to0
.color
(:string
) - the color of the progress bar (overridable, required).size
(:string
) - the size of the progress bar (overridable, required).class
(:any
) - the class of the progress bar (overridable,:css_classes
).- Global attributes are accepted.
Renders a simple form.
Examples
<.simple_form for={@form} phx-change="validate" phx-submit="save">
<.input field={@form[:email]} label="Email"/>
<.input field={@form[:username]} label="Username" />
<:actions>
<.button>Save</.button>
</:actions>
</.simple_form>
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.for
(:any
) (required) - the datastructure for the form.as
(:any
) - the server side parameter to collect all input under. Defaults tonil
.class
(:any
) - (overridable,:css_classes
).actions_class
(:any
) - (overridable,:css_classes
).- Global attributes are accepted. the arbitrary HTML attributes to apply to the form tag. Supports all globals plus:
["autocomplete", "name", "rel", "action", "enctype", "method", "novalidate", "target", "multipart"]
.
Slots
inner_block
(required)actions
- the slot for form actions, such as a submit button.
A slide-over component.
Example
<%= if @slide_over do %>
<.slide_over origin={@slide_over} max_width="sm" title="Slide Over">
<p>
This is a slide over.
</p>
</.slide_over>
<% end %>
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.close_icon_name
(:string
) - (overridable, required).title
(:string
) - the title of the slide-over.id
(:string
) - Defaults to"slide-over"
.origin
(:string
) - (overridable, required).max_width
(:string
) - (overridable, required).hide_js
(:any
) - (overridable, required).close_even_name
(:string
) - the name of the event to send when the slide-over is closed. Defaults to"close_slide_over"
.close_slide_over_target
(:string
) - the specific live component to target for the close event, e.g. close_slide_over_target={@myself}. Defaults tonil
.class
(:any
) - (overridable,:css_classes
).overlay_class
(:any
) - (overridable,:css_classes
).wrapper_class
(:any
) - (overridable,:css_classes
).header_class
(:any
) - (overridable,:css_classes
).header_inner_class
(:any
) - (overridable,:css_classes
).header_title_class
(:any
) - (overridable,:css_classes
).header_close_button_class
(:any
) - (overridable,:css_classes
).content_class
(:any
) - (overridable,:css_classes
).close_icon_class
(:any
) - (overridable,:css_classes
).- Global attributes are accepted.
Slots
inner_block
(required)
A simple spinner component.
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.show
(:boolean
) - show or hide spinner. Defaults totrue
.size
(:string
) - the size of the spinner (overridable, required).class
(:any
) - (overridable,:css_classes
).- Global attributes are accepted.
Renders a simple table with generic styling.
Examples
<.table id="users" rows={@users}>
<:col :let={user} label="id"><%= user.id %></:col>
<:col :let={user} label="username"><%= user.username %></:col>
</.table>
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.id
(:string
)row_click
(:any
) - the function for handling phx-click on each row. Defaults tonil
.row_id
(:any
) - the function for generating the row id. Defaults tonil
.row_item
(:any
) - the function for mapping each row before calling the :col and :action slots. Defaults to&Function.identity/1
.rows
(:list
) (required)class
(:any
) - (overridable,:css_classes
).action_class
(:any
) - (overridable,:css_classes
).action_td_class
(:any
) - (overridable,:css_classes
).action_wrapper_class
(:any
) - (overridable,:css_classes
).tbody_class
(:any
) - (overridable,:css_classes
).td_class
(:any
) - (overridable,:css_classes
).th_action_class
(:any
) - (overridable,:css_classes
).th_label_class
(:any
) - (overridable,:css_classes
).thead_class
(:any
) - (overridable,:css_classes
).tr_class
(:any
) - (overridable,:css_classes
).
Slots
col
(required) - Accepts attributes:label
(:string
)
action
- the slot for showing user actions in the last table column.
A tooltip component.
- JS hook that "nudges" tooltip into view
- Simple props for tooltip text and custom icon
- Optional slots for icon and/or tooltip content
Examples
<.tooltip id="tooltip-1" tooltip="A default tooltip!" />
<.tooltip id="tooltip-2"
icon_name="hero-light-bulb-solid"
tooltip="Custom icon." />
<.tooltip id="tooltip-3">
<:icon>?</:icon>
<div class="bg-red-500 text-white p-4 w-48 shadow-lg rounded">
Custom tooltip slot and custom icon slot.
</div>
</.tooltip>
Attributes
overrides
(:list
) - Manually set the overrides for this component (instead of config/default). Defaults tonil
.id
(:string
) (required)tooltip
(:string
) - Defaults tonil
.vertical_offset
(:string
) - (overridable, required).horizontal_offset
(:string
) - (overridable, required).icon_name
(:string
) - the name of the icon; seeicon/1
for details (overridable, required).class
(:any
) - (overridable,:css_classes
).tooltip_class
(:any
) - (overridable,:css_classes
).tooltip_text_class
(:any
) - (overridable,:css_classes
).icon_class
(:any
) - (overridable,:css_classes
).
Slots
icon
inner_block
Functions
Javascript to manage switching color schemes with color_scheme_switcher/1
.
Note:
This needs to be added in the page's head before
app.js
to prevent FOUC:
<head>
<.color_scheme_switcher_js />
<script defer phx-track-static type="text/javascript" src={~p"/assets/app.js"}>
</head>