Skip to content

Commit 896bf7f

Browse files
authored
feat!: pass assigns to post/page extension rendering (#102)
This allows you to use dynamic data in your content files if your converter supports it. Closes #99 BREAKING-CHANGE: Posts no longer fallback to the first <h1> tag as the title of the post
1 parent 0d9959f commit 896bf7f

File tree

13 files changed

+122
-131
lines changed

13 files changed

+122
-131
lines changed

lib/tableau/converter.ex

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,6 @@ defmodule Tableau.Converter do
88
99
Is given the file path, the content of the files (sans front matter), the front matter, and a list of options.
1010
"""
11-
@callback convert(filepath :: String.t(), content :: String.t(), front_matter :: map(), opts :: Keyword.t()) ::
11+
@callback convert(filepath :: String.t(), front_matter :: map(), content :: String.t(), opts :: Keyword.t()) ::
1212
String.t()
1313
end

lib/tableau/converters/mdex_converter.ex

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ defmodule Tableau.MDExConverter do
22
@moduledoc """
33
Converter to parse markdown content with `MDEx`
44
"""
5-
def convert(_filepath, body, _front_matter, _opts) do
5+
def convert(_filepath, _front_matter, body, _opts) do
66
{:ok, config} = Tableau.Config.get()
77

88
MDEx.to_html!(body, config.markdown[:mdex])

lib/tableau/extensions/common.ex

+2-5
Original file line numberDiff line numberDiff line change
@@ -5,14 +5,11 @@ defmodule Tableau.Extension.Common do
55
wildcard |> Path.wildcard() |> Enum.sort()
66
end
77

8-
def entries(paths, builder, opts) do
8+
def entries(paths, callback) do
99
for path <- paths do
1010
{front_matter, body} = Tableau.YamlFrontMatter.parse!(File.read!(path), atoms: true)
1111
"." <> ext = Path.extname(path)
12-
converter = opts[:converters][String.to_atom(ext)]
13-
body = converter.convert(path, body, front_matter, opts)
14-
15-
builder.build(path, front_matter, body)
12+
callback.(%{path: path, ext: String.to_atom(ext), front_matter: front_matter, pre_convert_body: body})
1613
end
1714
end
1815
end

lib/tableau/extensions/page_extension.ex

+25-6
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
defmodule Tableau.PageExtension do
22
@moduledoc """
3-
Markdown files (with YAML frontmatter) in the configured pages directory will be automatically compiled into Tableau pages.
3+
Content files (with YAML frontmatter) in the configured pages directory will be automatically compiled into Tableau pages.
44
55
Certain frontmatter keys are required and all keys are passed as options to the `Tableau.Page`.
66
@@ -44,7 +44,7 @@ defmodule Tableau.PageExtension do
4444
layout: "MyApp.PageLayout"
4545
```
4646
47-
## Other markup formats
47+
## Content formats
4848
4949
If you're interested in authoring your content in something other than markdown (or you want to use a different markdown parser), you can configure
5050
a converter for your format in the global configuration.
@@ -63,20 +63,39 @@ defmodule Tableau.PageExtension do
6363

6464
use Tableau.Extension, key: :pages, type: :pre_build, priority: 100
6565

66+
alias Tableau.Extension.Common
67+
alias Tableau.PageExtension.Page
68+
69+
@config Map.new(Application.compile_env(:tableau, Tableau.PageExtension, %{}))
70+
6671
def run(token) do
67-
pages = Tableau.PageExtension.Pages.pages()
72+
{:ok, config} = Tableau.PageExtension.Config.new(@config)
73+
74+
{:ok, %{converters: converters}} = Tableau.Config.get()
75+
76+
exts = Enum.map_join(converters, ",", fn {ext, _} -> to_string(ext) end)
77+
78+
pages =
79+
config.dir
80+
|> Path.join("**/*.{#{exts}}")
81+
|> Common.paths()
82+
|> Common.entries(fn %{path: path, ext: ext, front_matter: front_matter, pre_convert_body: pre_convert_body} ->
83+
renderer = fn assigns -> converters[ext].convert(path, front_matter, pre_convert_body, assigns) end
84+
85+
{Page.build(path, front_matter, pre_convert_body), renderer}
86+
end)
6887

6988
graph =
7089
Tableau.Graph.insert(
7190
token.graph,
72-
Enum.map(pages, fn page ->
73-
%Tableau.Page{parent: page.layout, permalink: page.permalink, template: page.body, opts: page}
91+
Enum.map(pages, fn {page, renderer} ->
92+
%Tableau.Page{parent: page.layout, permalink: page.permalink, template: renderer, opts: page}
7493
end)
7594
)
7695

7796
{:ok,
7897
token
79-
|> Map.put(:pages, pages)
98+
|> Map.put(:pages, pages |> Enum.unzip() |> elem(0))
8099
|> Map.put(:graph, graph)}
81100
end
82101
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
defmodule Tableau.PageExtension.Page do
2+
@moduledoc false
3+
def build(filename, front_matter, body) do
4+
{:ok, page_config} =
5+
Tableau.PageExtension.Config.new(Map.new(Application.get_env(:tableau, Tableau.PageExtension, %{})))
6+
7+
front_matter
8+
|> Map.put(:__tableau_page_extension__, true)
9+
|> Map.put(:body, body)
10+
|> Map.put(:file, filename)
11+
|> Map.put(:layout, Module.concat([front_matter.layout || page_config.layout]))
12+
|> build_permalink(page_config)
13+
end
14+
15+
defp build_permalink(%{permalink: permalink} = front_matter, _config) do
16+
permalink
17+
|> transform_permalink(front_matter)
18+
|> then(&Map.put(front_matter, :permalink, &1))
19+
end
20+
21+
defp build_permalink(front_matter, %{permalink: permalink}) when not is_nil(permalink) do
22+
permalink
23+
|> transform_permalink(front_matter)
24+
|> then(&Map.put(front_matter, :permalink, &1))
25+
end
26+
27+
defp build_permalink(%{file: filename} = front_matter, config) do
28+
filename
29+
|> Path.rootname()
30+
|> String.replace_prefix(config.dir, "")
31+
|> transform_permalink(front_matter)
32+
|> then(&Map.put(front_matter, :permalink, &1))
33+
end
34+
35+
defp transform_permalink(path, front_matter) do
36+
vars = Map.new(front_matter, fn {k, v} -> {":#{k}", v} end)
37+
38+
path
39+
|> String.replace(Map.keys(vars), &to_string(Map.fetch!(vars, &1)))
40+
|> String.replace(" ", "-")
41+
|> String.replace("_", "-")
42+
|> String.replace(~r/[^[:alnum:]\/\-.]/, "")
43+
|> String.downcase()
44+
end
45+
end

lib/tableau/extensions/page_extension/pages.ex

-23
This file was deleted.

lib/tableau/extensions/page_extension/pages/page.ex

-53
This file was deleted.

lib/tableau/extensions/post_extension.ex

+33-7
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,14 @@
11
defmodule Tableau.PostExtension do
22
@moduledoc """
3-
Markdown files (with YAML frontmatter) in the configured posts directory will be automatically compiled into Tableau pages.
3+
Content files (with YAML frontmatter) in the configured posts directory will be automatically compiled into Tableau pages.
44
55
Certain frontmatter keys are required and all keys are passed as options to the `Tableau.Page`.
66
77
## Options
88
99
Frontmatter is compiled with `yaml_elixir` and all keys are converted to atoms.
1010
11-
* `:title` - The title of the post. Falls back to the first `<h1>` tag if present in the body.
11+
* `:title` - The title of the post.
1212
* `:permalink` - The permalink of the post. `:title` will be replaced with the posts title and non alphanumeric characters removed. Optional.
1313
* `:date` - A string representation of an Elixir `NaiveDateTime`, often presented as a `sigil_N`. This will be converted to your configured timezone.
1414
* `:layout` - A string representation of a Tableau layout module.
@@ -51,7 +51,7 @@ defmodule Tableau.PostExtension do
5151
```
5252
5353
54-
## Other markup formats
54+
## Content formats
5555
5656
If you're interested in authoring your content in something other than markdown (or you want to use a different markdown parser), you can configure
5757
a converter for your format in the global configuration.
@@ -70,20 +70,46 @@ defmodule Tableau.PostExtension do
7070

7171
use Tableau.Extension, key: :posts, type: :pre_build, priority: 100
7272

73+
alias Tableau.Extension.Common
74+
alias Tableau.PostExtension.Post
75+
76+
@config Map.new(Application.compile_env(:tableau, Tableau.PostExtension, %{}))
77+
7378
def run(token) do
74-
posts = Tableau.PostExtension.Posts.posts()
79+
{:ok, config} = Tableau.PostExtension.Config.new(@config)
80+
81+
{:ok, %{converters: converters}} = Tableau.Config.get()
82+
83+
exts = Enum.map_join(converters, ",", fn {ext, _} -> to_string(ext) end)
84+
85+
posts =
86+
config.dir
87+
|> Path.join("**/*.{#{exts}}")
88+
|> Common.paths()
89+
|> Common.entries(fn %{path: path, ext: ext, front_matter: front_matter, pre_convert_body: pre_convert_body} ->
90+
renderer = fn assigns -> converters[ext].convert(path, front_matter, pre_convert_body, assigns) end
91+
92+
{Post.build(path, front_matter, pre_convert_body), renderer}
93+
end)
94+
|> then(fn posts ->
95+
if config.future do
96+
posts
97+
else
98+
Enum.reject(posts, fn {post, _} -> DateTime.after?(post.date, DateTime.utc_now()) end)
99+
end
100+
end)
75101

76102
graph =
77103
Tableau.Graph.insert(
78104
token.graph,
79-
Enum.map(posts, fn post ->
80-
%Tableau.Page{parent: post.layout, permalink: post.permalink, template: post.body, opts: post}
105+
Enum.map(posts, fn {post, renderer} ->
106+
%Tableau.Page{parent: post.layout, permalink: post.permalink, template: renderer, opts: post}
81107
end)
82108
)
83109

84110
{:ok,
85111
token
86-
|> Map.put(:posts, posts)
112+
|> Map.put(:posts, posts |> Enum.unzip() |> elem(0))
87113
|> Map.put(:graph, graph)}
88114
end
89115
end

lib/tableau/extensions/post_extension/posts/post.ex lib/tableau/extensions/post_extension/post.ex

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
defmodule Tableau.PostExtension.Posts.Post do
1+
defmodule Tableau.PostExtension.Post do
22
@moduledoc false
33
def build(filename, attrs, body) do
44
{:ok, post_config} =

lib/tableau/extensions/post_extension/posts.ex

-31
This file was deleted.

lib/tableau/lazy_page.ex

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
defmodule Tableau.LazyPage do
2+
@moduledoc false
3+
defstruct [:handler]
4+
5+
def run(lazy_page, assigns) do
6+
lazy_page.handler.(assigns)
7+
end
8+
end

lib/tableau/page.ex

+4-1
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,10 @@ defmodule Tableau.Page do
3232
defstruct [:parent, :permalink, :template, :opts]
3333

3434
defimpl Tableau.Graph.Nodable do
35-
def template(nodable, _assigns), do: nodable.template
35+
def template(nodable, assigns) do
36+
nodable.template.(assigns)
37+
end
38+
3639
def type(_nodable), do: {:ok, :page}
3740
def parent(nodable), do: {:ok, nodable.parent}
3841
def opts(nodable), do: nodable.opts

test/tableau/extensions/post_extension/posts/post_test.exs test/tableau/extensions/post_extension/post_test.exs

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
1-
defmodule Tableau.PostExtension.Posts.PostTest do
1+
defmodule Tableau.PostExtension.PostTest do
22
use ExUnit.Case, async: true
33

4-
alias Tableau.PostExtension.Posts.Post
4+
alias Tableau.PostExtension.Post
55

66
describe "build/3" do
77
test "substitutes arbitrary front matter into permalink" do

0 commit comments

Comments
 (0)