Skip to content

Commit 162420b

Browse files
committed
feat: PageExtension
The page extension allows you to create arbitrary pages with documents like markdown.
1 parent 5a4a20d commit 162420b

File tree

4 files changed

+179
-0
lines changed

4 files changed

+179
-0
lines changed
+102
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
defmodule Tableau.PageExtension do
2+
@moduledoc """
3+
Markdown files (with YAML frontmatter) in the configured pages directory will be automatically compiled into Tableau pages.
4+
5+
Certain frontmatter keys are required and all keys are passed as options to the `Tableau.Page`.
6+
7+
## Options
8+
9+
Frontmatter is compiled with `yaml_elixir` and all keys are converted to atoms.
10+
11+
* `:id` - An Elixir module to be used when compiling the backing `Tableau.Page`. A leaking implementation detail that should be fixed eventually.
12+
* `:title` - The title of the page
13+
* `:permalink` - The permalink of the page.
14+
* `:layout` - A string representation of a Tableau layout module.
15+
16+
## Example
17+
18+
```yaml
19+
id: "GettingStarted"
20+
title: "Getting Started"
21+
permalink: "/docs/:title"
22+
layout: "ElixirTools.PageLayout"
23+
```
24+
25+
## Permalink
26+
27+
The permalink is a string with colon prefixed template variables.
28+
29+
These variables will be swapped with the corresponding YAML Frontmatter key, with the result being piped through `to_string/1`.
30+
31+
## Configuration
32+
33+
- `:enabled` - boolean - Extension is active or not.
34+
- `:dir` - string - Directory to scan for markdown files. Defaults to `_pages`
35+
- `:permalink` - string - Default output path for pages. Accepts `:title` as a replacement keyword, replaced with the page's provided title. If a page has a `:permalink` provided, that will override this value _for that page_.
36+
- `:layout` - string - Elixir module providing page layout for pages. Default is nil.
37+
38+
### Example
39+
40+
```elixir
41+
config :tableau, Tableau.PageExtension,
42+
enabled: true,
43+
dir: "_docs",
44+
permalink: "/docs/:title",
45+
layout: "MyApp.PageLayout"
46+
```
47+
"""
48+
49+
use Tableau.Extension, key: :pages, type: :pre_build, priority: 100
50+
51+
{:ok, config} =
52+
Tableau.PageExtension.Config.new(Map.new(Application.compile_env(:tableau, Tableau.PageExtension, %{})))
53+
54+
@config config
55+
56+
def run(token) do
57+
:global.trans(
58+
{:create_pages_module, make_ref()},
59+
fn ->
60+
Module.create(
61+
Tableau.PageExtension.Pages,
62+
quote do
63+
use NimblePublisher,
64+
build: __MODULE__.Page,
65+
from: "#{unquote(@config.dir)}/**/*.md",
66+
as: :pages,
67+
parser: Tableau.PageExtension.Pages.Page,
68+
html_converter: Tableau.PageExtension.Pages.HTMLConverter
69+
70+
def pages(_opts \\ []) do
71+
@pages
72+
end
73+
end,
74+
Macro.Env.location(__ENV__)
75+
)
76+
77+
pages =
78+
for page <- apply(Tableau.PageExtension.Pages, :pages, []) do
79+
{:module, _module, _binary, _term} =
80+
Module.create(
81+
Module.concat([page.id]),
82+
quote do
83+
use Tableau.Page, unquote(Macro.escape(Keyword.new(page)))
84+
85+
@external_resource unquote(page.file)
86+
def template(_assigns) do
87+
unquote(page.body)
88+
end
89+
end,
90+
Macro.Env.location(__ENV__)
91+
)
92+
93+
page
94+
end
95+
96+
{:ok, Map.put(token, :pages, pages)}
97+
end,
98+
[Node.self()],
99+
:infinity
100+
)
101+
end
102+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
defmodule Tableau.PageExtension.Config do
2+
@moduledoc false
3+
4+
import Schematic
5+
6+
defstruct enabled: true, dir: "_pages", permalink: nil, layout: nil
7+
8+
def new(input), do: unify(schematic(), input)
9+
10+
def schematic do
11+
schema(
12+
__MODULE__,
13+
%{
14+
optional(:enabled) => bool(),
15+
optional(:dir) => str(),
16+
optional(:permalink) => str(),
17+
optional(:layout) => str()
18+
},
19+
convert: false
20+
)
21+
end
22+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
defmodule Tableau.PageExtension.Pages.HTMLConverter do
2+
@moduledoc false
3+
def convert(_filepath, body, _attrs, _opts) do
4+
{:ok, config} = Tableau.Config.new(Map.new(Application.get_env(:tableau, :config, %{})))
5+
6+
MDEx.to_html(body, config.markdown[:mdex])
7+
end
8+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
defmodule Tableau.PageExtension.Pages.Page do
2+
@moduledoc false
3+
def build(filename, attrs, body) do
4+
{:ok, page_config} =
5+
Tableau.PageExtension.Config.new(Map.new(Application.get_env(:tableau, Tableau.PageExtension, %{})))
6+
7+
attrs
8+
|> Map.put(:body, body)
9+
|> Map.put(:file, filename)
10+
|> Map.put(:layout, Module.concat([attrs.layout || page_config.layout]))
11+
|> build_permalink(page_config)
12+
end
13+
14+
def parse(_file_path, content) do
15+
Tableau.YamlFrontMatter.parse!(content, atoms: true)
16+
end
17+
18+
defp build_permalink(%{permalink: permalink} = attrs, _config) do
19+
permalink
20+
|> transform_permalink(attrs)
21+
|> then(&Map.put(attrs, :permalink, &1))
22+
end
23+
24+
defp build_permalink(attrs, %{permalink: permalink}) when not is_nil(permalink) do
25+
permalink
26+
|> transform_permalink(attrs)
27+
|> then(&Map.put(attrs, :permalink, &1))
28+
end
29+
30+
defp build_permalink(%{file: filename} = attrs, config) do
31+
filename
32+
|> Path.rootname()
33+
|> String.replace_prefix(config.dir, "")
34+
|> transform_permalink(attrs)
35+
|> then(&Map.put(attrs, :permalink, &1))
36+
end
37+
38+
defp transform_permalink(path, attrs) do
39+
vars = Map.new(attrs, fn {k, v} -> {":#{k}", v} end)
40+
41+
path
42+
|> String.replace(Map.keys(vars), &to_string(Map.fetch!(vars, &1)))
43+
|> String.replace(" ", "-")
44+
|> String.replace(~r/[^[:alnum:]\/\-]/, "")
45+
|> String.downcase()
46+
end
47+
end

0 commit comments

Comments
 (0)