Return to generating the icons from the filesystem
This commit is contained in:
80
lib/heroicons/generator.ex
Normal file
80
lib/heroicons/generator.ex
Normal file
@ -0,0 +1,80 @@
|
||||
defmodule Heroicons.Generator do
|
||||
defmacro __using__(icon_dir: icon_dir) do
|
||||
icon_paths =
|
||||
Path.absname(icon_dir, :code.priv_dir(:heroicons))
|
||||
|> Path.join("*.svg")
|
||||
|> Path.wildcard()
|
||||
|
||||
require Phoenix.Component
|
||||
|
||||
if function_exported?(Phoenix.Component, :assigns_to_attributes, 2) do
|
||||
Module.put_attribute(__CALLER__.module, :assign_mod, Phoenix.Component)
|
||||
Module.put_attribute(__CALLER__.module, :assigns_to_attrs_mod, Phoenix.Component)
|
||||
else
|
||||
Module.put_attribute(__CALLER__.module, :assign_mod, Phoenix.LiveView)
|
||||
Module.put_attribute(__CALLER__.module, :assigns_to_attrs_mod, Phoenix.LiveView.Helpers)
|
||||
end
|
||||
|
||||
for path <- icon_paths do
|
||||
generate(path)
|
||||
end
|
||||
end
|
||||
|
||||
@doc false
|
||||
def generate(path) do
|
||||
name =
|
||||
Path.basename(path, ".svg")
|
||||
|> String.replace("-", "_")
|
||||
|> String.to_atom()
|
||||
|
||||
doc = """
|
||||
)}) {: width=24px}
|
||||
## Examples
|
||||
Use as a `Phoenix.Component`
|
||||
<.#{name} />
|
||||
<.#{name} class="h-6 w-6 text-gray-500" />
|
||||
or as a function
|
||||
<%= #{name}() %>
|
||||
<%= #{name}(class: "h-6 w-6 text-gray-500") %>
|
||||
"""
|
||||
|
||||
quote do
|
||||
@doc unquote(doc)
|
||||
def unquote(name)(assigns_or_opts \\ [])
|
||||
|
||||
def unquote(name)(var!(assigns)) when is_map(var!(assigns)) do
|
||||
var!(attrs) = @assigns_to_attrs_mod.assigns_to_attributes(var!(assigns))
|
||||
var!(assigns) = @assign_mod.assign(var!(assigns), :attrs, var!(attrs))
|
||||
|
||||
EEx.compile_string("<svg {@attrs}" <> Heroicons.IconCache.icon_body(unquote(path)),
|
||||
engine: Phoenix.LiveView.HTMLEngine,
|
||||
file: __ENV__.file,
|
||||
line: __ENV__.line + 1,
|
||||
module: __ENV__.module,
|
||||
indentation: 0,
|
||||
assigns: var!(assigns)
|
||||
)
|
||||
end
|
||||
|
||||
def unquote(name)(opts) when is_list(opts) do
|
||||
attrs =
|
||||
for {k, v} <- opts do
|
||||
safe_k =
|
||||
k |> Atom.to_string() |> String.replace("_", "-") |> Phoenix.HTML.Safe.to_iodata()
|
||||
|
||||
safe_v = v |> Phoenix.HTML.Safe.to_iodata()
|
||||
|
||||
{:safe, [?\s, safe_k, ?=, ?", safe_v, ?"]}
|
||||
end
|
||||
|
||||
{:safe,
|
||||
[
|
||||
"<svg",
|
||||
Phoenix.HTML.Safe.to_iodata(attrs),
|
||||
" ",
|
||||
Heroicons.IconCache.icon_body(unquote(path))
|
||||
]}
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
12
lib/heroicons/icon_cache.ex
Normal file
12
lib/heroicons/icon_cache.ex
Normal file
@ -0,0 +1,12 @@
|
||||
defmodule Heroicons.IconCache do
|
||||
|
||||
@doc "Get's an icon's body from the filesystem"
|
||||
# TODO implement ETS-based caching & benchmark
|
||||
def icon_body(path) do
|
||||
icon = File.read!(path)
|
||||
|
||||
<<"<svg ", body::binary>> = icon
|
||||
|
||||
body
|
||||
end
|
||||
end
|
12387
lib/heroicons/mini.ex
12387
lib/heroicons/mini.ex
File diff suppressed because it is too large
Load Diff
12341
lib/heroicons/outline.ex
12341
lib/heroicons/outline.ex
File diff suppressed because it is too large
Load Diff
12404
lib/heroicons/solid.ex
12404
lib/heroicons/solid.ex
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user