Enable passing content or attrs as variables

Before, the guards in the macro definitions would only work for content
or attributes that were passed in as literals.
This commit is contained in:
Mitchell Hanberg 2019-05-08 03:31:13 -04:00
parent 6c6bd8dd3b
commit d81029c9be
2 changed files with 141 additions and 25 deletions

View file

@ -46,52 +46,68 @@ defmodule Dsl.Html do
el = unquote(el) el = unquote(el)
quote do quote do
unquote(el)([], nil) put_open_tag(var!(buff, Dsl.Html), unquote(el), [])
put_close_tag(var!(buff, Dsl.Html), unquote(el))
end end
end end
defmacro unquote(el)(attrs) when is_list(attrs) do defmacro unquote(el)([{:do, inner} | attrs]) do
el = unquote(el)
{inner, attrs} = Keyword.pop(attrs, :do, nil)
quote do
unquote(el)(unquote(attrs), unquote(inner))
end
end
defmacro unquote(el)(content) when is_binary(content) do
el = unquote(el) el = unquote(el)
quote do quote do
unquote(el)(unquote(content), []) put_open_tag(var!(buff, Dsl.Html), unquote(el), unquote(attrs))
_ = unquote(inner)
put_close_tag(var!(buff, Dsl.Html), unquote(el))
end end
end end
defmacro unquote(el)(content, attrs) when not is_list(content) and is_list(attrs) do defmacro unquote(el)(attrs_or_content) do
el = unquote(el) el = unquote(el)
text = {:text, [], [content]}
quote do quote do
unquote(el)(unquote(attrs), unquote(text)) put_open_tag(var!(buff, Dsl.Html), unquote(el), unquote(attrs_or_content))
put_close_tag(var!(buff, Dsl.Html), unquote(el))
end end
end end
defmacro unquote(el)(attrs, inner) when is_list(attrs) do defmacro unquote(el)(attrs, [{:do, inner}]) do
el = unquote(el) el = unquote(el)
quote do quote do
attrs = unquote(attrs) attrs = unquote(attrs)
put_buffer(var!(buff, Dsl.Html), "<#{unquote(el)}#{compile_attrs(attrs)}>") put_open_tag(var!(buff, Dsl.Html), unquote(el), attrs)
unquote(inner) _ = unquote(inner)
put_buffer(var!(buff, Dsl.Html), "</#{unquote(el)}>") put_close_tag(var!(buff, Dsl.Html), unquote(el))
end end
end end
defmacro unquote(el)(content, attrs) do
el = unquote(el)
quote do
attrs = unquote(attrs)
put_open_tag(var!(buff, Dsl.Html), unquote(el), attrs)
text unquote(content)
put_close_tag(var!(buff, Dsl.Html), unquote(el))
end
end
end
def put_open_tag(buff, el, attrs) when is_list(attrs) do
put_buffer(buff, "<#{el}#{compile_attrs(attrs)}>")
end
def put_open_tag(buff, el, content) when is_binary(content) do
put_buffer(buff, "<#{el}>")
put_buffer(buff, content)
end
def put_close_tag(buff, el) do
put_buffer(buff, "</#{el}>")
end end
for el <- @void_elements do for el <- @void_elements do
defmacro unquote(el)(attrs \\ []) defmacro unquote(el)(attrs \\ []) do
defmacro unquote(el)(attrs) do
el = unquote(el) el = unquote(el)
quote do quote do
@ -142,9 +158,7 @@ defmodule Dsl.Html do
defmacro defcomponent(name, do: block) do defmacro defcomponent(name, do: block) do
quote do quote do
defmacro unquote(name)(props \\ []) defmacro unquote(name)(props \\ []) do
defmacro unquote(name)(props) do
outer = unquote(Macro.escape(block)) outer = unquote(Macro.escape(block))
name = unquote(name) name = unquote(name)

View file

@ -56,8 +56,19 @@ defmodule Dsl.HtmlTest do
end end
end end
end end
defcomponent :variable_as_prop do
div id: @bob
end end
defcomponent :variable_as_prop_with_block do
div id: @bob do
@children
end
end
end
describe "non-void elements" do describe "non-void elements" do
test "renders two divs" do test "renders two divs" do
{:safe, result} = {:safe, result} =
@ -79,6 +90,41 @@ defmodule Dsl.HtmlTest do
assert result == "<div></div><span></span>" assert result == "<div></div><span></span>"
end end
test "renders an el that taks attrs and a block" do
{:safe, result} =
htm do
div class: "bob" do
span()
span()
end
end
assert result == ~s{<div class="bob"><span></span><span></span></div>}
end
test "renders one els nested inside an el" do
{:safe, result} =
htm do
div do
span()
end
end
assert result == "<div><span></span></div>"
end
test "renders two els nested inside an el" do
{:safe, result} =
htm do
div do
span()
span()
end
end
assert result == "<div><span></span><span></span></div>"
end
test "renders two divs that are rendered by a loop" do test "renders two divs that are rendered by a loop" do
{:safe, result} = {:safe, result} =
htm do htm do
@ -122,6 +168,20 @@ defmodule Dsl.HtmlTest do
assert result == ~s{<div class="hello"><div class="hi"></div></div>} assert result == ~s{<div class="hello"><div class="hi"></div></div>}
end end
test "renders an attribute on a div passed as a variable" do
attrs1 = [class: "hello"]
attrs2 = [class: "hi"]
{:safe, result} =
htm do
div attrs1 do
div(attrs2)
end
end
assert result == ~s{<div class="hello"><div class="hi"></div></div>}
end
test "renders multiple attributes on a div without block" do test "renders multiple attributes on a div without block" do
{:safe, result} = {:safe, result} =
htm do htm do
@ -140,6 +200,18 @@ defmodule Dsl.HtmlTest do
assert result == ~s{<div>CONTENT</div><div class="hi">MORE</div>} assert result == ~s{<div>CONTENT</div><div class="hi">MORE</div>}
end end
test "can accept content as first argument passed as a variable" do
content = "CONTENT"
more = "MORE"
{:safe, result} =
htm do
div(content)
div(more, class: "hi")
end
assert result == ~s{<div>CONTENT</div><div class="hi">MORE</div>}
end
end end
describe "void elements" do describe "void elements" do
@ -286,6 +358,36 @@ defmodule Dsl.HtmlTest do
~s|<div>:atom</div><div>%{key: &quot;value&quot;}</div><div>{:status, :tuple}</div><div>&quot;string&quot;</div><div>1</div><div>[1, 2, 3]</div>| ~s|<div>:atom</div><div>%{key: &quot;value&quot;}</div><div>{:status, :tuple}</div><div>&quot;string&quot;</div><div>1</div><div>[1, 2, 3]</div>|
end end
test "can pass a variable as a prop" do
import Component
bob = "hi"
{:safe, result} =
htm do
variable_as_prop bob: bob
end
assert result ==
~s|<div id="hi"></div>|
end
test "can pass a variable as a prop to a component with a block" do
import Component
bob = "hi"
class = "joe"
{:safe, result} =
htm do
variable_as_prop_with_block bob: bob do
div()
end
end
assert result == ~s|<div id="hi"><div></div></div>|
end
test "can use string interpolation in props" do test "can use string interpolation in props" do
interop = "hi" interop = "hi"