Better whitespace handling and control (#145)
* Fine tune whitespace The EEx outut now emits more human-readable and predictable formatting. This includes proper indenting, at least for each "root" template. * Internal whitespace control You can now use a bang version of any nonvoid tag to emit the markup witout the internal whitespace. This means that there will not be a newline emitted after the opening tag and before the closing tag.
This commit is contained in:
parent
87ddbaa6b5
commit
c965048f40
32 changed files with 613 additions and 148 deletions
|
@ -16,7 +16,6 @@ locals_without_parens = ~w[
|
||||||
details summary menuitem menu
|
details summary menuitem menu
|
||||||
meta link base
|
meta link base
|
||||||
area br col embed hr img input keygen param source track wbr
|
area br col embed hr img input keygen param source track wbr
|
||||||
txt partial
|
|
||||||
|
|
||||||
animate animateMotion animateTransform circle clipPath
|
animate animateMotion animateTransform circle clipPath
|
||||||
color-profile defs desc discard ellipse feBlend
|
color-profile defs desc discard ellipse feBlend
|
||||||
|
@ -26,13 +25,7 @@ locals_without_parens = ~w[
|
||||||
marker mask mesh meshgradient meshpatch meshrow metadata mpath path pattern polygon
|
marker mask mesh meshgradient meshpatch meshrow metadata mpath path pattern polygon
|
||||||
polyline radialGradient rect set solidcolor stop svg switch symbol text
|
polyline radialGradient rect set solidcolor stop svg switch symbol text
|
||||||
textPath tspan unknown use view
|
textPath tspan unknown use view
|
||||||
|
]a |> Enum.flat_map(fn e -> [{e, :*}, {:"#{e}!", :*}] end)
|
||||||
form_for inputs_for
|
|
||||||
checkbox color_input checkbox color_input date_input date_select datetime_local_input
|
|
||||||
datetime_select email_input file_input hidden_input number_input password_input range_input
|
|
||||||
search_input telephone_input textarea text_input time_input time_select url_input
|
|
||||||
reset submit phx_label radio_button multiple_select select phx_link phx_button
|
|
||||||
]a |> Enum.map(fn e -> {e, :*} end)
|
|
||||||
|
|
||||||
[
|
[
|
||||||
inputs: ["*.{ex,exs}", "{config,lib,test}/**/*.{ex,exs}"],
|
inputs: ["*.{ex,exs}", "{config,lib,test}/**/*.{ex,exs}"],
|
||||||
|
|
|
@ -29,7 +29,7 @@ defmodule Temple do
|
||||||
|
|
||||||
# The class attribute also can take a keyword list of classes to conditionally render, based on the boolean result of the value.
|
# The class attribute also can take a keyword list of classes to conditionally render, based on the boolean result of the value.
|
||||||
|
|
||||||
div class: ["text-red-500": false, "text-green-500": true ] do
|
div class: ["text-red-500": false, "text-green-500": true] do
|
||||||
"Alert!"
|
"Alert!"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -71,6 +71,34 @@ defmodule Temple do
|
||||||
end
|
end
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Whitespace Control
|
||||||
|
|
||||||
|
By default, Temple will emit internal whitespace into tags, something like this.
|
||||||
|
|
||||||
|
```elixir
|
||||||
|
span do
|
||||||
|
"Hello, world!"
|
||||||
|
end
|
||||||
|
```
|
||||||
|
|
||||||
|
```html
|
||||||
|
<span>
|
||||||
|
Hello, world!
|
||||||
|
</span>
|
||||||
|
```
|
||||||
|
|
||||||
|
If you need to create a "tight" tag, you can call the "bang" version of the desired tag.
|
||||||
|
|
||||||
|
```elixir
|
||||||
|
span! do
|
||||||
|
"Hello, world!"
|
||||||
|
end
|
||||||
|
```
|
||||||
|
|
||||||
|
```html
|
||||||
|
<span>Hello, world!</span>
|
||||||
|
```
|
||||||
|
|
||||||
## Configuration
|
## Configuration
|
||||||
|
|
||||||
### Mode
|
### Mode
|
||||||
|
@ -151,7 +179,8 @@ defmodule Temple do
|
||||||
markup =
|
markup =
|
||||||
block
|
block
|
||||||
|> Parser.parse()
|
|> Parser.parse()
|
||||||
|> Enum.map(&Temple.Generator.to_eex/1)
|
|> Enum.map(fn parsed -> Temple.Generator.to_eex(parsed, 0) end)
|
||||||
|
|> Enum.intersperse("\n")
|
||||||
|> :erlang.iolist_to_binary()
|
|> :erlang.iolist_to_binary()
|
||||||
|
|
||||||
quote location: :keep do
|
quote location: :keep do
|
||||||
|
@ -163,7 +192,8 @@ defmodule Temple do
|
||||||
quote location: :keep do
|
quote location: :keep do
|
||||||
unquote(block)
|
unquote(block)
|
||||||
|> Parser.parse()
|
|> Parser.parse()
|
||||||
|> Enum.map(&Temple.Generator.to_eex/1)
|
|> Enum.map(fn parsed -> Temple.Generator.to_eex(parsed, 0) end)
|
||||||
|
|> Enum.intersperse("\n")
|
||||||
|> :erlang.iolist_to_binary()
|
|> :erlang.iolist_to_binary()
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -190,7 +220,8 @@ defmodule Temple do
|
||||||
markup =
|
markup =
|
||||||
block
|
block
|
||||||
|> Parser.parse()
|
|> Parser.parse()
|
||||||
|> Enum.map(&Temple.Generator.to_eex/1)
|
|> Enum.map(fn parsed -> Temple.Generator.to_eex(parsed, 0) end)
|
||||||
|
|> Enum.intersperse("\n")
|
||||||
|> :erlang.iolist_to_binary()
|
|> :erlang.iolist_to_binary()
|
||||||
|
|
||||||
EEx.compile_string(markup, engine: engine, line: __CALLER__.line, file: __CALLER__.file)
|
EEx.compile_string(markup, engine: engine, line: __CALLER__.line, file: __CALLER__.file)
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
defprotocol Temple.Generator do
|
defprotocol Temple.Generator do
|
||||||
@moduledoc false
|
@moduledoc false
|
||||||
|
|
||||||
def to_eex(ast)
|
def to_eex(ast, indent \\ 0)
|
||||||
end
|
end
|
||||||
|
|
|
@ -58,7 +58,7 @@ defmodule Temple.Parser do
|
||||||
option textarea output progress meter
|
option textarea output progress meter
|
||||||
details summary menuitem menu
|
details summary menuitem menu
|
||||||
html
|
html
|
||||||
]a
|
]a |> Enum.flat_map(fn el -> [el, :"#{el}!"] end)
|
||||||
|
|
||||||
@nonvoid_elements_aliases Enum.map(@nonvoid_elements, fn el ->
|
@nonvoid_elements_aliases Enum.map(@nonvoid_elements, fn el ->
|
||||||
Keyword.get(@aliases, el, el)
|
Keyword.get(@aliases, el, el)
|
||||||
|
|
|
@ -32,14 +32,14 @@ defmodule Temple.Parser.AnonymousFunctions do
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(%{elixir_ast: {name, _, args}, children: children}) do
|
def to_eex(%{elixir_ast: {name, _, args}, children: children}, indent \\ 0) do
|
||||||
{_do_and_else, args} = Temple.Parser.Utils.split_args(args)
|
{_do_and_else, args} = Temple.Parser.Utils.split_args(args)
|
||||||
|
|
||||||
{args, {func, _, [{arrow, _, [[{arg, _, _}], _block]}]}, args2} =
|
{args, {func, _, [{arrow, _, [[{arg, _, _}], _block]}]}, args2} =
|
||||||
Temple.Parser.Utils.split_on_fn(args, {[], nil, []})
|
Temple.Parser.Utils.split_on_fn(args, {[], nil, []})
|
||||||
|
|
||||||
[
|
[
|
||||||
"<%= ",
|
"#{Parser.Utils.indent(indent)}<%= ",
|
||||||
to_string(name),
|
to_string(name),
|
||||||
" ",
|
" ",
|
||||||
Enum.map(args, &Macro.to_string(&1)) |> Enum.join(", "),
|
Enum.map(args, &Macro.to_string(&1)) |> Enum.join(", "),
|
||||||
|
@ -51,16 +51,16 @@ defmodule Temple.Parser.AnonymousFunctions do
|
||||||
to_string(arrow),
|
to_string(arrow),
|
||||||
" %>",
|
" %>",
|
||||||
"\n",
|
"\n",
|
||||||
for(child <- children, do: Temple.Generator.to_eex(child)),
|
for(child <- children, do: Temple.Generator.to_eex(child, indent + 1)),
|
||||||
if Enum.any?(args2) do
|
if Enum.any?(args2) do
|
||||||
[
|
[
|
||||||
"<% end, ",
|
"#{Parser.Utils.indent(indent)}<% end, ",
|
||||||
Enum.map(args2, fn arg -> Macro.to_string(arg) end)
|
Enum.map(args2, fn arg -> Macro.to_string(arg) end)
|
||||||
|> Enum.join(", "),
|
|> Enum.join(", "),
|
||||||
" %>"
|
" %>"
|
||||||
]
|
]
|
||||||
else
|
else
|
||||||
["<% end %>", "\n"]
|
["#{Parser.Utils.indent(indent)}<% end %>", "\n"]
|
||||||
end
|
end
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,6 +2,8 @@ defmodule Temple.Parser.Components do
|
||||||
@moduledoc false
|
@moduledoc false
|
||||||
@behaviour Temple.Parser
|
@behaviour Temple.Parser
|
||||||
|
|
||||||
|
alias Temple.Parser
|
||||||
|
|
||||||
defstruct module: nil, assigns: [], children: [], slots: []
|
defstruct module: nil, assigns: [], children: [], slots: []
|
||||||
|
|
||||||
@impl Temple.Parser
|
@impl Temple.Parser
|
||||||
|
@ -88,12 +90,12 @@ defmodule Temple.Parser.Components do
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(%{module: module, assigns: assigns, children: children, slots: slots}) do
|
def to_eex(%{module: module, assigns: assigns, children: children, slots: slots}, indent \\ 0) do
|
||||||
component_function = Temple.Config.mode().component_function
|
component_function = Temple.Config.mode().component_function
|
||||||
renderer = Temple.Config.mode().renderer.(module)
|
renderer = Temple.Config.mode().renderer.(module)
|
||||||
|
|
||||||
[
|
[
|
||||||
"<%= #{component_function} ",
|
"#{Parser.Utils.indent(indent)}<%= #{component_function} ",
|
||||||
renderer,
|
renderer,
|
||||||
", ",
|
", ",
|
||||||
Macro.to_string(assigns),
|
Macro.to_string(assigns),
|
||||||
|
@ -102,23 +104,22 @@ defmodule Temple.Parser.Components do
|
||||||
" do %>\n",
|
" do %>\n",
|
||||||
if not Enum.empty?(children) do
|
if not Enum.empty?(children) do
|
||||||
[
|
[
|
||||||
"<% {:default, _} -> %>\n",
|
"#{Parser.Utils.indent(indent + 1)}<% {:default, _} -> %>\n",
|
||||||
for(child <- children, do: Temple.Generator.to_eex(child)),
|
for(child <- children, do: Temple.Generator.to_eex(child, indent + 2))
|
||||||
"\n"
|
|
||||||
]
|
]
|
||||||
else
|
else
|
||||||
""
|
""
|
||||||
end,
|
end,
|
||||||
for slot <- slots do
|
for slot <- slots do
|
||||||
[
|
[
|
||||||
"<% {:",
|
"#{Parser.Utils.indent(indent + 1)}<% {:",
|
||||||
to_string(slot.name),
|
to_string(slot.name),
|
||||||
", ",
|
", ",
|
||||||
"#{Macro.to_string(slot.assigns)}} -> %>\n",
|
"#{Macro.to_string(slot.assigns)}} -> %>\n",
|
||||||
for(child <- slot.content, do: Temple.Generator.to_eex(child))
|
for(child <- slot.content, do: Temple.Generator.to_eex(child, indent + 2))
|
||||||
]
|
]
|
||||||
end,
|
end,
|
||||||
"<% end %>"
|
"\n#{Parser.Utils.indent(indent)}<% end %>"
|
||||||
]
|
]
|
||||||
else
|
else
|
||||||
" %>"
|
" %>"
|
||||||
|
|
|
@ -15,8 +15,8 @@ defmodule Temple.Parser.Default do
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(%{elixir_ast: expression}) do
|
def to_eex(%{elixir_ast: expression}, indent \\ 0) do
|
||||||
["<%= ", Macro.to_string(expression), " %>\n"]
|
["#{Parser.Utils.indent(indent)}<%= ", Macro.to_string(expression), " %>"]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -30,18 +30,23 @@ defmodule Temple.Parser.DoExpressions do
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(%{elixir_ast: expression, children: [do_body, else_body]}) do
|
def to_eex(%{elixir_ast: expression, children: [do_body, else_body]}, indent \\ 0) do
|
||||||
[
|
[
|
||||||
"<%= ",
|
"#{Parser.Utils.indent(indent)}<%= ",
|
||||||
Macro.to_string(expression),
|
Macro.to_string(expression),
|
||||||
" do %>",
|
" do %>",
|
||||||
"\n",
|
"\n",
|
||||||
for(child <- do_body, do: Temple.Generator.to_eex(child)),
|
for(child <- do_body, do: Temple.Generator.to_eex(child, indent + 1))
|
||||||
|
|> Enum.intersperse("\n"),
|
||||||
if(else_body != nil,
|
if(else_body != nil,
|
||||||
do: ["<% else %>\n", for(child <- else_body, do: Temple.Generator.to_eex(child))],
|
do: [
|
||||||
|
"#{Parser.Utils.indent(indent)}\n<% else %>\n",
|
||||||
|
for(child <- else_body, do: Temple.Generator.to_eex(child, indent + 1))
|
||||||
|
|> Enum.intersperse("\n")
|
||||||
|
],
|
||||||
else: ""
|
else: ""
|
||||||
),
|
),
|
||||||
"<% end %>"
|
"\n#{Parser.Utils.indent(indent)}<% end %>"
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
33
lib/temple/parser/element_list.ex
Normal file
33
lib/temple/parser/element_list.ex
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
defmodule Temple.Parser.ElementList do
|
||||||
|
@moduledoc false
|
||||||
|
|
||||||
|
@behaviour Temple.Parser
|
||||||
|
|
||||||
|
defstruct children: [], whitespace: :loose
|
||||||
|
|
||||||
|
@impl Temple.Parser
|
||||||
|
def applicable?(asts), do: is_list(asts)
|
||||||
|
|
||||||
|
@impl Temple.Parser
|
||||||
|
def run(asts) do
|
||||||
|
children = Enum.flat_map(asts, &Temple.Parser.parse/1)
|
||||||
|
|
||||||
|
Temple.Ast.new(__MODULE__, children: children)
|
||||||
|
end
|
||||||
|
|
||||||
|
defimpl Temple.Generator do
|
||||||
|
def to_eex(%{children: children, whitespace: whitespace}, indent \\ 0) do
|
||||||
|
child_indent = if whitespace == :loose, do: indent + 1, else: 0
|
||||||
|
self_indent = if whitespace == :loose, do: indent, else: 0
|
||||||
|
whitespace = if whitespace == :tight, do: [], else: ["\n"]
|
||||||
|
|
||||||
|
[
|
||||||
|
whitespace,
|
||||||
|
for(child <- children, do: Temple.Generator.to_eex(child, child_indent))
|
||||||
|
|> Enum.intersperse("\n"),
|
||||||
|
whitespace,
|
||||||
|
Temple.Parser.Utils.indent(self_indent)
|
||||||
|
]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -16,7 +16,7 @@ defmodule Temple.Parser.Empty do
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(_) do
|
def to_eex(_, _ \\ 0) do
|
||||||
[]
|
[]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -19,8 +19,8 @@ defmodule Temple.Parser.Match do
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(%{elixir_ast: elixir_ast}) do
|
def to_eex(%{elixir_ast: elixir_ast}, indent \\ 0) do
|
||||||
["<% ", Macro.to_string(elixir_ast), " %>"]
|
["#{Parser.Utils.indent(indent)}<% ", Macro.to_string(elixir_ast), " %>"]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,7 +2,7 @@ defmodule Temple.Parser.NonvoidElementsAliases do
|
||||||
@moduledoc false
|
@moduledoc false
|
||||||
@behaviour Temple.Parser
|
@behaviour Temple.Parser
|
||||||
|
|
||||||
defstruct name: nil, attrs: [], children: []
|
defstruct name: nil, attrs: [], children: [], meta: %{}
|
||||||
|
|
||||||
alias Temple.Parser
|
alias Temple.Parser
|
||||||
|
|
||||||
|
@ -23,18 +23,34 @@ defmodule Temple.Parser.NonvoidElementsAliases do
|
||||||
|
|
||||||
children = Temple.Parser.parse(do_and_else[:do])
|
children = Temple.Parser.parse(do_and_else[:do])
|
||||||
|
|
||||||
Temple.Ast.new(__MODULE__, name: to_string(name), attrs: args, children: children)
|
Temple.Ast.new(__MODULE__,
|
||||||
|
name: to_string(name) |> String.replace_suffix("!", ""),
|
||||||
|
attrs: args,
|
||||||
|
children:
|
||||||
|
Temple.Ast.new(Temple.Parser.ElementList,
|
||||||
|
children: children,
|
||||||
|
whitespace: whitespace(to_string(name))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
defp whitespace(name) do
|
||||||
|
if String.ends_with?(name, "!") do
|
||||||
|
:tight
|
||||||
|
else
|
||||||
|
:loose
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(%{name: name, attrs: attrs, children: children}) do
|
def to_eex(%{name: name, attrs: attrs, children: children}, indent \\ 0) do
|
||||||
[
|
[
|
||||||
"<",
|
"#{Parser.Utils.indent(indent)}<",
|
||||||
name,
|
name,
|
||||||
Temple.Parser.Utils.compile_attrs(attrs),
|
Temple.Parser.Utils.compile_attrs(attrs),
|
||||||
">\n",
|
">",
|
||||||
for(child <- children, do: Temple.Generator.to_eex(child)),
|
Temple.Generator.to_eex(children, indent),
|
||||||
"\n</",
|
"</",
|
||||||
name,
|
name,
|
||||||
">"
|
">"
|
||||||
]
|
]
|
||||||
|
|
|
@ -18,12 +18,12 @@ defmodule Temple.Parser.RightArrow do
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(%{elixir_ast: elixir_ast, children: children}) do
|
def to_eex(%{elixir_ast: elixir_ast, children: children}, indent \\ 0) do
|
||||||
[
|
[
|
||||||
"<% ",
|
"#{Parser.Utils.indent(indent)}<% ",
|
||||||
Macro.to_string(elixir_ast),
|
Macro.to_string(elixir_ast),
|
||||||
" -> %>\n",
|
" -> %>\n",
|
||||||
for(child <- children, do: Temple.Generator.to_eex(child))
|
for(child <- children, do: Temple.Generator.to_eex(child, indent + 1))
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
defmodule Temple.Parser.Slot do
|
defmodule Temple.Parser.Slot do
|
||||||
@moduledoc false
|
@moduledoc false
|
||||||
@behaviour Temple.Parser
|
@behaviour Temple.Parser
|
||||||
|
alias Temple.Parser.Utils
|
||||||
|
|
||||||
defstruct name: nil, args: []
|
defstruct name: nil, args: []
|
||||||
|
|
||||||
|
@ -26,15 +27,15 @@ defmodule Temple.Parser.Slot do
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(%{name: name, args: args}) do
|
def to_eex(%{name: name, args: args}, indent \\ 0) do
|
||||||
render_block_function = Temple.Config.mode().render_block_function
|
render_block_function = Temple.Config.mode().render_block_function
|
||||||
|
|
||||||
[
|
[
|
||||||
"<%= #{render_block_function}(@inner_block, {:",
|
"#{Utils.indent(indent)}<%= #{render_block_function}(@inner_block, {:",
|
||||||
to_string(name),
|
to_string(name),
|
||||||
", ",
|
", ",
|
||||||
Macro.to_string(quote(do: Enum.into(unquote(args), %{}))),
|
Macro.to_string(quote(do: Enum.into(unquote(args), %{}))),
|
||||||
"}) %>"
|
"}) %>\n"
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -16,8 +16,8 @@ defmodule Temple.Parser.Text do
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(%{text: text}) do
|
def to_eex(%{text: text}, indent \\ 0) do
|
||||||
[text, "\n"]
|
[Parser.Utils.indent(indent), text]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -113,4 +113,8 @@ defmodule Temple.Parser.Utils do
|
||||||
def pop_compact?(args) do
|
def pop_compact?(args) do
|
||||||
Keyword.pop(args, :compact, false)
|
Keyword.pop(args, :compact, false)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def indent(level) do
|
||||||
|
String.duplicate(" ", level * 2)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,6 +2,8 @@ defmodule Temple.Parser.VoidElementsAliases do
|
||||||
@moduledoc false
|
@moduledoc false
|
||||||
@behaviour Temple.Parser
|
@behaviour Temple.Parser
|
||||||
|
|
||||||
|
alias Temple.Parser.Utils
|
||||||
|
|
||||||
defstruct name: nil, attrs: []
|
defstruct name: nil, attrs: []
|
||||||
|
|
||||||
@impl Temple.Parser
|
@impl Temple.Parser
|
||||||
|
@ -28,12 +30,12 @@ defmodule Temple.Parser.VoidElementsAliases do
|
||||||
end
|
end
|
||||||
|
|
||||||
defimpl Temple.Generator do
|
defimpl Temple.Generator do
|
||||||
def to_eex(%{name: name, attrs: attrs}) do
|
def to_eex(%{name: name, attrs: attrs}, indent \\ 0) do
|
||||||
[
|
[
|
||||||
"<",
|
"#{Utils.indent(indent)}<",
|
||||||
to_string(name),
|
to_string(name),
|
||||||
Temple.Parser.Utils.compile_attrs(attrs),
|
Temple.Parser.Utils.compile_attrs(attrs),
|
||||||
">\n"
|
">"
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -18,7 +18,19 @@ defmodule Temple.ComponentTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert evaluate_template(result) ==
|
assert evaluate_template(result) ==
|
||||||
~s{<div class="font-bold">Hello, world</div><div><aside class="foobar">I'm a component!</aside></div>}
|
~s"""
|
||||||
|
<div class="font-bold">
|
||||||
|
Hello, world
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
|
||||||
|
<aside class="foobar">
|
||||||
|
I'm a component!
|
||||||
|
</aside>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "function components can accept local assigns" do
|
test "function components can accept local assigns" do
|
||||||
|
@ -34,7 +46,16 @@ defmodule Temple.ComponentTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert evaluate_template(result) ==
|
assert evaluate_template(result) ==
|
||||||
~s{<div class="font-bold">Hello, world</div><div class="bg-red">I'm a component!</div>}
|
~s"""
|
||||||
|
<div class="font-bold">
|
||||||
|
Hello, world
|
||||||
|
</div>
|
||||||
|
<div class="bg-red">
|
||||||
|
|
||||||
|
I'm a component!
|
||||||
|
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "function components can use other components" do
|
test "function components can use other components" do
|
||||||
|
@ -50,8 +71,21 @@ defmodule Temple.ComponentTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert evaluate_template(result) == ~s"""
|
assert evaluate_template(result) == ~s"""
|
||||||
<div id="inner" outer-id="from-outer">outer!</div>
|
<div id="inner" outer-id="from-outer">
|
||||||
<div id="inner" outer-id="set by root inner">inner!</div>
|
|
||||||
|
|
||||||
|
outer!
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<div id="inner" outer-id="set by root inner">
|
||||||
|
|
||||||
|
inner!
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
"""
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -63,7 +97,14 @@ defmodule Temple.ComponentTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert evaluate_template(result) == ~s{<div class="barbarbar">doo doo</div>}
|
assert evaluate_template(result) == ~s"""
|
||||||
|
<div class="barbarbar">
|
||||||
|
|
||||||
|
doo doo
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "components can be void elements" do
|
test "components can be void elements" do
|
||||||
|
@ -72,7 +113,11 @@ defmodule Temple.ComponentTest do
|
||||||
c Temple.Components.VoidComponent, foo: :bar
|
c Temple.Components.VoidComponent, foo: :bar
|
||||||
end
|
end
|
||||||
|
|
||||||
assert evaluate_template(result) == ~s{<div class="void!!">bar</div>}
|
assert evaluate_template(result) == ~s"""
|
||||||
|
<div class="void!!">
|
||||||
|
bar
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "components can have named slots" do
|
test "components can have named slots" do
|
||||||
|
@ -94,6 +139,22 @@ defmodule Temple.ComponentTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert evaluate_template(result, assigns) ==
|
assert evaluate_template(result, assigns) ==
|
||||||
~s{<div><div>the value is Header</div><div class="wrapped"><button class="btn" phx-click="toggle">bob</button></div></div>}
|
~s"""
|
||||||
|
<div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
the value is Header
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="wrapped">
|
||||||
|
|
||||||
|
<button class="btn" phx-click="toggle">
|
||||||
|
bob
|
||||||
|
</button>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -7,9 +7,9 @@ defmodule Temple.Parser.AnonymousFunctionsTest do
|
||||||
test "returns true when the node contains an anonymous function as an argument to a function" do
|
test "returns true when the node contains an anonymous function as an argument to a function" do
|
||||||
raw_asts = [
|
raw_asts = [
|
||||||
quote do
|
quote do
|
||||||
form_for changeset, Routes.foo_path(conn, :create), fn form ->
|
form_for(changeset, Routes.foo_path(conn, :create), fn form ->
|
||||||
Does.something!(form)
|
Does.something!(form)
|
||||||
end
|
end)
|
||||||
end
|
end
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -47,9 +47,9 @@ defmodule Temple.Parser.AnonymousFunctionsTest do
|
||||||
|
|
||||||
raw_ast =
|
raw_ast =
|
||||||
quote do
|
quote do
|
||||||
form_for changeset, Routes.foo_path(conn, :create), fn form ->
|
form_for(changeset, Routes.foo_path(conn, :create), fn form ->
|
||||||
unquote(expected_child)
|
unquote(expected_child)
|
||||||
end
|
end)
|
||||||
end
|
end
|
||||||
|
|
||||||
ast = AnonymousFunctions.run(raw_ast)
|
ast = AnonymousFunctions.run(raw_ast)
|
||||||
|
@ -69,9 +69,9 @@ defmodule Temple.Parser.AnonymousFunctionsTest do
|
||||||
test "emits eex" do
|
test "emits eex" do
|
||||||
raw_ast =
|
raw_ast =
|
||||||
quote do
|
quote do
|
||||||
form_for changeset, Routes.foo_path(conn, :create), fn form ->
|
form_for(changeset, Routes.foo_path(conn, :create), fn form ->
|
||||||
Does.something!(form)
|
Does.something!(form)
|
||||||
end
|
end)
|
||||||
end
|
end
|
||||||
|
|
||||||
result =
|
result =
|
||||||
|
|
|
@ -187,7 +187,12 @@ defmodule Temple.Parser.ComponentsTest do
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() ==
|
assert result |> :erlang.iolist_to_binary() ==
|
||||||
~s|<%= Temple.Component.__component__ SomeModule, [foo: :bar] do %>\n<% {:default, _} -> %>\nI'm a component!\n<% end %>|
|
~s"""
|
||||||
|
<%= Temple.Component.__component__ SomeModule, [foo: :bar] do %>
|
||||||
|
<% {:default, _} -> %>
|
||||||
|
I'm a component!
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "emits eex for void component with slots" do
|
test "emits eex for void component with slots" do
|
||||||
|
@ -208,7 +213,14 @@ defmodule Temple.Parser.ComponentsTest do
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() ==
|
assert result |> :erlang.iolist_to_binary() ==
|
||||||
~s|<%= Temple.Component.__component__ SomeModule, [foo: :bar] do %>\n<% {:foo, %{form: form}} -> %>\n<div>\nin the slot\n\n</div>\n<% end %>|
|
~s"""
|
||||||
|
<%= Temple.Component.__component__ SomeModule, [foo: :bar] do %>
|
||||||
|
<% {:foo, %{form: form}} -> %>
|
||||||
|
<div>
|
||||||
|
in the slot
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "emits eex for nonvoid component with slots" do
|
test "emits eex for nonvoid component with slots" do
|
||||||
|
@ -233,7 +245,18 @@ defmodule Temple.Parser.ComponentsTest do
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() ==
|
assert result |> :erlang.iolist_to_binary() ==
|
||||||
~s|<%= Temple.Component.__component__ SomeModule, [foo: :bar] do %>\n<% {:default, _} -> %>\n<div>\ninner content\n\n</div>\n<% {:foo, %{form: form}} -> %><div>\nin the slot\n\n</div><% end %>|
|
~s"""
|
||||||
|
<%= Temple.Component.__component__ SomeModule, [foo: :bar] do %>
|
||||||
|
<% {:default, _} -> %>
|
||||||
|
<div>
|
||||||
|
inner content
|
||||||
|
</div>
|
||||||
|
<% {:foo, %{form: form}} -> %>
|
||||||
|
<div>
|
||||||
|
in the slot
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "emits eex for void component" do
|
test "emits eex for void component" do
|
||||||
|
|
|
@ -2,6 +2,7 @@ defmodule Temple.Parser.DefaultTest do
|
||||||
use ExUnit.Case, async: true
|
use ExUnit.Case, async: true
|
||||||
|
|
||||||
alias Temple.Parser.Default
|
alias Temple.Parser.Default
|
||||||
|
alias Temple.Support.Utils
|
||||||
|
|
||||||
describe "applicable?/1" do
|
describe "applicable?/1" do
|
||||||
test "returns true when the node is an elixir expression" do
|
test "returns true when the node is an elixir expression" do
|
||||||
|
@ -35,8 +36,11 @@ defmodule Temple.Parser.DefaultTest do
|
||||||
end
|
end
|
||||||
|> Default.run()
|
|> Default.run()
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|> Utils.iolist_to_binary()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() == ~s|<%= Foo.bar!(baz) %>\n|
|
assert result == ~s"""
|
||||||
|
<%= Foo.bar!(baz) %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,6 +2,7 @@ defmodule Temple.Parser.DoExpressionsTest do
|
||||||
use ExUnit.Case, async: true
|
use ExUnit.Case, async: true
|
||||||
|
|
||||||
alias Temple.Parser.DoExpressions
|
alias Temple.Parser.DoExpressions
|
||||||
|
alias Temple.Support.Utils
|
||||||
|
|
||||||
describe "applicable?/1" do
|
describe "applicable?/1" do
|
||||||
test "returns true when the node contains a do expression" do
|
test "returns true when the node contains a do expression" do
|
||||||
|
@ -47,9 +48,14 @@ defmodule Temple.Parser.DoExpressionsTest do
|
||||||
end
|
end
|
||||||
|> DoExpressions.run()
|
|> DoExpressions.run()
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|> Utils.iolist_to_binary()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() ==
|
assert result ==
|
||||||
~s|<%= for(big <- boys) do %>\nbob\n<% end %>|
|
~s"""
|
||||||
|
<%= for(big <- boys) do %>
|
||||||
|
bob
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "emits eex for that includes in else clause" do
|
test "emits eex for that includes in else clause" do
|
||||||
|
@ -65,9 +71,17 @@ defmodule Temple.Parser.DoExpressionsTest do
|
||||||
end
|
end
|
||||||
|> DoExpressions.run()
|
|> DoExpressions.run()
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|> Utils.iolist_to_binary()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() ==
|
assert result ==
|
||||||
~s|<%= if(foo?) do %>\nbob\nbobby\n<% else %>\ncarol\n<% end %>|
|
~s"""
|
||||||
|
<%= if(foo?) do %>
|
||||||
|
bob
|
||||||
|
bobby
|
||||||
|
<% else %>
|
||||||
|
carol
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "emits eex for a case expression" do
|
test "emits eex for a case expression" do
|
||||||
|
@ -80,9 +94,15 @@ defmodule Temple.Parser.DoExpressionsTest do
|
||||||
end
|
end
|
||||||
|> DoExpressions.run()
|
|> DoExpressions.run()
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|> Utils.iolist_to_binary()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() ==
|
assert result ==
|
||||||
~s|<%= case(foo?) do %>\n<% :bing -> %>\n<%= :bong %>\n<% end %>|
|
~s"""
|
||||||
|
<%= case(foo?) do %>
|
||||||
|
<% :bing -> %>
|
||||||
|
<%= :bong %>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,6 +2,8 @@ defmodule Temple.Parser.NonvoidElementsAliasesTest do
|
||||||
use ExUnit.Case, async: true
|
use ExUnit.Case, async: true
|
||||||
|
|
||||||
alias Temple.Parser.NonvoidElementsAliases
|
alias Temple.Parser.NonvoidElementsAliases
|
||||||
|
alias Temple.Parser.ElementList
|
||||||
|
alias Temple.Support.Utils
|
||||||
|
|
||||||
describe "applicable?/1" do
|
describe "applicable?/1" do
|
||||||
test "returns true when the node is a nonvoid element or alias" do
|
test "returns true when the node is a nonvoid element or alias" do
|
||||||
|
@ -63,19 +65,25 @@ defmodule Temple.Parser.NonvoidElementsAliasesTest do
|
||||||
assert %NonvoidElementsAliases{
|
assert %NonvoidElementsAliases{
|
||||||
name: "div",
|
name: "div",
|
||||||
attrs: [class: "foo", id: {:var, [], _}],
|
attrs: [class: "foo", id: {:var, [], _}],
|
||||||
|
children: %ElementList{
|
||||||
children: [
|
children: [
|
||||||
%NonvoidElementsAliases{
|
%NonvoidElementsAliases{
|
||||||
name: "select",
|
name: "select",
|
||||||
|
children: %ElementList{
|
||||||
children: [
|
children: [
|
||||||
%NonvoidElementsAliases{
|
%NonvoidElementsAliases{
|
||||||
name: "option",
|
name: "option",
|
||||||
|
children: %ElementList{
|
||||||
children: [
|
children: [
|
||||||
%Temple.Parser.Text{text: "foo"}
|
%Temple.Parser.Text{text: "foo"}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
]
|
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
} = ast
|
} = ast
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -94,9 +102,44 @@ defmodule Temple.Parser.NonvoidElementsAliasesTest do
|
||||||
end
|
end
|
||||||
|> NonvoidElementsAliases.run()
|
|> NonvoidElementsAliases.run()
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|> Utils.iolist_to_binary()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() ==
|
assert result ==
|
||||||
~s|<div class="foo"<%= {:safe, Temple.Parser.Utils.build_attr("id", var)} %>>\n<select>\n<option>\nfoo\n\n</option>\n</select>\n</div>|
|
~s"""
|
||||||
|
<div class="foo"<%= {:safe, Temple.Parser.Utils.build_attr("id", var)} %>>
|
||||||
|
<select>
|
||||||
|
<option>
|
||||||
|
foo
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
|
end
|
||||||
|
|
||||||
|
test "produce 'tight' markup" do
|
||||||
|
result =
|
||||||
|
quote do
|
||||||
|
div class: "foo", id: var do
|
||||||
|
select__ do
|
||||||
|
option! do
|
||||||
|
"foo"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|> NonvoidElementsAliases.run()
|
||||||
|
|> Temple.Generator.to_eex()
|
||||||
|
|> :erlang.iolist_to_binary()
|
||||||
|
|> Kernel.<>("\n")
|
||||||
|
|
||||||
|
assert result ==
|
||||||
|
~s"""
|
||||||
|
<div class="foo"<%= {:safe, Temple.Parser.Utils.build_attr("id", var)} %>>
|
||||||
|
<select>
|
||||||
|
<option>foo</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,6 +2,7 @@ defmodule Temple.Parser.RightArrowTest do
|
||||||
use ExUnit.Case, async: true
|
use ExUnit.Case, async: true
|
||||||
|
|
||||||
alias Temple.Parser.RightArrow
|
alias Temple.Parser.RightArrow
|
||||||
|
alias Temple.Support.Utils
|
||||||
|
|
||||||
describe "applicable?/1" do
|
describe "applicable?/1" do
|
||||||
test "returns true when the node contains a right arrow" do
|
test "returns true when the node contains a right arrow" do
|
||||||
|
@ -70,9 +71,13 @@ defmodule Temple.Parser.RightArrowTest do
|
||||||
|> List.first()
|
|> List.first()
|
||||||
|> RightArrow.run()
|
|> RightArrow.run()
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|> Utils.iolist_to_binary()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() ==
|
assert result ==
|
||||||
~s|<% :bing -> %>\n<%= :bong %>\n|
|
~s"""
|
||||||
|
<% :bing -> %>
|
||||||
|
<%= :bong %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -42,7 +42,9 @@ defmodule Temple.Parser.SlotTest do
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() ==
|
assert result |> :erlang.iolist_to_binary() ==
|
||||||
~s|<%= Temple.Component.__render_block__(@inner_block, {:header, Enum.into([value: Form.form_for(changeset, action)], %{})}) %>|
|
~s"""
|
||||||
|
<%= Temple.Component.__render_block__(@inner_block, {:header, Enum.into([value: Form.form_for(changeset, action)], %{})}) %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -3,6 +3,7 @@ defmodule Temple.Parser.TempleNamespaceNonvoidTest do
|
||||||
|
|
||||||
alias Temple.Parser.NonvoidElementsAliases
|
alias Temple.Parser.NonvoidElementsAliases
|
||||||
alias Temple.Parser.TempleNamespaceNonvoid
|
alias Temple.Parser.TempleNamespaceNonvoid
|
||||||
|
alias Temple.Support.Utils
|
||||||
|
|
||||||
describe "applicable?/1" do
|
describe "applicable?/1" do
|
||||||
test "returns true when the node is a Temple aliased nonvoid element" do
|
test "returns true when the node is a Temple aliased nonvoid element" do
|
||||||
|
@ -50,7 +51,10 @@ defmodule Temple.Parser.TempleNamespaceNonvoidTest do
|
||||||
assert %NonvoidElementsAliases{
|
assert %NonvoidElementsAliases{
|
||||||
name: "div",
|
name: "div",
|
||||||
attrs: [class: "foo", id: {:var, [], _}],
|
attrs: [class: "foo", id: {:var, [], _}],
|
||||||
children: [%Temple.Parser.Text{text: "foo"}]
|
children: %Temple.Parser.ElementList{
|
||||||
|
children: [%Temple.Parser.Text{text: "foo"}],
|
||||||
|
whitespace: :loose
|
||||||
|
}
|
||||||
} = ast
|
} = ast
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -65,9 +69,14 @@ defmodule Temple.Parser.TempleNamespaceNonvoidTest do
|
||||||
end
|
end
|
||||||
|> TempleNamespaceNonvoid.run()
|
|> TempleNamespaceNonvoid.run()
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|> Utils.iolist_to_binary()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() ==
|
assert result ==
|
||||||
~s|<div class="foo"<%= {:safe, Temple.Parser.Utils.build_attr("id", var)} %>>\nfoo\n\n</div>|
|
~s"""
|
||||||
|
<div class="foo"<%= {:safe, Temple.Parser.Utils.build_attr("id", var)} %>>
|
||||||
|
foo
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -3,6 +3,7 @@ defmodule Temple.Parser.TempleNamespaceVoidTest do
|
||||||
|
|
||||||
alias Temple.Parser.TempleNamespaceVoid
|
alias Temple.Parser.TempleNamespaceVoid
|
||||||
alias Temple.Parser.VoidElementsAliases
|
alias Temple.Parser.VoidElementsAliases
|
||||||
|
alias Temple.Support.Utils
|
||||||
|
|
||||||
describe "applicable?/1" do
|
describe "applicable?/1" do
|
||||||
test "returns true when the node is a Temple aliased nonvoid element" do
|
test "returns true when the node is a Temple aliased nonvoid element" do
|
||||||
|
@ -58,8 +59,9 @@ defmodule Temple.Parser.TempleNamespaceVoidTest do
|
||||||
end
|
end
|
||||||
|> TempleNamespaceVoid.run()
|
|> TempleNamespaceVoid.run()
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|> Utils.iolist_to_binary()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() == ~s|<meta content="foo">\n|
|
assert result == ~s|<meta content="foo">\n|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,6 +2,7 @@ defmodule Temple.Parser.TextTest do
|
||||||
use ExUnit.Case, async: true
|
use ExUnit.Case, async: true
|
||||||
|
|
||||||
alias Temple.Parser.Text
|
alias Temple.Parser.Text
|
||||||
|
alias Temple.Support.Utils
|
||||||
|
|
||||||
describe "applicable?/1" do
|
describe "applicable?/1" do
|
||||||
test "returns true when the node is a string literal" do
|
test "returns true when the node is a string literal" do
|
||||||
|
@ -35,8 +36,9 @@ defmodule Temple.Parser.TextTest do
|
||||||
"string literal"
|
"string literal"
|
||||||
|> Text.run()
|
|> Text.run()
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|> Utils.iolist_to_binary()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() == ~s|string literal\n|
|
assert result == ~s|string literal\n|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,6 +2,7 @@ defmodule Temple.Parser.VoidElementsAliasesTest do
|
||||||
use ExUnit.Case, async: true
|
use ExUnit.Case, async: true
|
||||||
|
|
||||||
alias Temple.Parser.VoidElementsAliases
|
alias Temple.Parser.VoidElementsAliases
|
||||||
|
alias Temple.Support.Utils
|
||||||
|
|
||||||
describe "applicable?/1" do
|
describe "applicable?/1" do
|
||||||
test "returns true when the node is a nonvoid element or alias" do
|
test "returns true when the node is a nonvoid element or alias" do
|
||||||
|
@ -63,8 +64,9 @@ defmodule Temple.Parser.VoidElementsAliasesTest do
|
||||||
end
|
end
|
||||||
|> VoidElementsAliases.run()
|
|> VoidElementsAliases.run()
|
||||||
|> Temple.Generator.to_eex()
|
|> Temple.Generator.to_eex()
|
||||||
|
|> Utils.iolist_to_binary()
|
||||||
|
|
||||||
assert result |> :erlang.iolist_to_binary() == ~s|<meta content="foo">\n|
|
assert result == ~s|<meta content="foo">\n|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -33,4 +33,20 @@ defmodule Temple.Support.Utils do
|
||||||
|> elem(0)
|
|> elem(0)
|
||||||
|> Phoenix.HTML.safe_to_string()
|
|> Phoenix.HTML.safe_to_string()
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@doc """
|
||||||
|
Converts an iolist to a binary and appends a new line.
|
||||||
|
"""
|
||||||
|
def iolist_to_binary(iolist) do
|
||||||
|
iolist
|
||||||
|
|> :erlang.iolist_to_binary()
|
||||||
|
|> append_new_line()
|
||||||
|
end
|
||||||
|
|
||||||
|
@doc """
|
||||||
|
Appends a new line to a string.
|
||||||
|
"""
|
||||||
|
def append_new_line(string) do
|
||||||
|
string <> "\n"
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -11,7 +11,12 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
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 void element" do
|
test "renders void element" do
|
||||||
|
@ -32,7 +37,12 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result == ~s{<div class="hello">hifoo</div>}
|
assert result == ~s"""
|
||||||
|
<div class="hello">
|
||||||
|
hi
|
||||||
|
foo
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders a variable text node as eex" do
|
test "renders a variable text node as eex" do
|
||||||
|
@ -43,7 +53,11 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result == ~s{<div class="hello"><%= foo %></div>}
|
assert result == ~s"""
|
||||||
|
<div class="hello">
|
||||||
|
<%= foo %>
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders an assign text node as eex" do
|
test "renders an assign text node as eex" do
|
||||||
|
@ -54,7 +68,11 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result == ~s{<div class="hello"><%= @foo %></div>}
|
assert result == ~s"""
|
||||||
|
<div class="hello">
|
||||||
|
<%= @foo %>
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders a match expression" do
|
test "renders a match expression" do
|
||||||
|
@ -67,7 +85,12 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result == ~s{<% x = 420 %><div>blaze it</div>}
|
assert result == ~s"""
|
||||||
|
<% x = 420 %>
|
||||||
|
<div>
|
||||||
|
blaze it
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders a non-match expression" do
|
test "renders a non-match expression" do
|
||||||
|
@ -80,7 +103,12 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result == ~s{<%= IO.inspect(:foo) %><div>bar</div>}
|
assert result == ~s"""
|
||||||
|
<%= IO.inspect(:foo) %>
|
||||||
|
<div>
|
||||||
|
bar
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders an expression in attr as eex" do
|
test "renders an expression in attr as eex" do
|
||||||
|
@ -102,7 +130,12 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s|<div<%= {:safe, Temple.Parser.Utils.build_attr("class", Enum.map([:one, :two], fn x -> x end))} %>><div class="hi"></div></div>|
|
~s"""
|
||||||
|
<div<%= {:safe, Temple.Parser.Utils.build_attr("class", Enum.map([:one, :two], fn x -> x end))} %>>
|
||||||
|
<div class="hi">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders a for comprehension as eex" do
|
test "renders a for comprehension as eex" do
|
||||||
|
@ -113,7 +146,12 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result == ~s{<%= for(x <- 1..5) do %><div class="hi"></div><% end %>}
|
assert result == ~s"""
|
||||||
|
<%= for(x <- 1..5) do %>
|
||||||
|
<div class="hi">
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders an if expression as eex" do
|
test "renders an if expression as eex" do
|
||||||
|
@ -124,7 +162,12 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result == ~s{<%= if(true == false) do %><div class="hi"></div><% end %>}
|
assert result == ~s"""
|
||||||
|
<%= if(true == false) do %>
|
||||||
|
<div class="hi">
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders an if/else expression as eex" do
|
test "renders an if/else expression as eex" do
|
||||||
|
@ -138,7 +181,15 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<%= if(true == false) do %><div class="hi"></div><% else %><div class="haha"></div><% end %>}
|
~s"""
|
||||||
|
<%= if(true == false) do %>
|
||||||
|
<div class="hi">
|
||||||
|
</div>
|
||||||
|
<% else %>
|
||||||
|
<div class="haha">
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders an unless expression as eex" do
|
test "renders an unless expression as eex" do
|
||||||
|
@ -149,7 +200,12 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result == ~s{<%= unless(true == false) do %><div class="hi"></div><% end %>}
|
assert result == ~s"""
|
||||||
|
<%= unless(true == false) do %>
|
||||||
|
<div class="hi">
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders a case expression as eex" do
|
test "renders a case expression as eex" do
|
||||||
|
@ -161,14 +217,12 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
expected =
|
expected = ~S"""
|
||||||
~S"""
|
|
||||||
<%= case(@foo) do %>
|
<%= case(@foo) do %>
|
||||||
<% :baz -> %>
|
<% :baz -> %>
|
||||||
<%= some_component(form: @form) %>
|
<%= some_component(form: @form) %>
|
||||||
<% end %>
|
<% end %>
|
||||||
"""
|
"""
|
||||||
|> String.trim()
|
|
||||||
|
|
||||||
assert result == expected
|
assert result == expected
|
||||||
end
|
end
|
||||||
|
@ -176,70 +230,99 @@ defmodule TempleTest do
|
||||||
test "renders multiline anonymous function with 1 arg before the function" do
|
test "renders multiline anonymous function with 1 arg before the function" do
|
||||||
result =
|
result =
|
||||||
temple do
|
temple do
|
||||||
form_for Routes.user_path(@conn, :create), fn f ->
|
form_for(Routes.user_path(@conn, :create), fn f ->
|
||||||
"Name: "
|
"Name: "
|
||||||
text_input f, :name
|
text_input(f, :name)
|
||||||
end
|
end)
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<%= form_for Routes.user_path(@conn, :create), fn f -> %>Name: <%= text_input(f, :name) %><% end %>}
|
~s"""
|
||||||
|
<%= form_for Routes.user_path(@conn, :create), fn f -> %>
|
||||||
|
Name:
|
||||||
|
<%= text_input(f, :name) %>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders multiline anonymous functions with 2 args before the function" do
|
test "renders multiline anonymous functions with 2 args before the function" do
|
||||||
result =
|
result =
|
||||||
temple do
|
temple do
|
||||||
form_for @changeset, Routes.user_path(@conn, :create), fn f ->
|
form_for(@changeset, Routes.user_path(@conn, :create), fn f ->
|
||||||
"Name: "
|
"Name: "
|
||||||
text_input f, :name
|
text_input(f, :name)
|
||||||
end
|
end)
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<%= form_for @changeset, Routes.user_path(@conn, :create), fn f -> %>Name: <%= text_input(f, :name) %><% end %>}
|
~s"""
|
||||||
|
<%= form_for @changeset, Routes.user_path(@conn, :create), fn f -> %>
|
||||||
|
Name:
|
||||||
|
<%= text_input(f, :name) %>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders multiline anonymous functions with complex nested children" do
|
test "renders multiline anonymous functions with complex nested children" do
|
||||||
result =
|
result =
|
||||||
temple do
|
temple do
|
||||||
form_for @changeset, Routes.user_path(@conn, :create), fn f ->
|
form_for(@changeset, Routes.user_path(@conn, :create), fn f ->
|
||||||
div do
|
div do
|
||||||
"Name: "
|
"Name: "
|
||||||
text_input f, :name
|
text_input(f, :name)
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
end)
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<%= form_for @changeset, Routes.user_path(@conn, :create), fn f -> %><div>Name: <%= text_input(f, :name) %></div><% end %>}
|
~s"""
|
||||||
|
<%= form_for @changeset, Routes.user_path(@conn, :create), fn f -> %>
|
||||||
|
<div>
|
||||||
|
Name:
|
||||||
|
<%= text_input(f, :name) %>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders multiline anonymous function with 3 arg before the function" do
|
test "renders multiline anonymous function with 3 arg before the function" do
|
||||||
result =
|
result =
|
||||||
temple do
|
temple do
|
||||||
form_for @changeset, Routes.user_path(@conn, :create), [foo: :bar], fn f ->
|
form_for(@changeset, Routes.user_path(@conn, :create), [foo: :bar], fn f ->
|
||||||
"Name: "
|
"Name: "
|
||||||
text_input f, :name
|
text_input(f, :name)
|
||||||
end
|
end)
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<%= form_for @changeset, Routes.user_path(@conn, :create), [foo: :bar], fn f -> %>Name: <%= text_input(f, :name) %><% end %>}
|
~s"""
|
||||||
|
<%= form_for @changeset, Routes.user_path(@conn, :create), [foo: :bar], fn f -> %>
|
||||||
|
Name:
|
||||||
|
<%= text_input(f, :name) %>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "renders multiline anonymous function with 1 arg before the function and 1 arg after" do
|
test "renders multiline anonymous function with 1 arg before the function and 1 arg after" do
|
||||||
result =
|
result =
|
||||||
temple do
|
temple do
|
||||||
form_for @changeset,
|
form_for(
|
||||||
|
@changeset,
|
||||||
fn f ->
|
fn f ->
|
||||||
"Name: "
|
"Name: "
|
||||||
text_input f, :name
|
text_input(f, :name)
|
||||||
end,
|
end,
|
||||||
foo: :bar
|
foo: :bar
|
||||||
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<%= form_for @changeset, fn f -> %>Name: <%= text_input(f, :name) %><% end, [foo: :bar] %>}
|
~s"""
|
||||||
|
<%= form_for @changeset, fn f -> %>
|
||||||
|
Name:
|
||||||
|
<%= text_input(f, :name) %>
|
||||||
|
<% end, [foo: :bar] %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "tags prefixed with Temple. should be interpreted as temple tags" do
|
test "tags prefixed with Temple. should be interpreted as temple tags" do
|
||||||
|
@ -252,7 +335,13 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result == ~s{<div><span>bob</span></div>}
|
assert result == ~s"""
|
||||||
|
<div>
|
||||||
|
<span>
|
||||||
|
bob
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "can pass do as an arg instead of a block" do
|
test "can pass do as an arg instead of a block" do
|
||||||
|
@ -267,7 +356,17 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<div class="font-bold">Hello, world</div><div class="font-bold">Hello, world</div><div>Hello, world</div>}
|
~s"""
|
||||||
|
<div class="font-bold">
|
||||||
|
Hello, world
|
||||||
|
</div>
|
||||||
|
<div class="font-bold">
|
||||||
|
Hello, world
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
Hello, world
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "for with 2 generators" do
|
test "for with 2 generators" do
|
||||||
|
@ -280,7 +379,16 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<%= for(x <- 1..5, y <- 6..10) do %><div><%= x %></div><div><%= y %></div><% end %>}
|
~s"""
|
||||||
|
<%= for(x <- 1..5, y <- 6..10) do %>
|
||||||
|
<div>
|
||||||
|
<%= x %>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<%= y %>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "can pass an expression as assigns" do
|
test "can pass an expression as assigns" do
|
||||||
|
@ -292,7 +400,15 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<fieldset<%= Temple.Parser.Utils.runtime_attrs(if(true == false) do [disabled: true]else []end) %>><input type="text"></fieldset>}
|
~s"""
|
||||||
|
<fieldset<%= Temple.Parser.Utils.runtime_attrs(if(true == false) do
|
||||||
|
[disabled: true]
|
||||||
|
else
|
||||||
|
[]
|
||||||
|
end) %>>
|
||||||
|
<input type="text">
|
||||||
|
</fieldset>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "can pass a variable as assigns" do
|
test "can pass a variable as assigns" do
|
||||||
|
@ -304,7 +420,11 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<fieldset<%= Temple.Parser.Utils.runtime_attrs(foo_bar) %>><input type="text"></fieldset>}
|
~s"""
|
||||||
|
<fieldset<%= Temple.Parser.Utils.runtime_attrs(foo_bar) %>>
|
||||||
|
<input type="text">
|
||||||
|
</fieldset>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "can pass a function as assigns" do
|
test "can pass a function as assigns" do
|
||||||
|
@ -316,7 +436,11 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert result ==
|
assert result ==
|
||||||
~s{<fieldset<%= Temple.Parser.Utils.runtime_attrs(Foo.foo_bar()) %>><input type="text"></fieldset>}
|
~s"""
|
||||||
|
<fieldset<%= Temple.Parser.Utils.runtime_attrs(Foo.foo_bar()) %>>
|
||||||
|
<input type="text">
|
||||||
|
</fieldset>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "hr tag works" do
|
test "hr tag works" do
|
||||||
|
@ -334,7 +458,23 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
assert evaluate_template(result, assigns) ==
|
assert evaluate_template(result, assigns) ==
|
||||||
~s{<div>foo</div><hr><div>foo</div><hr class="foofoo"><div>bar</div><hr class="foofoo"><div>bar</div>}
|
~s"""
|
||||||
|
<div>
|
||||||
|
foo
|
||||||
|
</div>
|
||||||
|
<hr>
|
||||||
|
<div>
|
||||||
|
foo
|
||||||
|
</div>
|
||||||
|
<hr class="foofoo">
|
||||||
|
<div>
|
||||||
|
bar
|
||||||
|
</div>
|
||||||
|
<hr class="foofoo">
|
||||||
|
<div>
|
||||||
|
bar
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
test "boolean attributes" do
|
test "boolean attributes" do
|
||||||
|
@ -363,6 +503,10 @@ defmodule TempleTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
assert evaluate_template(result, assigns) == ~s{<div class="text-red">\nfoo\n\n</div>}
|
assert evaluate_template(result, assigns) == ~s"""
|
||||||
|
<div class="text-red">
|
||||||
|
foo
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
46
test/whitespace_test.exs
Normal file
46
test/whitespace_test.exs
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
defmodule Temple.WhitespaceTest do
|
||||||
|
use ExUnit.Case, async: true
|
||||||
|
|
||||||
|
import Temple
|
||||||
|
|
||||||
|
alias Temple.Support.Utils
|
||||||
|
|
||||||
|
test "only emits a single new line" do
|
||||||
|
result =
|
||||||
|
temple do
|
||||||
|
div class: "hello" do
|
||||||
|
span id: "foo" do
|
||||||
|
"Howdy, "
|
||||||
|
end
|
||||||
|
|
||||||
|
div class: "hi" do
|
||||||
|
"Jim Bob"
|
||||||
|
end
|
||||||
|
|
||||||
|
c WhoaNelly, foo: "bar" do
|
||||||
|
slot :silver do
|
||||||
|
"esketit"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|> Utils.append_new_line()
|
||||||
|
|
||||||
|
expected = ~s"""
|
||||||
|
<div class="hello">
|
||||||
|
<span id="foo">
|
||||||
|
Howdy,
|
||||||
|
</span>
|
||||||
|
<div class="hi">
|
||||||
|
Jim Bob
|
||||||
|
</div>
|
||||||
|
<%= Temple.Component.__component__ WhoaNelly, [foo: "bar"] do %>
|
||||||
|
<% {:silver, %{}} -> %>
|
||||||
|
esketit
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
|
"""
|
||||||
|
|
||||||
|
assert result == expected
|
||||||
|
end
|
||||||
|
end
|
Reference in a new issue