File: makeup.ex

package info (click to toggle)
elixir-makeup 1.2.1-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 384 kB
  • sloc: javascript: 24; makefile: 9
file content (82 lines) | stat: -rw-r--r-- 2,703 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
defmodule Makeup do
  @moduledoc """
  Syntax highlighting library for code, inspired by Pygments.

  By default, it doesn't include any lexers. You must import
  them separately (even the Elixir lexer).
  """
  alias Makeup.Formatters.HTML.HTMLFormatter
  alias Makeup.Lexers.ElixirLexer
  alias Makeup.Registry

  @doc """
  Highlights the given string using the given lexer and formatter.

  By default it highlight the Elixir language using
  `Makeup.Formatters.HTML.HTMLFormatter`.

  ## Options:

  - `:lexer` - module name of the lexer to use (default: `Makeup.Lexers.ElixirLexer`)
  - `:lexer_options` - list of options for the lexer
  - `:formatter` - module name of the formatter to use (defult: `Makeup.Formatters.HTML.HTMLFormatter`)
  - `:formatter_options` - list of options for the formatter. For the included HTMLFormatter, that's:
    - `:css_class` - css class(es) of the main `<pre>` element (default: `"highlight"`)
    - `:highlight_tag` - tag that wraps every token (default: `"span"`)
  """
  def highlight(source, options \\ []) do
    {lexer, lexer_options} = fetch_lexer(options)

    formatter =
      case options[:formatter] do
        nil -> HTMLFormatter
        module when is_atom(module) -> module
      end

    formatter_options = Keyword.get(options, :formatter_options, [])
    tokens = lexer.lex(source, lexer_options)
    formatter.format_as_binary(tokens, formatter_options)
  end

  @doc """
  Convenience for formatting as the inner bits of HTML.
  """
  def highlight_inner_html(source, options \\ []) do
    {lexer, lexer_options} = fetch_lexer(options)
    formatter_options = Keyword.get(options, :formatter_options, [])

    tokens = lexer.lex(source, lexer_options)
    HTMLFormatter.format_inner_as_binary(tokens, formatter_options)
  end

  defp fetch_lexer(options) do
    {lexer, lexer_options} =
      case options[:lexer] do
        nil -> {ElixirLexer, []}
        module when is_atom(module) -> {module, []}
        name -> Registry.fetch_lexer_by_name!(name)
      end

    {lexer, Keyword.merge(lexer_options, Keyword.get(options, :lexer_options, []))}
  end

  @doc """
  Generates a CSS stylesheet for highlighted code for the given style.

  It expects a `style`, either as an atom name or as "style map",
  and the `css_class` as the top level class for highlighted code.

  For example, if the `css_class` is `"highlight"` (the default), the stylesheet
  has the form:

  ```css
  .highlight .someclass {...}
  .highlight .anotherclass {...}
  ```

  See `Makeup.Styles.HTML.StyleMap` for all style maps.
  """
  def stylesheet(style \\ :default_style, css_class \\ "highlight") do
    HTMLFormatter.stylesheet(style, css_class)
  end
end