philomena/lib/philomena_web/views/app_view.ex

256 lines
7 KiB
Elixir
Raw Permalink Normal View History

2019-08-18 20:14:36 +02:00
defmodule PhilomenaWeb.AppView do
2019-10-01 03:21:48 +02:00
use Phoenix.HTML
2019-08-18 20:14:36 +02:00
@time_strings %{
seconds: "less than a minute",
minute: "about a minute",
minutes: "%d minutes",
hour: "about an hour",
hours: "about %d hours",
day: "a day",
days: "%d days",
month: "about a month",
months: "%d months",
year: "about a year",
years: "%d years"
}
2019-11-27 06:51:20 +01:00
@months %{
1 => "January",
2 => "February",
3 => "March",
4 => "April",
5 => "May",
6 => "June",
7 => "July",
8 => "August",
9 => "September",
10 => "October",
11 => "November",
12 => "December"
}
2019-08-18 20:14:36 +02:00
def pretty_time(time) do
now = DateTime.utc_now()
seconds = DateTime.diff(now, time, :second)
2019-08-18 20:14:36 +02:00
relation = if seconds < 0, do: "from now", else: "ago"
2019-12-08 21:32:54 +01:00
words = distance_of_time_in_words(now, time)
2019-08-18 20:14:36 +02:00
2020-01-11 05:20:19 +01:00
content_tag(:time, "#{words} #{relation}",
datetime: DateTime.to_iso8601(time),
title: datetime_string(time)
)
2019-08-18 20:14:36 +02:00
end
2019-10-01 03:21:48 +02:00
def tag_list(image) do
Philomena.Images.tag_list(image)
end
2019-12-08 21:32:54 +01:00
def distance_of_time_in_words(time_2, time_1) do
seconds = abs(DateTime.diff(time_2, time_1, :second))
2019-12-08 21:32:54 +01:00
minutes = div(seconds, 60)
hours = div(minutes, 60)
days = div(hours, 24)
months = div(days, 30)
years = div(days, 365)
cond do
seconds < 45 -> String.replace(@time_strings[:seconds], "%d", to_string(seconds))
seconds < 90 -> String.replace(@time_strings[:minute], "%d", to_string(1))
minutes < 45 -> String.replace(@time_strings[:minutes], "%d", to_string(minutes))
minutes < 90 -> String.replace(@time_strings[:hour], "%d", to_string(1))
hours < 24 -> String.replace(@time_strings[:hours], "%d", to_string(hours))
hours < 42 -> String.replace(@time_strings[:day], "%d", to_string(1))
days < 30 -> String.replace(@time_strings[:days], "%d", to_string(days))
days < 45 -> String.replace(@time_strings[:month], "%d", to_string(1))
days < 365 -> String.replace(@time_strings[:months], "%d", to_string(months))
days < 548 -> String.replace(@time_strings[:year], "%d", to_string(1))
true -> String.replace(@time_strings[:years], "%d", to_string(years))
end
end
2019-10-01 03:21:48 +02:00
def can?(conn, action, model) do
Canada.Can.can?(conn.assigns.current_user, action, model)
end
2019-10-09 17:51:14 +02:00
2019-12-08 16:14:19 +01:00
def map_join(enumerable, joiner, map_fn) do
enumerable
|> Enum.map(map_fn)
|> Enum.intersperse(joiner)
end
2019-12-05 14:55:49 +01:00
def number_with_delimiter(nil), do: "0"
2020-01-11 05:20:19 +01:00
2019-10-09 17:51:14 +02:00
def number_with_delimiter(number) do
number
|> to_charlist()
|> Enum.reverse()
|> Enum.chunk_every(3)
|> Enum.map(&Enum.reverse(&1))
|> Enum.reverse()
|> Enum.join(",")
end
def pluralize(singular, plural, count) do
if count == 1 do
singular
else
plural
end
end
2019-11-15 16:15:21 +01:00
def button_to(text, route, args \\ []) do
2019-10-09 17:51:14 +02:00
method = Keyword.get(args, :method, "get")
class = Keyword.get(args, :class, nil)
data = Keyword.get(args, :data, [])
form_for(nil, route, [method: method, class: "button_to"], fn _f ->
2020-01-11 05:20:19 +01:00
submit(text, class: class, data: data)
2019-10-09 17:51:14 +02:00
end)
end
2019-11-27 06:51:20 +01:00
2019-12-09 05:41:35 +01:00
def escape_nl2br(nil), do: nil
2020-01-11 05:20:19 +01:00
2021-04-01 18:49:41 +02:00
# Everything is escaped before being sent to raw/1
# sobelow_skip ["XSS.Raw"]
2019-12-06 04:05:21 +01:00
def escape_nl2br(text) do
text
|> String.split("\n")
2024-06-25 05:23:43 +02:00
|> Enum.map_intersperse("<br />", &safe_to_string(html_escape(&1)))
2019-12-06 04:05:21 +01:00
|> raw()
end
2019-11-27 06:51:20 +01:00
defp datetime_string(time) do
:io_lib.format("~2..0B:~2..0B:~2..0B, ~s ~B, ~B", [
time.hour,
time.minute,
time.second,
@months[time.month],
time.day,
time.year
])
|> to_string()
end
2019-12-05 21:06:18 +01:00
2019-12-14 18:50:30 +01:00
def link_to_ip(_conn, nil), do: content_tag(:code, "null")
2020-01-11 05:20:19 +01:00
def link_to_ip(_conn, ip) do
link(to: "/ip_profiles/#{ip}") do
2019-12-10 17:29:54 +01:00
[
content_tag(:i, "", class: "fas fa-network-wired"),
" ",
2019-12-14 18:50:30 +01:00
to_string(ip)
2019-12-10 17:29:54 +01:00
]
end
2019-12-05 21:06:18 +01:00
end
2019-12-14 18:50:30 +01:00
def link_to_fingerprint(_conn, nil), do: content_tag(:code, "null")
2020-01-11 05:20:19 +01:00
def link_to_fingerprint(_conn, fp) do
link(to: "/fingerprint_profiles/#{fp}") do
2019-12-10 17:29:54 +01:00
[
content_tag(:i, "", class: "fas fa-desktop"),
" ",
2019-12-14 18:50:30 +01:00
String.slice(fp, 0..6)
2019-12-10 17:29:54 +01:00
]
end
2019-12-05 21:06:18 +01:00
end
def communication_body_class(%{destroyed_content: true}), do: "communication--destroyed"
def communication_body_class(_communication), do: nil
2022-03-22 22:23:30 +01:00
def can_view_communication?(conn, communication) do
user_id =
case conn.assigns.current_user do
nil -> -1
user -> user.id
end
cond do
can?(conn, :hide, communication) and not hide_staff_tools?(conn) -> true
communication.destroyed_content -> false
not communication.approved and communication.user_id != user_id -> false
true -> true
end
end
2019-12-20 23:35:46 +01:00
def hide_staff_tools?(conn),
do: conn.cookies["hide_staff_tools"] == "true"
def blank?(nil), do: true
def blank?(""), do: true
def blank?([]), do: true
def blank?(map) when is_map(map), do: map == %{}
def blank?(str) when is_binary(str), do: String.trim(str) == ""
def blank?(_object), do: false
def present?(object), do: not blank?(object)
2023-02-18 22:26:26 +01:00
# This and related functions are taken from
# https://github.com/adam12/phoenix_mtm
# Copied due to the package no longer being actively updated
# and screwing up our dependencies.
# Credit goes to the respective authors.
def collection_checkboxes(form, field, collection, opts \\ []) do
name = input_name(form, field) <> "[]"
selected = Keyword.get(opts, :selected, [])
input_opts = Keyword.get(opts, :input_opts, [])
label_opts = Keyword.get(opts, :label_opts, [])
mapper = Keyword.get(opts, :mapper, &mapper_unwrapped/6)
2023-02-19 17:26:09 +01:00
wrapper = Keyword.get(opts, :wrapper, & &1)
2023-02-18 22:26:26 +01:00
2023-02-19 17:26:09 +01:00
inputs =
Enum.map(collection, fn {label_content, value} ->
id = input_id(form, field) <> "_#{value}"
2023-02-18 22:26:26 +01:00
2023-02-19 17:26:09 +01:00
input_opts =
input_opts
|> Keyword.put(:type, "checkbox")
|> Keyword.put(:id, id)
|> Keyword.put(:name, name)
|> Keyword.put(:value, "#{value}")
|> put_selected(selected, value)
2023-02-18 22:26:26 +01:00
2023-02-19 17:26:09 +01:00
label_opts = label_opts ++ [for: id]
2023-02-18 22:26:26 +01:00
2023-02-19 17:26:09 +01:00
mapper.(form, field, input_opts, label_content, label_opts, opts)
|> wrapper.()
end)
2023-02-18 22:26:26 +01:00
html_escape(
inputs ++
2023-02-19 17:26:09 +01:00
hidden_input(form, field, name: name, value: "")
2023-02-18 22:26:26 +01:00
)
end
defp put_selected(opts, selected, value) do
if Enum.member?(selected, value) do
Keyword.put(opts, :checked, true)
else
opts
end
end
defp mapper_unwrapped(form, field, input_opts, label_content, label_opts, _opts) do
[
tag(:input, input_opts),
label(form, field, "#{label_content}", label_opts)
]
end
2023-05-18 16:23:17 +02:00
def image_has_sources(image), do: Enum.count(image.sources) > 0
def image_first_source(image) do
if image_has_sources(image), do: Enum.at(image.sources, 0).source, else: ""
end
2023-05-18 16:23:17 +02:00
def get_flash(%{assigns: %{flash: nil}}), do: %{}
def get_flash(%{assigns: %{flash: flash}}), do: flash
def get_flash(_), do: %{}
def get_flash(%{assigns: %{flash: nil}}, _key), do: %{}
def get_flash(%{assigns: %{flash: flash}}, key), do: Phoenix.Flash.get(flash, key)
def get_flash(_, _key), do: %{}
2019-08-19 03:43:06 +02:00
end