feat: add product sorting to collection pages with tests
Add sort functionality to /collections/:slug pages with 6 sort options (featured, newest, price ascending/descending, name A-Z/Z-A). Sort selection persists across category navigation via URL query params. Refactor handle_params to be DRY using load_collection/1 helper. Add comprehensive unit tests for PreviewData category functions and LiveView tests for the Collection page sorting and navigation. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
parent
9fb836ca0d
commit
fe29c1ad36
@ -5,6 +5,15 @@ defmodule SimpleshopThemeWeb.ShopLive.Collection do
|
|||||||
alias SimpleshopTheme.Media
|
alias SimpleshopTheme.Media
|
||||||
alias SimpleshopTheme.Theme.{CSSCache, CSSGenerator, PreviewData}
|
alias SimpleshopTheme.Theme.{CSSCache, CSSGenerator, PreviewData}
|
||||||
|
|
||||||
|
@sort_options [
|
||||||
|
{"featured", "Featured"},
|
||||||
|
{"newest", "Newest"},
|
||||||
|
{"price_asc", "Price: Low to High"},
|
||||||
|
{"price_desc", "Price: High to Low"},
|
||||||
|
{"name_asc", "Name: A-Z"},
|
||||||
|
{"name_desc", "Name: Z-A"}
|
||||||
|
]
|
||||||
|
|
||||||
@impl true
|
@impl true
|
||||||
def mount(_params, _session, socket) do
|
def mount(_params, _session, socket) do
|
||||||
theme_settings = Settings.get_theme_settings()
|
theme_settings = Settings.get_theme_settings()
|
||||||
@ -32,40 +41,62 @@ defmodule SimpleshopThemeWeb.ShopLive.Collection do
|
|||||||
|> assign(:cart_count, 0)
|
|> assign(:cart_count, 0)
|
||||||
|> assign(:cart_subtotal, "£0.00")
|
|> assign(:cart_subtotal, "£0.00")
|
||||||
|> assign(:categories, PreviewData.categories())
|
|> assign(:categories, PreviewData.categories())
|
||||||
|
|> assign(:sort_options, @sort_options)
|
||||||
|
|> assign(:current_sort, "featured")
|
||||||
|
|
||||||
{:ok, socket}
|
{:ok, socket}
|
||||||
end
|
end
|
||||||
|
|
||||||
@impl true
|
@impl true
|
||||||
def handle_params(%{"slug" => "all"}, _uri, socket) do
|
def handle_params(%{"slug" => slug} = params, _uri, socket) do
|
||||||
|
sort = params["sort"] || "featured"
|
||||||
|
|
||||||
|
case load_collection(slug) do
|
||||||
|
{:ok, title, category, products} ->
|
||||||
{:noreply,
|
{:noreply,
|
||||||
socket
|
socket
|
||||||
|> assign(:page_title, "All Products")
|
|> assign(:page_title, title)
|
||||||
|> assign(:collection_title, "All Products")
|
|> assign(:collection_title, title)
|
||||||
|> assign(:current_category, nil)
|
|> assign(:current_category, category)
|
||||||
|> assign(:products, PreviewData.products())}
|
|> assign(:current_sort, sort)
|
||||||
end
|
|> assign(:products, sort_products(products, sort))}
|
||||||
|
|
||||||
def handle_params(%{"slug" => slug}, _uri, socket) do
|
:not_found ->
|
||||||
case PreviewData.category_by_slug(slug) do
|
|
||||||
nil ->
|
|
||||||
{:noreply,
|
{:noreply,
|
||||||
socket
|
socket
|
||||||
|> put_flash(:error, "Collection not found")
|
|> put_flash(:error, "Collection not found")
|
||||||
|> push_navigate(to: ~p"/collections/all")}
|
|> push_navigate(to: ~p"/collections/all")}
|
||||||
|
|
||||||
category ->
|
|
||||||
products = PreviewData.products_by_category(slug)
|
|
||||||
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> assign(:page_title, category.name)
|
|
||||||
|> assign(:collection_title, category.name)
|
|
||||||
|> assign(:current_category, category)
|
|
||||||
|> assign(:products, products)}
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
defp load_collection("all") do
|
||||||
|
{:ok, "All Products", nil, PreviewData.products()}
|
||||||
|
end
|
||||||
|
|
||||||
|
defp load_collection(slug) do
|
||||||
|
case PreviewData.category_by_slug(slug) do
|
||||||
|
nil -> :not_found
|
||||||
|
category -> {:ok, category.name, category, PreviewData.products_by_category(slug)}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
@impl true
|
||||||
|
def handle_event("sort_changed", %{"sort" => sort}, socket) do
|
||||||
|
slug = if socket.assigns.current_category, do: socket.assigns.current_category.slug, else: "all"
|
||||||
|
{:noreply, push_patch(socket, to: ~p"/collections/#{slug}?sort=#{sort}")}
|
||||||
|
end
|
||||||
|
|
||||||
|
defp sort_products(products, "featured"), do: products
|
||||||
|
defp sort_products(products, "newest"), do: Enum.reverse(products)
|
||||||
|
defp sort_products(products, "price_asc"), do: Enum.sort_by(products, & &1.price)
|
||||||
|
defp sort_products(products, "price_desc"), do: Enum.sort_by(products, & &1.price, :desc)
|
||||||
|
defp sort_products(products, "name_asc"), do: Enum.sort_by(products, & &1.name)
|
||||||
|
defp sort_products(products, "name_desc"), do: Enum.sort_by(products, & &1.name, :desc)
|
||||||
|
defp sort_products(products, _), do: products
|
||||||
|
|
||||||
|
defp collection_path(slug, "featured"), do: ~p"/collections/#{slug}"
|
||||||
|
defp collection_path(slug, sort), do: ~p"/collections/#{slug}?sort=#{sort}"
|
||||||
|
|
||||||
@impl true
|
@impl true
|
||||||
def render(assigns) do
|
def render(assigns) do
|
||||||
~H"""
|
~H"""
|
||||||
@ -92,7 +123,12 @@ defmodule SimpleshopThemeWeb.ShopLive.Collection do
|
|||||||
/>
|
/>
|
||||||
|
|
||||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
|
<div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
|
||||||
<.collection_filter_bar categories={@categories} current_slug={@current_category && @current_category.slug} />
|
<.collection_filter_bar
|
||||||
|
categories={@categories}
|
||||||
|
current_slug={@current_category && @current_category.slug}
|
||||||
|
sort_options={@sort_options}
|
||||||
|
current_sort={@current_sort}
|
||||||
|
/>
|
||||||
|
|
||||||
<SimpleshopThemeWeb.ShopComponents.product_grid theme_settings={@theme_settings}>
|
<SimpleshopThemeWeb.ShopComponents.product_grid theme_settings={@theme_settings}>
|
||||||
<%= for product <- @products do %>
|
<%= for product <- @products do %>
|
||||||
@ -128,11 +164,12 @@ defmodule SimpleshopThemeWeb.ShopLive.Collection do
|
|||||||
|
|
||||||
defp collection_filter_bar(assigns) do
|
defp collection_filter_bar(assigns) do
|
||||||
~H"""
|
~H"""
|
||||||
<nav class="mb-8" aria-label="Collection filters">
|
<div class="flex flex-wrap items-center justify-between gap-4 mb-8">
|
||||||
|
<nav aria-label="Collection filters">
|
||||||
<ul class="flex flex-wrap gap-2">
|
<ul class="flex flex-wrap gap-2">
|
||||||
<li>
|
<li>
|
||||||
<.link
|
<.link
|
||||||
navigate={~p"/collections/all"}
|
navigate={collection_path("all", @current_sort)}
|
||||||
class={[
|
class={[
|
||||||
"px-4 py-2 rounded-full text-sm transition-colors",
|
"px-4 py-2 rounded-full text-sm transition-colors",
|
||||||
if(@current_slug == nil, do: "font-medium", else: "hover:opacity-80")
|
if(@current_slug == nil, do: "font-medium", else: "hover:opacity-80")
|
||||||
@ -148,7 +185,7 @@ defmodule SimpleshopThemeWeb.ShopLive.Collection do
|
|||||||
<%= for category <- @categories do %>
|
<%= for category <- @categories do %>
|
||||||
<li>
|
<li>
|
||||||
<.link
|
<.link
|
||||||
navigate={~p"/collections/#{category.slug}"}
|
navigate={collection_path(category.slug, @current_sort)}
|
||||||
class={[
|
class={[
|
||||||
"px-4 py-2 rounded-full text-sm transition-colors",
|
"px-4 py-2 rounded-full text-sm transition-colors",
|
||||||
if(@current_slug == category.slug, do: "font-medium", else: "hover:opacity-80")
|
if(@current_slug == category.slug, do: "font-medium", else: "hover:opacity-80")
|
||||||
@ -164,6 +201,20 @@ defmodule SimpleshopThemeWeb.ShopLive.Collection do
|
|||||||
<% end %>
|
<% end %>
|
||||||
</ul>
|
</ul>
|
||||||
</nav>
|
</nav>
|
||||||
|
|
||||||
|
<form phx-change="sort_changed">
|
||||||
|
<select
|
||||||
|
name="sort"
|
||||||
|
class="px-4 py-2 text-sm"
|
||||||
|
style="background-color: var(--t-surface-raised); color: var(--t-text-primary); border: 1px solid var(--t-border-default); border-radius: var(--t-radius-input);"
|
||||||
|
aria-label="Sort products"
|
||||||
|
>
|
||||||
|
<%= for {value, label} <- @sort_options do %>
|
||||||
|
<option value={value} selected={@current_sort == value}>{label}</option>
|
||||||
|
<% end %>
|
||||||
|
</select>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
"""
|
"""
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@ -198,4 +198,81 @@ defmodule SimpleshopTheme.Theme.PreviewDataTest do
|
|||||||
assert PreviewData.has_real_products?() == false
|
assert PreviewData.has_real_products?() == false
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
describe "category_by_slug/1" do
|
||||||
|
test "returns category when slug exists" do
|
||||||
|
category = PreviewData.category_by_slug("art-prints")
|
||||||
|
|
||||||
|
assert category != nil
|
||||||
|
assert category.slug == "art-prints"
|
||||||
|
assert is_binary(category.name)
|
||||||
|
end
|
||||||
|
|
||||||
|
test "returns nil when slug does not exist" do
|
||||||
|
assert PreviewData.category_by_slug("nonexistent") == nil
|
||||||
|
end
|
||||||
|
|
||||||
|
test "finds all known categories by slug" do
|
||||||
|
categories = PreviewData.categories()
|
||||||
|
|
||||||
|
for category <- categories do
|
||||||
|
found = PreviewData.category_by_slug(category.slug)
|
||||||
|
assert found != nil
|
||||||
|
assert found.id == category.id
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe "products_by_category/1" do
|
||||||
|
test "returns all products for nil" do
|
||||||
|
all_products = PreviewData.products()
|
||||||
|
filtered = PreviewData.products_by_category(nil)
|
||||||
|
|
||||||
|
assert filtered == all_products
|
||||||
|
end
|
||||||
|
|
||||||
|
test "returns all products for 'all'" do
|
||||||
|
all_products = PreviewData.products()
|
||||||
|
filtered = PreviewData.products_by_category("all")
|
||||||
|
|
||||||
|
assert filtered == all_products
|
||||||
|
end
|
||||||
|
|
||||||
|
test "returns empty list for nonexistent category" do
|
||||||
|
assert PreviewData.products_by_category("nonexistent") == []
|
||||||
|
end
|
||||||
|
|
||||||
|
test "returns only products matching the category" do
|
||||||
|
category = List.first(PreviewData.categories())
|
||||||
|
products = PreviewData.products_by_category(category.slug)
|
||||||
|
|
||||||
|
assert is_list(products)
|
||||||
|
|
||||||
|
for product <- products do
|
||||||
|
assert product.category == category.name
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
test "products are filtered correctly for each category" do
|
||||||
|
categories = PreviewData.categories()
|
||||||
|
|
||||||
|
for category <- categories do
|
||||||
|
products = PreviewData.products_by_category(category.slug)
|
||||||
|
|
||||||
|
for product <- products do
|
||||||
|
assert product.category == category.name
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
test "all products belong to at least one category" do
|
||||||
|
all_products = PreviewData.products()
|
||||||
|
categories = PreviewData.categories()
|
||||||
|
category_names = Enum.map(categories, & &1.name)
|
||||||
|
|
||||||
|
for product <- all_products do
|
||||||
|
assert product.category in category_names
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
154
test/simpleshop_theme_web/live/shop_live/collection_test.exs
Normal file
154
test/simpleshop_theme_web/live/shop_live/collection_test.exs
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
defmodule SimpleshopThemeWeb.ShopLive.CollectionTest do
|
||||||
|
use SimpleshopThemeWeb.ConnCase, async: false
|
||||||
|
|
||||||
|
import Phoenix.LiveViewTest
|
||||||
|
|
||||||
|
alias SimpleshopTheme.Theme.PreviewData
|
||||||
|
|
||||||
|
describe "Collection page" do
|
||||||
|
test "renders collection page for /collections/all", %{conn: conn} do
|
||||||
|
{:ok, _view, html} = live(conn, ~p"/collections/all")
|
||||||
|
|
||||||
|
assert html =~ "All Products"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "renders collection page for specific category", %{conn: conn} do
|
||||||
|
category = List.first(PreviewData.categories())
|
||||||
|
{:ok, _view, html} = live(conn, ~p"/collections/#{category.slug}")
|
||||||
|
|
||||||
|
assert html =~ category.name
|
||||||
|
end
|
||||||
|
|
||||||
|
test "displays products", %{conn: conn} do
|
||||||
|
{:ok, _view, html} = live(conn, ~p"/collections/all")
|
||||||
|
|
||||||
|
products = PreviewData.products()
|
||||||
|
first_product = List.first(products)
|
||||||
|
|
||||||
|
assert html =~ first_product.name
|
||||||
|
end
|
||||||
|
|
||||||
|
test "displays category filter buttons", %{conn: conn} do
|
||||||
|
{:ok, _view, html} = live(conn, ~p"/collections/all")
|
||||||
|
|
||||||
|
categories = PreviewData.categories()
|
||||||
|
|
||||||
|
for category <- categories do
|
||||||
|
assert html =~ category.name
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
test "displays sort dropdown", %{conn: conn} do
|
||||||
|
{:ok, _view, html} = live(conn, ~p"/collections/all")
|
||||||
|
|
||||||
|
assert html =~ "Featured"
|
||||||
|
assert html =~ "Newest"
|
||||||
|
assert html =~ "Price: Low to High"
|
||||||
|
assert html =~ "Price: High to Low"
|
||||||
|
assert html =~ "Name: A-Z"
|
||||||
|
assert html =~ "Name: Z-A"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "redirects to /collections/all for unknown category", %{conn: conn} do
|
||||||
|
{:error, {:live_redirect, %{to: to, flash: flash}}} =
|
||||||
|
live(conn, ~p"/collections/nonexistent")
|
||||||
|
|
||||||
|
assert to == "/collections/all"
|
||||||
|
assert flash["error"] == "Collection not found"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "filters products by category", %{conn: conn} do
|
||||||
|
category = List.first(PreviewData.categories())
|
||||||
|
{:ok, _view, html} = live(conn, ~p"/collections/#{category.slug}")
|
||||||
|
|
||||||
|
products = PreviewData.products_by_category(category.slug)
|
||||||
|
|
||||||
|
for product <- products do
|
||||||
|
assert html =~ product.name
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe "Sorting" do
|
||||||
|
test "sorts by price ascending", %{conn: conn} do
|
||||||
|
{:ok, view, _html} = live(conn, ~p"/collections/all")
|
||||||
|
|
||||||
|
html =
|
||||||
|
view
|
||||||
|
|> element("form[phx-change='sort_changed']")
|
||||||
|
|> render_change(%{sort: "price_asc"})
|
||||||
|
|
||||||
|
assert html =~ "Price: Low to High"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "sorts by price descending", %{conn: conn} do
|
||||||
|
{:ok, view, _html} = live(conn, ~p"/collections/all")
|
||||||
|
|
||||||
|
html =
|
||||||
|
view
|
||||||
|
|> element("form[phx-change='sort_changed']")
|
||||||
|
|> render_change(%{sort: "price_desc"})
|
||||||
|
|
||||||
|
assert html =~ "Price: High to Low"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "sorts by name ascending", %{conn: conn} do
|
||||||
|
{:ok, view, _html} = live(conn, ~p"/collections/all")
|
||||||
|
|
||||||
|
html =
|
||||||
|
view
|
||||||
|
|> element("form[phx-change='sort_changed']")
|
||||||
|
|> render_change(%{sort: "name_asc"})
|
||||||
|
|
||||||
|
assert html =~ "Name: A-Z"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "sorts by name descending", %{conn: conn} do
|
||||||
|
{:ok, view, _html} = live(conn, ~p"/collections/all")
|
||||||
|
|
||||||
|
html =
|
||||||
|
view
|
||||||
|
|> element("form[phx-change='sort_changed']")
|
||||||
|
|> render_change(%{sort: "name_desc"})
|
||||||
|
|
||||||
|
assert html =~ "Name: Z-A"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "sort parameter is preserved in URL", %{conn: conn} do
|
||||||
|
{:ok, view, _html} = live(conn, ~p"/collections/all?sort=price_asc")
|
||||||
|
|
||||||
|
html = render(view)
|
||||||
|
|
||||||
|
assert html =~ "selected"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "default sort is featured", %{conn: conn} do
|
||||||
|
{:ok, _view, html} = live(conn, ~p"/collections/all")
|
||||||
|
|
||||||
|
assert html =~ ~r/<option[^>]*value="featured"[^>]*selected/
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe "Navigation" do
|
||||||
|
test "category links preserve sort order", %{conn: conn} do
|
||||||
|
{:ok, _view, html} = live(conn, ~p"/collections/all?sort=price_desc")
|
||||||
|
|
||||||
|
category = List.first(PreviewData.categories())
|
||||||
|
assert html =~ "/collections/#{category.slug}?sort=price_desc"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "All button link preserves sort order", %{conn: conn} do
|
||||||
|
{:ok, _view, html} = live(conn, ~p"/collections/art-prints?sort=name_asc")
|
||||||
|
|
||||||
|
assert html =~ "/collections/all?sort=name_asc"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "featured sort does not include query param in links", %{conn: conn} do
|
||||||
|
{:ok, _view, html} = live(conn, ~p"/collections/all")
|
||||||
|
|
||||||
|
category = List.first(PreviewData.categories())
|
||||||
|
assert html =~ ~s(href="/collections/#{category.slug}")
|
||||||
|
refute html =~ "/collections/#{category.slug}?sort=featured"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
Loading…
Reference in New Issue
Block a user