This repository has been archived on 2024-06-09. You can view files and clone it, but cannot push or open issues or pull requests.
linkify/lib/auto_linker/builder.ex

192 lines
4.8 KiB
Elixir
Raw Normal View History

2017-03-29 20:27:13 +00:00
defmodule AutoLinker.Builder do
@moduledoc """
Module for building the auto generated link.
"""
@doc """
Create a link.
"""
2019-02-23 13:09:43 +00:00
def create_link(text, opts) do
url = add_scheme(text)
2019-02-22 10:41:23 +00:00
2017-03-29 20:27:13 +00:00
[]
|> build_attrs(url, opts, :rel)
|> build_attrs(url, opts, :target)
|> build_attrs(url, opts, :class)
2019-02-22 10:41:23 +00:00
|> build_attrs(url, opts, :href)
2019-02-23 13:09:43 +00:00
|> format_url(text, opts)
2017-03-29 20:27:13 +00:00
end
2018-01-23 00:46:47 +00:00
def create_markdown_links(text, opts) do
[]
|> build_attrs(text, opts, :rel)
|> build_attrs(text, opts, :target)
|> build_attrs(text, opts, :class)
|> format_markdown(text, opts)
end
2019-02-22 10:41:23 +00:00
defp build_attrs(attrs, uri, %{rel: get_rel}, :rel) when is_function(get_rel, 1) do
case get_rel.(uri) do
nil -> attrs
rel -> [{:rel, rel} | attrs]
end
end
2017-03-29 20:27:13 +00:00
defp build_attrs(attrs, _, opts, :rel) do
2019-02-05 11:22:51 +00:00
if rel = Map.get(opts, :rel, "noopener noreferrer"), do: [{:rel, rel} | attrs], else: attrs
2017-03-29 20:27:13 +00:00
end
2019-02-05 11:22:51 +00:00
2017-03-29 20:27:13 +00:00
defp build_attrs(attrs, _, opts, :target) do
2019-02-05 11:22:51 +00:00
if Map.get(opts, :new_window, true), do: [{:target, :_blank} | attrs], else: attrs
2017-03-29 20:27:13 +00:00
end
2019-02-05 11:22:51 +00:00
2017-03-29 20:27:13 +00:00
defp build_attrs(attrs, _, opts, :class) do
2019-02-05 11:22:51 +00:00
if cls = Map.get(opts, :class, "auto-linker"), do: [{:class, cls} | attrs], else: attrs
2017-03-29 20:27:13 +00:00
end
2019-02-05 11:22:51 +00:00
2019-02-22 10:41:23 +00:00
defp build_attrs(attrs, url, _opts, :href) do
[{:href, url} | attrs]
2017-03-29 20:27:13 +00:00
end
2019-02-22 10:41:23 +00:00
defp add_scheme("http://" <> _ = url), do: url
defp add_scheme("https://" <> _ = url), do: url
defp add_scheme(url), do: "http://" <> url
2017-03-29 20:27:13 +00:00
defp format_url(attrs, url, opts) do
url =
url
|> strip_prefix(Map.get(opts, :strip_prefix, true))
|> truncate(Map.get(opts, :truncate, false))
2019-02-05 11:22:51 +00:00
2018-01-23 00:46:47 +00:00
attrs = format_attrs(attrs)
2019-02-23 13:19:46 +00:00
"<a #{attrs}>#{url}</a>"
2017-03-29 20:27:13 +00:00
end
2018-01-23 00:46:47 +00:00
defp format_attrs(attrs) do
attrs
2019-02-18 11:34:38 +00:00
|> Enum.map(fn {key, value} -> ~s(#{key}="#{value}") end)
2018-01-23 00:46:47 +00:00
|> Enum.join(" ")
end
defp format_markdown(attrs, text, _opts) do
attrs =
case format_attrs(attrs) do
"" -> ""
attrs -> " " <> attrs
end
2019-02-05 11:22:51 +00:00
2018-01-23 00:46:47 +00:00
Regex.replace(~r/\[(.+?)\]\((.+?)\)/, text, "<a href='\\2'#{attrs}>\\1</a>")
end
2017-03-29 20:27:13 +00:00
defp truncate(url, false), do: url
defp truncate(url, len) when len < 3, do: url
2019-02-05 11:22:51 +00:00
2017-03-29 20:27:13 +00:00
defp truncate(url, len) do
if String.length(url) > len, do: String.slice(url, 0, len - 2) <> "..", else: url
end
defp strip_prefix(url, true) do
url
|> String.replace(~r/^https?:\/\//, "")
|> String.replace(~r/^www\./, "")
end
2019-02-05 11:22:51 +00:00
2017-03-29 20:27:13 +00:00
defp strip_prefix(url, _), do: url
2017-11-17 18:36:37 +00:00
2019-02-23 13:19:46 +00:00
def create_phone_link([], buffer, _), do: buffer
2019-02-05 11:22:51 +00:00
2017-11-17 18:36:37 +00:00
def create_phone_link([h | t], buffer, opts) do
2019-02-05 11:22:51 +00:00
create_phone_link(t, format_phone_link(h, buffer, opts), opts)
2017-11-17 18:36:37 +00:00
end
def format_phone_link([h | _], buffer, opts) do
val =
h
|> String.replace(~r/[\.\+\- x\(\)]+/, "")
|> format_phone_link(h, opts)
2019-02-05 11:22:51 +00:00
2017-11-17 18:36:37 +00:00
# val = ~s'<a href="#" class="phone-number" data-phone="#{number}">#{h}</a>'
String.replace(buffer, h, val)
end
def format_phone_link(number, original, opts) do
tag = opts[:tag] || "a"
class = opts[:class] || "phone-number"
data_phone = opts[:data_phone] || "data-phone"
attrs = format_attributes(opts[:attributes] || [])
href = opts[:href] || "#"
2019-02-05 11:22:51 +00:00
~s'<#{tag} href="#{href}" class="#{class}" #{data_phone}="#{number}"#{attrs}>#{original}</#{
tag
}>'
2017-11-17 18:36:37 +00:00
end
def create_mention_link("@" <> name, _buffer, opts) do
mention_prefix = opts[:mention_prefix]
url = mention_prefix <> name
2019-02-22 10:41:23 +00:00
[]
|> build_attrs(url, opts, :rel)
|> build_attrs(url, opts, :target)
|> build_attrs(url, opts, :class)
2019-02-22 10:41:23 +00:00
|> build_attrs(url, opts, :href)
|> format_mention(name, opts)
end
2019-02-18 11:39:56 +00:00
def create_hashtag_link("#" <> tag, _buffer, opts) do
hashtag_prefix = opts[:hashtag_prefix]
url = hashtag_prefix <> tag
2019-02-22 10:41:23 +00:00
[]
|> build_attrs(url, opts, :rel)
|> build_attrs(url, opts, :target)
|> build_attrs(url, opts, :class)
2019-02-22 10:41:23 +00:00
|> build_attrs(url, opts, :href)
|> format_hashtag(tag, opts)
end
def create_email_link(email, opts) do
[]
|> build_attrs(email, opts, :class)
2019-02-22 10:41:23 +00:00
|> build_attrs("mailto:#{email}", opts, :href)
|> format_email(email, opts)
end
def create_extra_link(uri, opts) do
[]
|> build_attrs(uri, opts, :class)
2019-02-22 10:41:23 +00:00
|> build_attrs(uri, opts, :rel)
|> build_attrs(uri, opts, :target)
|> build_attrs(uri, opts, :href)
|> format_extra(uri, opts)
end
def format_mention(attrs, name, _opts) do
attrs = format_attrs(attrs)
2019-02-22 10:41:23 +00:00
"<a #{attrs}>@#{name}</a>"
end
def format_hashtag(attrs, tag, _opts) do
attrs = format_attrs(attrs)
2019-02-22 10:41:23 +00:00
"<a #{attrs}>##{tag}</a>"
end
def format_email(attrs, email, _opts) do
attrs = format_attrs(attrs)
2019-02-22 10:41:23 +00:00
~s(<a #{attrs}>#{email}</a>)
end
def format_extra(attrs, uri, _opts) do
2019-02-22 10:41:23 +00:00
attrs = format_attrs(attrs)
~s(<a #{attrs}>#{uri}</a>)
end
2017-11-17 18:36:37 +00:00
defp format_attributes(attrs) do
Enum.reduce(attrs, "", fn {name, value}, acc ->
acc <> ~s' #{name}="#{value}"'
end)
end
2017-03-29 20:27:13 +00:00
end