chiya/lib/chiya_web/components/public_components.ex

414 lines
14 KiB
Elixir
Raw Normal View History

defmodule ChiyaWeb.PublicComponents do
use Phoenix.Component
use Phoenix.VerifiedRoutes,
endpoint: ChiyaWeb.Endpoint,
router: ChiyaWeb.Router,
statics: ChiyaWeb.static_paths()
2023-05-07 10:30:32 +02:00
import ChiyaWeb.Format
import ChiyaWeb.Markdown, only: [render: 1]
import Phoenix.HTML, only: [raw: 1]
2023-06-19 21:49:01 +02:00
2023-08-03 06:49:48 +02:00
import ChiyaWeb.DarkModeToggle
2023-06-19 21:49:01 +02:00
@doc """
Renders a [Hero Icon](https://heroicons.com).
Hero icons come in three styles outline, solid, and mini.
By default, the outline style is used, but solid an mini may
be applied by using the `-solid` and `-mini` suffix.
You can customize the size and colors of the icons by setting
width, height, and background color classes.
Icons are extracted from your `priv/hero_icons` directory and bundled
within your compiled app.css by the plugin in your `assets/tailwind.config.js`.
## Examples
<.icon name="hero-cake" />
<.icon name="hero-cake-solid" />
<.icon name="hero-cake-mini" />
<.icon name="hero-bolt" class="bg-blue-500 w-10 h-10" />
"""
attr :name, :string, required: true
attr :class, :string, default: nil
def icon(%{name: "hero-" <> _} = assigns) do
~H"""
<span class={[@name, @class]} />
"""
end
2023-05-07 10:30:32 +02:00
2023-05-22 20:05:44 +02:00
@doc """
Renders a middot as divider
"""
attr :class, :string, default: "text-theme-primary"
def dot(assigns),
do: ~H"""
<span class={["font-bold", @class]}>·</span>
2023-05-22 20:05:44 +02:00
"""
@doc """
Renders a horizontal line
"""
def line(assigns),
do: ~H"""
<hr class="my-6 border-theme-base/20" />
"""
2023-05-22 20:05:44 +02:00
attr :text, :string, default: ""
2023-06-03 09:27:43 +02:00
def divider(assigns) do
~H"""
2023-06-28 16:02:30 +02:00
<div class="flex items-center my-8 text-theme-primary before:flex-1 after:flex-1 before:content-[''] after:content-[''] before:p-[0.5px] after:p-[0.5px] before:bg-theme-base/25 after:bg-theme-base/25 w-full mx-auto last:hidden">
<%= assigns.text %>
</div>
"""
end
attr :note, :map, required: true
attr :class_tag, :string, default: ""
2023-08-01 07:32:06 +02:00
attr :linked, :boolean, default: true
def tags(assigns) do
~H"""
<span class="inline-flex flex-row gap-1">
<%= for tag <- @note.tags do %>
2023-08-01 07:32:06 +02:00
<%= if assigns.linked do %>
2023-08-02 20:48:45 +02:00
<a class={["p-category", @class_tag]} href={~p"/tagged-with/#{tag.slug}"}>
<%= tag.name %>
</a>
2023-08-01 07:32:06 +02:00
<% else %>
2023-08-02 20:48:45 +02:00
<span class={["p-category", @class_tag]}>
<%= tag.name %>
</span>
2023-08-01 07:32:06 +02:00
<% end %>
<.dot class="text-theme-base/50 last:hidden" />
<% end %>
</span>
"""
end
@doc """
Renders a note-header with title.
"""
attr :class, :string, default: nil
2023-07-06 21:28:56 +02:00
attr :class_title, :string, default: nil
attr :class_subtitle, :string, default: nil
2023-07-06 21:28:56 +02:00
slot :subtitle, required: false
def header(assigns) do
~H"""
2023-07-06 21:28:56 +02:00
<header class={["p-8 rounded bg-theme-background1", @class]}>
<h1 class={["text-3xl leading-10 text-theme-base", @class_title]}>
2023-07-09 20:01:03 +02:00
<%= render_slot(@inner_block) %>
</h1>
2023-07-09 20:01:03 +02:00
<p
:if={@subtitle != []}
class={["mt-4 leading-7 font-semibold text-theme-base/75", @class_subtitle]}
>
<%= render_slot(@subtitle) %>
</p>
</header>
"""
end
2023-05-07 10:30:32 +02:00
attr :layout, :atom, default: :list
attr :notes, :list, required: true
2023-07-27 20:31:23 +02:00
attr :show_content, :boolean, default: true
2023-05-07 10:30:32 +02:00
def note_list(assigns) do
case assigns.layout do
:gallery ->
2023-06-27 22:04:44 +02:00
note_list_gallery(assigns)
:microblog ->
note_list_microblog(assigns)
_ ->
2023-08-01 07:32:06 +02:00
note_list_default(assigns)
2023-06-27 22:04:44 +02:00
end
end
2023-07-09 20:01:03 +02:00
attr :notes, :list, required: true
@doc """
Default note list that renders a list of rounded boxes,
which show the note title and an excerpt of the content
"""
2023-08-01 07:32:06 +02:00
def note_list_default(assigns) do
2023-06-27 22:04:44 +02:00
~H"""
2023-08-01 07:32:06 +02:00
<section class="note-list default | sm:w-auto flex flex-col gap-3">
2023-06-27 22:04:44 +02:00
<%= for note <- assigns.notes do %>
2023-07-09 20:01:03 +02:00
<a
href={~p"/note/#{note.slug}"}
class="block rounded-lg px-6 py-4 border border-theme-background1 hover:bg-theme-background1 transition"
2023-07-09 20:01:03 +02:00
>
2023-07-27 20:31:23 +02:00
<header class="flex flex-row items-center gap-1">
2023-07-20 07:26:48 +02:00
<span class="text-theme-primary text-lg font-semibold leading-8 flex-1">
<%= note.name %>
</span>
<span class="text-theme-base/75 text-sm">
<%= pretty_date(note.published_at) %>
</span>
</header>
2023-07-27 20:31:23 +02:00
<%= if assigns.show_content do %>
<p class="text-theme-base">
<%= String.slice(note.content, 0..150) %>
</p>
2023-07-27 20:31:23 +02:00
<% end %>
<%= if not Enum.empty?(note.tags) do %>
2023-08-02 20:48:45 +02:00
<span class="inline-block">
<.tags note={note} linked={false} />
</span>
<% end %>
2023-06-27 22:04:44 +02:00
</a>
<% end %>
</section>
"""
end
attr :notes, :list, required: true
2023-06-27 22:04:44 +02:00
def note_list_microblog(assigns) do
~H"""
<section class="note-list microblog | mt-6 text-theme-base">
<%= for note <- assigns.notes do %>
<article class="mt-4 first:mt-0">
<.featured_images note={note} />
<header class="mt-4 text-lg">
<%= if(note.kind == :bookmark) do %>
<strong><%= note.name %></strong>
<% end %>
</header>
<div class="prose prose-gruvbox mt-4">
2023-06-27 22:04:44 +02:00
<%= raw(render(note.content)) %>
</div>
2023-07-25 06:55:04 +02:00
<footer class="mt-4">
<a href={~p"/note/#{note.slug}"}>
<time class="text-theme-base/75">
<%= pretty_datetime(note.published_at) %>
</time>
</a>
2023-07-09 20:01:03 +02:00
<%= if not Enum.empty?(note.tags) do %>
2023-06-27 22:04:44 +02:00
<.dot />
<.tags note={note} class_tag="text-theme-base/75" />
2023-06-27 22:04:44 +02:00
<% end %>
</footer>
</article>
<.divider />
<% end %>
</section>
"""
end
attr :notes, :list, required: true
2023-06-27 22:04:44 +02:00
def note_list_gallery(assigns) do
~H"""
<section class="note-list gallery | mt-6">
<%= for note <- assigns.notes do %>
<article>
<section class="flex flex-wrap justify-start gap-3">
<%= for image <- note.images do %>
2023-05-07 10:30:32 +02:00
<a
2023-06-27 22:04:44 +02:00
href={ChiyaWeb.Helpers.image_url(image, :full)}
class="lightbox | w-28"
data-gallery={gallery_name(note)}
data-description={ChiyaWeb.Markdown.render(image.content)}
2023-05-07 10:30:32 +02:00
>
2023-06-27 22:04:44 +02:00
<img src={ChiyaWeb.Helpers.image_url(image, :thumb)} loading="lazy" />
2023-05-07 10:30:32 +02:00
</a>
2023-06-27 22:04:44 +02:00
<% end %>
</section>
<a
href={~p"/note/#{note.slug}"}
class="text-theme-secondary text-lg/10 font-semibold rounded-lg -mx-2 -my-0.5 px-2 py-0.5 hover:bg-theme-secondary/10 transition"
>
<%= note.name %>
<span class="text-theme-base/75 text-sm">
<%= pretty_date(note.published_at) %>
</span>
</a>
</article>
<.line />
<% end %>
</section>
"""
end
2023-08-03 06:49:48 +02:00
attr :user, :map, required: true
def site_header(assigns) do
~H"""
<nav class="mx-auto max-w-2xl">
<ul class="flex gap-3">
<li>
<a href="/" class="button">
<.icon name="hero-home" />
</a>
</li>
<li>
<a href="/about" class="button">
<svg
xmlns="http://www.w3.org/2000/svg"
width="16"
height="16"
fill="currentColor"
class="bi bi-umbrella inline"
viewBox="0 0 16 16"
>
<path d="M8 0a.5.5 0 0 1 .5.5v.514C12.625 1.238 16 4.22 16 8c0 0 0 .5-.5.5-.149 0-.352-.145-.352-.145l-.004-.004-.025-.023a3.484 3.484 0 0 0-.555-.394A3.166 3.166 0 0 0 13 7.5c-.638 0-1.178.213-1.564.434a3.484 3.484 0 0 0-.555.394l-.025.023-.003.003s-.204.146-.353.146-.352-.145-.352-.145l-.004-.004-.025-.023a3.484 3.484 0 0 0-.555-.394 3.3 3.3 0 0 0-1.064-.39V13.5H8h.5v.039l-.005.083a2.958 2.958 0 0 1-.298 1.102 2.257 2.257 0 0 1-.763.88C7.06 15.851 6.587 16 6 16s-1.061-.148-1.434-.396a2.255 2.255 0 0 1-.763-.88 2.958 2.958 0 0 1-.302-1.185v-.025l-.001-.009v-.003s0-.002.5-.002h-.5V13a.5.5 0 0 1 1 0v.506l.003.044a1.958 1.958 0 0 0 .195.726c.095.191.23.367.423.495.19.127.466.229.879.229s.689-.102.879-.229c.193-.128.328-.304.424-.495a1.958 1.958 0 0 0 .197-.77V7.544a3.3 3.3 0 0 0-1.064.39 3.482 3.482 0 0 0-.58.417l-.004.004S5.65 8.5 5.5 8.5c-.149 0-.352-.145-.352-.145l-.004-.004a3.482 3.482 0 0 0-.58-.417A3.166 3.166 0 0 0 3 7.5c-.638 0-1.177.213-1.564.434a3.482 3.482 0 0 0-.58.417l-.004.004S.65 8.5.5 8.5C0 8.5 0 8 0 8c0-3.78 3.375-6.762 7.5-6.986V.5A.5.5 0 0 1 8 0zM6.577 2.123c-2.833.5-4.99 2.458-5.474 4.854A4.124 4.124 0 0 1 3 6.5c.806 0 1.48.25 1.962.511a9.706 9.706 0 0 1 .344-2.358c.242-.868.64-1.765 1.271-2.53zm-.615 4.93A4.16 4.16 0 0 1 8 6.5a4.16 4.16 0 0 1 2.038.553 8.688 8.688 0 0 0-.307-2.13C9.434 3.858 8.898 2.83 8 2.117c-.898.712-1.434 1.74-1.731 2.804a8.687 8.687 0 0 0-.307 2.131zm3.46-4.93c.631.765 1.03 1.662 1.272 2.53.233.833.328 1.66.344 2.358A4.14 4.14 0 0 1 13 6.5c.77 0 1.42.23 1.897.477-.484-2.396-2.641-4.355-5.474-4.854z" />
</svg>
<span class="hidden sm:inline-block">About</span>
</a>
</li>
<li>
<a href="/wiki" class="button">
<svg
xmlns="http://www.w3.org/2000/svg"
width="16"
height="16"
fill="currentColor"
class="bi bi-journals inline"
viewBox="0 0 16 16"
>
<path d="M5 0h8a2 2 0 0 1 2 2v10a2 2 0 0 1-2 2 2 2 0 0 1-2 2H3a2 2 0 0 1-2-2h1a1 1 0 0 0 1 1h8a1 1 0 0 0 1-1V4a1 1 0 0 0-1-1H3a1 1 0 0 0-1 1H1a2 2 0 0 1 2-2h8a2 2 0 0 1 2 2v9a1 1 0 0 0 1-1V2a1 1 0 0 0-1-1H5a1 1 0 0 0-1 1H3a2 2 0 0 1 2-2z" />
<path d="M1 6v-.5a.5.5 0 0 1 1 0V6h.5a.5.5 0 0 1 0 1h-2a.5.5 0 0 1 0-1H1zm0 3v-.5a.5.5 0 0 1 1 0V9h.5a.5.5 0 0 1 0 1h-2a.5.5 0 0 1 0-1H1zm0 2.5v.5H.5a.5.5 0 0 0 0 1h2a.5.5 0 0 0 0-1H2v-.5a.5.5 0 0 0-1 0z" />
</svg>
<span class="hidden sm:inline">Wiki</span>
</a>
</li>
<li>
<a href="/bookmarks" class="button">
<svg
xmlns="http://www.w3.org/2000/svg"
width="16"
height="16"
fill="currentColor"
class="bi bi-bookmark-fill inline"
viewBox="0 0 16 16"
>
<path d="M2 2v13.5a.5.5 0 0 0 .74.439L8 13.069l5.26 2.87A.5.5 0 0 0 14 15.5V2a2 2 0 0 0-2-2H4a2 2 0 0 0-2 2z" />
</svg>
<span class="hidden sm:inline-block">Bookmarks</span>
</a>
</li>
<li class="flex-1"></li>
<%= if @user do %>
<li>
<a href="/admin" class="button">
<.icon name="hero-beaker" />
</a>
</li>
<% end %>
<li>
<.darkmode_toggle class="button" />
</li>
</ul>
</nav>
"""
end
attr :note, :map, required: true
2023-06-27 22:04:44 +02:00
def featured_images(assigns) do
images = main_images(assigns.note)
2023-05-07 10:30:32 +02:00
2023-06-27 22:04:44 +02:00
case Enum.count(images) do
0 ->
~H"""
<figure />
2023-05-07 10:30:32 +02:00
"""
2023-06-27 22:04:44 +02:00
1 ->
assigns = assign(assigns, :image, List.first(images))
2023-05-07 10:30:32 +02:00
~H"""
2023-07-25 06:55:04 +02:00
<figure>
<.featured_image image={assigns.image} size={:full} class="rounded-lg" />
2023-06-27 22:04:44 +02:00
</figure>
"""
2023-06-27 22:04:44 +02:00
2 ->
assigns =
assigns
|> assign(:first, Enum.at(images, 0))
|> assign(:second, Enum.at(images, 1))
~H"""
2023-06-27 22:04:44 +02:00
<figure class="flex gap-1">
2023-07-25 06:55:04 +02:00
<.featured_image image={assigns.first} size={:thumb} class="rounded-l flex-1 w-full" />
<.featured_image image={assigns.second} size={:thumb} class="rounded-r flex-1 w-full" />
2023-06-27 22:04:44 +02:00
</figure>
2023-05-07 10:30:32 +02:00
"""
2023-06-27 22:04:44 +02:00
2023-07-05 06:49:37 +02:00
3 ->
assigns =
assigns
|> assign(:first, Enum.at(images, 0))
|> assign(:second, Enum.at(images, 1))
|> assign(:third, Enum.at(images, 2))
~H"""
<figure class="flex gap-1">
2023-07-25 06:55:04 +02:00
<.featured_image image={assigns.first} size={:thumb} class="flex-1 w-full rounded-l" />
<.featured_image image={assigns.second} size={:thumb} class="flex-1 w-full" />
<.featured_image image={assigns.third} size={:thumb} class="flex-1 w-full rounded-r" />
</figure>
"""
_ ->
assigns =
assigns
|> assign(:first, Enum.at(images, 0))
|> assign(:second, Enum.at(images, 1))
|> assign(:third, Enum.at(images, 2))
|> assign(:fourth, Enum.at(images, 3))
~H"""
<figure class="flex gap-1 flex-col">
<section class="flex gap-1">
<.featured_image image={assigns.first} size={:thumb} class="flex-1 w-full rounded-tl-lg" />
<.featured_image image={assigns.second} size={:thumb} class="flex-1 w-full rounded-tr-lg" />
</section>
<section class="flex gap-1">
<.featured_image image={assigns.third} size={:thumb} class="flex-1 w-full rounded-bl-lg" />
<.featured_image image={assigns.fourth} size={:thumb} class="flex-1 w-full rounded-br-lg" />
</section>
2023-07-05 06:49:37 +02:00
</figure>
"""
2023-05-07 10:30:32 +02:00
end
end
2023-07-25 06:55:04 +02:00
attr :image, :map, required: true
attr :class, :string
attr :size, :atom, default: :thumb
defp featured_image(assigns) do
~H"""
<img
src={ChiyaWeb.Helpers.image_url(assigns.image, assigns.size)}
class={assigns.class}
title={assigns.image.content}
/>
"""
end
2023-05-07 10:30:32 +02:00
defp gallery_name(note), do: "gallery-#{note.id}"
2023-06-27 22:04:44 +02:00
defp main_images(note),
do: Enum.filter(note.images, fn image -> image.featured end)
end