5b4e6ed473
Fixes: #1096
123 lines
3.1 KiB
Elixir
123 lines
3.1 KiB
Elixir
defmodule Mobilizon.Addresses.Address do
|
|
@moduledoc """
|
|
Represents an address for an event or a group.
|
|
"""
|
|
|
|
use Ecto.Schema
|
|
|
|
import Ecto.Changeset
|
|
|
|
alias Mobilizon.Events.Event
|
|
|
|
alias Mobilizon.Web.Endpoint
|
|
|
|
@type t :: %__MODULE__{
|
|
country: String.t() | nil,
|
|
locality: String.t() | nil,
|
|
region: String.t() | nil,
|
|
description: String.t() | nil,
|
|
geom: Geo.Point.t() | nil,
|
|
postal_code: String.t() | nil,
|
|
street: String.t() | nil,
|
|
type: String.t() | nil,
|
|
url: String.t(),
|
|
origin_id: String.t() | nil,
|
|
events: [Event.t()],
|
|
timezone: String.t() | nil
|
|
}
|
|
|
|
@required_attrs [:url]
|
|
@optional_attrs [
|
|
:description,
|
|
:geom,
|
|
:country,
|
|
:locality,
|
|
:region,
|
|
:postal_code,
|
|
:street,
|
|
:origin_id,
|
|
:type,
|
|
:timezone
|
|
]
|
|
@attrs @required_attrs ++ @optional_attrs
|
|
|
|
schema "addresses" do
|
|
field(:country, :string)
|
|
field(:locality, :string)
|
|
field(:region, :string)
|
|
field(:description, :string)
|
|
field(:geom, Geo.PostGIS.Geometry)
|
|
field(:postal_code, :string)
|
|
field(:street, :string)
|
|
field(:type, :string)
|
|
field(:url, :string)
|
|
field(:origin_id, :string)
|
|
field(:timezone, :string)
|
|
|
|
has_many(:events, Event, foreign_key: :physical_address_id)
|
|
|
|
timestamps()
|
|
end
|
|
|
|
@doc false
|
|
@spec changeset(t | Ecto.Schema.t(), map) :: Ecto.Changeset.t()
|
|
def changeset(%__MODULE__{} = address, attrs) do
|
|
address
|
|
|> cast(attrs, @attrs)
|
|
|> maybe_set_timezone()
|
|
|> set_url()
|
|
|> validate_required(@required_attrs)
|
|
|> unique_constraint(:url, name: :addresses_url_index)
|
|
end
|
|
|
|
@spec set_url(Ecto.Changeset.t()) :: Ecto.Changeset.t()
|
|
defp set_url(%Ecto.Changeset{changes: changes} = changeset) do
|
|
url = Map.get(changes, :url, "#{Endpoint.url()}/address/#{Ecto.UUID.generate()}")
|
|
|
|
put_change(changeset, :url, url)
|
|
end
|
|
|
|
@spec coords(nil | t) :: nil | {float, float}
|
|
def coords(nil), do: nil
|
|
|
|
def coords(%__MODULE__{} = address) do
|
|
with %Geo.Point{coordinates: {longitude, latitude}, srid: 4326} <- address.geom do
|
|
{latitude, longitude}
|
|
end
|
|
end
|
|
|
|
@spec representation(nil | t) :: nil | String.t()
|
|
def representation(nil), do: nil
|
|
|
|
def representation(%__MODULE__{} = address) do
|
|
String.trim(
|
|
"#{address.description} #{address.street} #{address.postal_code} #{address.locality} #{address.region} #{address.country}"
|
|
)
|
|
end
|
|
|
|
@spec maybe_set_timezone(Ecto.Changeset.t()) :: Ecto.Changeset.t()
|
|
defp maybe_set_timezone(%Ecto.Changeset{} = changeset) do
|
|
case get_change(changeset, :geom) do
|
|
nil ->
|
|
changeset
|
|
|
|
geom ->
|
|
case get_field(changeset, :timezone) do
|
|
# Only update the timezone if the geom has change and we don't already have a set timezone
|
|
nil -> put_change(changeset, :timezone, timezone(geom))
|
|
_ -> changeset
|
|
end
|
|
end
|
|
end
|
|
|
|
@spec timezone(Geo.PostGIS.Geometry.t() | nil) :: String.t() | nil
|
|
defp timezone(nil), do: nil
|
|
|
|
defp timezone(geom) do
|
|
case TzWorld.timezone_at(geom) do
|
|
{:ok, tz} -> tz
|
|
{:error, _err} -> nil
|
|
end
|
|
end
|
|
end
|