localspot/lib/localspot_web/components/layouts.ex
Kevin Sivic 821af5b24f Update site branding and navigation for LocalSpot
- Replace Phoenix default navigation with LocalSpot nav
- Add links to Businesses, Categories, Map, and Add Business
- Update home page with hero, feature cards, and CTA
- Update page title suffix to LocalSpot
- Change pre-commit hook to auto-format instead of check

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-12-01 00:15:09 -05:00

169 lines
5.2 KiB
Elixir

defmodule LocalspotWeb.Layouts do
@moduledoc """
This module holds layouts and related functionality
used by your application.
"""
use LocalspotWeb, :html
# Embed all files in layouts/* within this module.
# The default root.html.heex file contains the HTML
# skeleton of your application, namely HTML headers
# and other static content.
embed_templates "layouts/*"
@doc """
Renders your app layout.
This function is typically invoked from every template,
and it often contains your application menu, sidebar,
or similar.
## Examples
<Layouts.app flash={@flash}>
<h1>Content</h1>
</Layouts.app>
"""
attr :flash, :map, required: true, doc: "the map of flash messages"
attr :current_scope, :map,
default: nil,
doc: "the current [scope](https://hexdocs.pm/phoenix/scopes.html)"
slot :inner_block, required: true
def app(assigns) do
~H"""
<header class="navbar bg-base-200 px-4 sm:px-6 lg:px-8 shadow-sm">
<div class="flex-1">
<a href={~p"/"} class="btn btn-ghost text-xl gap-2">
<.icon name="hero-map-pin" class="w-6 h-6 text-primary" />
<span class="font-bold">LocalSpot</span>
</a>
</div>
<div class="flex-none">
<ul class="menu menu-horizontal px-1 items-center gap-1">
<li>
<a href={~p"/businesses"} class="btn btn-ghost btn-sm">
<.icon name="hero-building-storefront" class="w-4 h-4" /> Businesses
</a>
</li>
<li>
<a href={~p"/categories"} class="btn btn-ghost btn-sm">
<.icon name="hero-squares-2x2" class="w-4 h-4" /> Categories
</a>
</li>
<li>
<a href={~p"/businesses/map"} class="btn btn-ghost btn-sm">
<.icon name="hero-map" class="w-4 h-4" /> Map
</a>
</li>
<li>
<.theme_toggle />
</li>
<li>
<a href={~p"/businesses/new"} class="btn btn-primary btn-sm">
<.icon name="hero-plus" class="w-4 h-4" /> Add Business
</a>
</li>
</ul>
</div>
</header>
<main class="px-4 py-8 sm:px-6 lg:px-8">
<div class="mx-auto max-w-6xl space-y-4">
{render_slot(@inner_block)}
</div>
</main>
<footer class="footer footer-center p-4 bg-base-200 text-base-content mt-8">
<aside>
<p>LocalSpot - Discover locally owned businesses in your community</p>
</aside>
</footer>
<.flash_group flash={@flash} />
"""
end
@doc """
Shows the flash group with standard titles and content.
## Examples
<.flash_group flash={@flash} />
"""
attr :flash, :map, required: true, doc: "the map of flash messages"
attr :id, :string, default: "flash-group", doc: "the optional id of flash container"
def flash_group(assigns) do
~H"""
<div id={@id} aria-live="polite">
<.flash kind={:info} flash={@flash} />
<.flash kind={:error} flash={@flash} />
<.flash
id="client-error"
kind={:error}
title={gettext("We can't find the internet")}
phx-disconnected={show(".phx-client-error #client-error") |> JS.remove_attribute("hidden")}
phx-connected={hide("#client-error") |> JS.set_attribute({"hidden", ""})}
hidden
>
{gettext("Attempting to reconnect")}
<.icon name="hero-arrow-path" class="ml-1 size-3 motion-safe:animate-spin" />
</.flash>
<.flash
id="server-error"
kind={:error}
title={gettext("Something went wrong!")}
phx-disconnected={show(".phx-server-error #server-error") |> JS.remove_attribute("hidden")}
phx-connected={hide("#server-error") |> JS.set_attribute({"hidden", ""})}
hidden
>
{gettext("Attempting to reconnect")}
<.icon name="hero-arrow-path" class="ml-1 size-3 motion-safe:animate-spin" />
</.flash>
</div>
"""
end
@doc """
Provides dark vs light theme toggle based on themes defined in app.css.
See <head> in root.html.heex which applies the theme before page load.
"""
def theme_toggle(assigns) do
~H"""
<div class="card relative flex flex-row items-center border-2 border-base-300 bg-base-300 rounded-full">
<div class="absolute w-1/3 h-full rounded-full border-1 border-base-200 bg-base-100 brightness-200 left-0 [[data-theme=light]_&]:left-1/3 [[data-theme=dark]_&]:left-2/3 transition-[left]" />
<button
class="flex p-2 cursor-pointer w-1/3"
phx-click={JS.dispatch("phx:set-theme")}
data-phx-theme="system"
>
<.icon name="hero-computer-desktop-micro" class="size-4 opacity-75 hover:opacity-100" />
</button>
<button
class="flex p-2 cursor-pointer w-1/3"
phx-click={JS.dispatch("phx:set-theme")}
data-phx-theme="light"
>
<.icon name="hero-sun-micro" class="size-4 opacity-75 hover:opacity-100" />
</button>
<button
class="flex p-2 cursor-pointer w-1/3"
phx-click={JS.dispatch("phx:set-theme")}
data-phx-theme="dark"
>
<.icon name="hero-moon-micro" class="size-4 opacity-75 hover:opacity-100" />
</button>
</div>
"""
end
end