mirror of
https://github.com/philomena-dev/philomena.git
synced 2025-01-19 22:27:59 +01:00
226 lines
6.6 KiB
Elixir
226 lines
6.6 KiB
Elixir
defmodule PhilomenaWeb.GalleryController do
|
|
use PhilomenaWeb, :controller
|
|
|
|
alias PhilomenaWeb.ImageLoader
|
|
alias PhilomenaWeb.NotificationCountPlug
|
|
alias Philomena.Elasticsearch
|
|
alias Philomena.Interactions
|
|
alias Philomena.Galleries.Gallery
|
|
alias Philomena.Galleries
|
|
alias Philomena.Images.Image
|
|
import Ecto.Query
|
|
|
|
plug PhilomenaWeb.FilterBannedUsersPlug when action in [:new, :create, :edit, :update, :delete]
|
|
plug PhilomenaWeb.MapParameterPlug, [param: "gallery"] when action in [:index]
|
|
|
|
plug :load_and_authorize_resource,
|
|
model: Gallery,
|
|
except: [:index],
|
|
preload: [:creator, thumbnail: [tags: :aliases]]
|
|
|
|
def index(conn, params) do
|
|
galleries =
|
|
Gallery
|
|
|> Elasticsearch.search_definition(
|
|
%{
|
|
query: %{
|
|
bool: %{
|
|
must: parse_search(params)
|
|
}
|
|
},
|
|
sort: parse_sort(params)
|
|
},
|
|
conn.assigns.pagination
|
|
)
|
|
|> Elasticsearch.search_records(preload(Gallery, [:creator, thumbnail: [tags: :aliases]]))
|
|
|
|
render(conn, "index.html",
|
|
title: "Galleries",
|
|
galleries: galleries,
|
|
layout_class: "layout--wide"
|
|
)
|
|
end
|
|
|
|
def show(conn, _params) do
|
|
gallery = conn.assigns.gallery
|
|
user = conn.assigns.current_user
|
|
query = "gallery_id:#{gallery.id}"
|
|
|
|
conn =
|
|
update_in(
|
|
conn.params,
|
|
&Map.merge(&1, %{
|
|
"q" => query,
|
|
"sf" => "gallery_id:#{gallery.id}",
|
|
"sd" => position_order(gallery)
|
|
})
|
|
)
|
|
|
|
{:ok, {images, _tags}} = ImageLoader.search_string(conn, query)
|
|
{gallery_prev, gallery_next} = prev_next_page_images(conn, query)
|
|
|
|
[images, gallery_prev, gallery_next] =
|
|
Elasticsearch.msearch_records_with_hits(
|
|
[images, gallery_prev, gallery_next],
|
|
[
|
|
preload(Image, tags: :aliases),
|
|
preload(Image, tags: :aliases),
|
|
preload(Image, tags: :aliases)
|
|
]
|
|
)
|
|
|
|
interactions = Interactions.user_interactions([images, gallery_prev, gallery_next], user)
|
|
|
|
watching = Galleries.subscribed?(gallery, user)
|
|
|
|
gallery_images =
|
|
Enum.to_list(gallery_prev) ++ Enum.to_list(images) ++ Enum.to_list(gallery_next)
|
|
|
|
gallery_json = Jason.encode!(Enum.map(gallery_images, &elem(&1, 0).id))
|
|
|
|
Galleries.clear_notification(gallery, user)
|
|
|
|
conn
|
|
|> NotificationCountPlug.call([])
|
|
|> assign(:clientside_data, gallery_images: gallery_json)
|
|
|> render("show.html",
|
|
title: "Showing Gallery",
|
|
layout_class: "layout--wide",
|
|
watching: watching,
|
|
gallery: gallery,
|
|
gallery_prev: Enum.any?(gallery_prev),
|
|
gallery_next: Enum.any?(gallery_next),
|
|
gallery_images: gallery_images,
|
|
images: images,
|
|
interactions: interactions
|
|
)
|
|
end
|
|
|
|
def new(conn, _params) do
|
|
changeset = Galleries.change_gallery(%Gallery{})
|
|
render(conn, "new.html", title: "New Gallery", changeset: changeset)
|
|
end
|
|
|
|
def create(conn, %{"gallery" => gallery_params}) do
|
|
user = conn.assigns.current_user
|
|
|
|
case Galleries.create_gallery(user, gallery_params) do
|
|
{:ok, gallery} ->
|
|
conn
|
|
|> put_flash(:info, "Gallery successfully created.")
|
|
|> redirect(to: Routes.gallery_path(conn, :show, gallery))
|
|
|
|
{:error, changeset} ->
|
|
conn
|
|
|> render("new.html", changeset: changeset)
|
|
end
|
|
end
|
|
|
|
def edit(conn, _params) do
|
|
gallery = conn.assigns.gallery
|
|
changeset = Galleries.change_gallery(gallery)
|
|
|
|
render(conn, "edit.html", title: "Editing Gallery", gallery: gallery, changeset: changeset)
|
|
end
|
|
|
|
def update(conn, %{"gallery" => gallery_params}) do
|
|
gallery = conn.assigns.gallery
|
|
|
|
case Galleries.update_gallery(gallery, gallery_params) do
|
|
{:ok, gallery} ->
|
|
conn
|
|
|> put_flash(:info, "Gallery successfully updated.")
|
|
|> redirect(to: Routes.gallery_path(conn, :show, gallery))
|
|
|
|
{:error, changeset} ->
|
|
conn
|
|
|> render("edit.html", gallery: gallery, changeset: changeset)
|
|
end
|
|
end
|
|
|
|
def delete(conn, _params) do
|
|
gallery = conn.assigns.gallery
|
|
|
|
{:ok, _gallery} = Galleries.delete_gallery(gallery)
|
|
|
|
conn
|
|
|> put_flash(:info, "Gallery successfully destroyed.")
|
|
|> redirect(to: Routes.gallery_path(conn, :index))
|
|
end
|
|
|
|
defp prev_next_page_images(conn, query) do
|
|
limit = conn.assigns.image_pagination.page_size
|
|
offset = (conn.assigns.image_pagination.page_number - 1) * limit
|
|
|
|
# Inconsistency: Elasticsearch doesn't allow requesting offsets which are less than 0,
|
|
# but it does allow requesting offsets which are beyond the total number of results.
|
|
|
|
prev_image = gallery_image(offset - 1, conn, query)
|
|
next_image = gallery_image(offset + limit, conn, query)
|
|
|
|
{prev_image, next_image}
|
|
end
|
|
|
|
defp gallery_image(offset, _conn, _query) when offset < 0 do
|
|
Elasticsearch.search_definition(Image, %{query: %{match_none: %{}}})
|
|
end
|
|
|
|
defp gallery_image(offset, conn, query) do
|
|
pagination_params = %{page_number: offset + 1, page_size: 1}
|
|
|
|
{:ok, {image, _tags}} = ImageLoader.search_string(conn, query, pagination: pagination_params)
|
|
|
|
image
|
|
end
|
|
|
|
defp parse_search(%{"gallery" => gallery_params}) do
|
|
parse_title(gallery_params) ++
|
|
parse_creator(gallery_params) ++
|
|
parse_included_image(gallery_params) ++
|
|
parse_description(gallery_params)
|
|
end
|
|
|
|
defp parse_search(_params), do: [%{match_all: %{}}]
|
|
|
|
defp parse_title(%{"title" => title}) when is_binary(title) and title not in [nil, ""],
|
|
do: [%{wildcard: %{title: "*" <> String.downcase(title) <> "*"}}]
|
|
|
|
defp parse_title(_params), do: []
|
|
|
|
defp parse_creator(%{"creator" => creator})
|
|
when is_binary(creator) and creator not in [nil, ""],
|
|
do: [%{term: %{creator: String.downcase(creator)}}]
|
|
|
|
defp parse_creator(_params), do: []
|
|
|
|
defp parse_included_image(%{"include_image" => image_id})
|
|
when is_binary(image_id) and image_id not in [nil, ""] do
|
|
with {image_id, _rest} <- Integer.parse(image_id) do
|
|
[%{term: %{image_ids: image_id}}]
|
|
else
|
|
_ ->
|
|
[]
|
|
end
|
|
end
|
|
|
|
defp parse_included_image(_params), do: []
|
|
|
|
defp parse_description(%{"description" => description})
|
|
when is_binary(description) and description not in [nil, ""],
|
|
do: [%{match_phrase: %{description: description}}]
|
|
|
|
defp parse_description(_params), do: []
|
|
|
|
defp parse_sort(%{"gallery" => %{"sf" => sf, "sd" => sd}})
|
|
when sf in ["created_at", "updated_at", "image_count", "_score"] and
|
|
sd in ["desc", "asc"] do
|
|
%{sf => sd}
|
|
end
|
|
|
|
defp parse_sort(_params) do
|
|
%{created_at: :desc}
|
|
end
|
|
|
|
defp position_order(%{order_position_asc: true}), do: "asc"
|
|
defp position_order(_gallery), do: "desc"
|
|
end
|