mirror of
https://github.com/livebook-dev/livebook.git
synced 2024-11-11 18:09:58 +08:00
90e7941fe4
* Update cell actions * Add new focus indicator * Update headings typography * Update cell actions and insert buttons * Add sidebar menu * Add settings modal * Update homepage * Update settings dialog * Rename classes * Add floating menu * Update icon colors on hover * Fix homepage tests * Format assets source * Update monaco editor * Fix editor width on resize * Add more padding to the notebook content * Update settings dialog title * Show reevaluate button when the cell is in evaluated state * Show section actions on focus or hover only * Pre-fill runtime selector with the current configuration * Ignore cmd + enter in Markdown cells
73 lines
2.4 KiB
Elixir
73 lines
2.4 KiB
Elixir
defmodule LivebookWeb.MemoryProvider do
|
|
@moduledoc false
|
|
|
|
@gzippable_exts ~w(.js .css .txt .text .html .json .svg .eot .ttf)
|
|
|
|
# Configurable implementation of `LivebookWeb.StaticPlug.Provider` behaviour,
|
|
# that bundles the files into the module compiled source.
|
|
#
|
|
# ## `use` options
|
|
#
|
|
# * `:from` (**required**) - where to read the static files from. See `Plug.Static` for more details.
|
|
#
|
|
# * `:gzip` - whether to bundle gzipped version of the files,
|
|
# in which case the uncompressed files are not included. Defaults to `false`.
|
|
|
|
defmacro __using__(opts) do
|
|
quote bind_quoted: [opts: opts] do
|
|
@behaviour LivebookWeb.StaticPlug.Provider
|
|
|
|
from = Keyword.fetch!(opts, :from)
|
|
static_path = LivebookWeb.StaticPlug.Provider.static_path(from)
|
|
paths = LivebookWeb.MemoryProvider.__paths__(static_path)
|
|
files = LivebookWeb.MemoryProvider.__preload_files__!(static_path, paths, opts)
|
|
|
|
for path <- paths do
|
|
abs_path = Path.join(static_path, path)
|
|
@external_resource Path.relative_to_cwd(abs_path)
|
|
end
|
|
|
|
@impl true
|
|
def get_file(segments, compression)
|
|
|
|
for {segments, compression, file} <- files do
|
|
def get_file(unquote(segments), unquote(compression)), do: unquote(Macro.escape(file))
|
|
end
|
|
|
|
def get_file(_, _), do: nil
|
|
|
|
# Force recompilation if the static files change.
|
|
def __mix_recompile__? do
|
|
current_paths = LivebookWeb.MemoryProvider.__paths__(unquote(static_path))
|
|
:erlang.md5(current_paths) != unquote(:erlang.md5(paths))
|
|
end
|
|
end
|
|
end
|
|
|
|
def __preload_files__!(static_path, paths, opts) do
|
|
gzip? = Keyword.get(opts, :gzip, false)
|
|
|
|
Enum.map(paths, fn path ->
|
|
segments = Path.split(path)
|
|
abs_path = Path.join(static_path, path)
|
|
content = File.read!(abs_path)
|
|
digest = content |> :erlang.md5() |> Base.encode16(case: :lower)
|
|
|
|
if gzip? and Path.extname(path) in @gzippable_exts do
|
|
gzipped_content = :zlib.gzip(content)
|
|
|
|
{segments, :gzip, %LivebookWeb.StaticPlug.File{content: gzipped_content, digest: digest}}
|
|
else
|
|
{segments, nil, %LivebookWeb.StaticPlug.File{content: content, digest: digest}}
|
|
end
|
|
end)
|
|
end
|
|
|
|
def __paths__(static_path) do
|
|
Path.join(static_path, "**")
|
|
|> Path.wildcard()
|
|
|> Enum.reject(&File.dir?/1)
|
|
|> Enum.map(&String.replace_leading(&1, static_path <> "/", ""))
|
|
|> Enum.sort()
|
|
end
|
|
end
|