philomena/lib/philomena_media/processors/gif.ex

168 lines
3.8 KiB
Elixir

defmodule PhilomenaMedia.Processors.Gif do
@moduledoc false
alias PhilomenaMedia.Features
alias PhilomenaMedia.Intensities
alias PhilomenaMedia.Analyzers.Result
alias PhilomenaMedia.Remote
alias PhilomenaMedia.Processors.Processor
alias PhilomenaMedia.Processors
@behaviour Processor
@spec versions(Processors.version_list()) :: [Processors.version_filename()]
def versions(sizes) do
sizes
|> Enum.map(fn {name, _} -> "#{name}.gif" end)
|> Kernel.++(["full.webm", "full.mp4", "rendered.png"])
end
@spec process(Result.t(), Path.t(), Processors.version_list()) :: Processors.edit_script()
def process(analysis, file, versions) do
duration = analysis.duration
preview = preview(duration, file)
palette = palette(file)
{:ok, intensities} = Intensities.file(preview)
{:ok, features} = Features.file(preview)
scaled = Enum.flat_map(versions, &scale(palette, file, &1))
videos = generate_videos(file)
[
intensities: intensities,
features: features,
thumbnails: scaled ++ videos ++ [{:copy, preview, "rendered.png"}]
]
end
@spec post_process(Result.t(), Path.t()) :: Processors.edit_script()
def post_process(_analysis, file) do
[replace_original: optimize(file)]
end
@spec features(Result.t(), Path.t()) :: Features.t()
def features(analysis, file) do
{:ok, features} = Features.file(preview(analysis.duration, file))
features
end
@spec intensities(Result.t(), Path.t()) :: Intensities.t()
def intensities(analysis, file) do
{:ok, intensities} = Intensities.file(preview(analysis.duration, file))
intensities
end
defp optimize(file) do
optimized = Briefly.create!(extname: ".gif")
{_output, 0} = Remote.cmd("gifsicle", ["--careful", "-O2", file, "-o", optimized])
optimized
end
defp preview(duration, file) do
preview = Briefly.create!(extname: ".png")
{_output, 0} = Remote.cmd("mediathumb", [file, to_string(duration / 2), preview])
preview
end
defp palette(file) do
palette = Briefly.create!(extname: ".png")
{_output, 0} =
Remote.cmd("ffmpeg", [
"-loglevel",
"0",
"-y",
"-i",
file,
"-vf",
"palettegen=stats_mode=diff",
palette
])
palette
end
defp scale(palette, file, {thumb_name, {width, height}}) do
scaled = Briefly.create!(extname: ".gif")
scale_filter = "scale=w=#{width}:h=#{height}:force_original_aspect_ratio=decrease"
palette_filter =
"paletteuse=dither=bayer:bayer_scale=5:diff_mode=rectangle:alpha_threshold=255"
filter_graph = "[0:v]#{scale_filter}[x];[x][1:v]#{palette_filter}"
{_output, 0} =
Remote.cmd("ffmpeg", [
"-loglevel",
"0",
"-y",
"-i",
file,
"-i",
palette,
"-lavfi",
filter_graph,
scaled
])
[{:copy, scaled, "#{thumb_name}.gif"}]
end
defp generate_videos(file) do
webm = Briefly.create!(extname: ".webm")
mp4 = Briefly.create!(extname: ".mp4")
{_output, 0} =
Remote.cmd("ffmpeg", [
"-loglevel",
"0",
"-y",
"-i",
file,
"-pix_fmt",
"yuv420p",
"-c:v",
"libvpx",
"-deadline",
"good",
"-b:v",
"5M",
webm
])
{_output, 0} =
Remote.cmd("ffmpeg", [
"-loglevel",
"0",
"-y",
"-i",
file,
"-vf",
"scale=ceil(iw/2)*2:ceil(ih/2)*2",
"-c:v",
"libx264",
"-preset",
"medium",
"-pix_fmt",
"yuv420p",
"-profile:v",
"main",
"-crf",
"18",
"-b:v",
"5M",
mp4
])
[
{:copy, webm, "full.webm"},
{:copy, mp4, "full.mp4"}
]
end
end