|
| 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 |
0 commit comments