mirror of
https://github.com/kovidgoyal/kitty.git
synced 2025-12-13 20:36:22 +01:00
711 lines
20 KiB
Go
711 lines
20 KiB
Go
// License: GPLv3 Copyright: 2023, Kovid Goyal, <kovid at kovidgoyal.net>
|
|
|
|
package images
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"image"
|
|
"image/color"
|
|
"image/gif"
|
|
"io"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"slices"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/kovidgoyal/kitty/tools/utils"
|
|
"github.com/kovidgoyal/kitty/tools/utils/shm"
|
|
|
|
"github.com/kovidgoyal/exiffix"
|
|
"github.com/kovidgoyal/imaging"
|
|
)
|
|
|
|
var _ = fmt.Print
|
|
|
|
const TempTemplate = "kitty-tty-graphics-protocol-*"
|
|
|
|
func CreateTemp() (*os.File, error) {
|
|
return os.CreateTemp("", TempTemplate)
|
|
}
|
|
|
|
func CreateTempInRAM() (*os.File, error) {
|
|
if shm.SHM_DIR != "" {
|
|
f, err := os.CreateTemp(shm.SHM_DIR, TempTemplate)
|
|
if err == nil {
|
|
return f, err
|
|
}
|
|
}
|
|
return CreateTemp()
|
|
}
|
|
|
|
type ImageFrame struct {
|
|
Width, Height, Left, Top int
|
|
Number int // 1-based number
|
|
Compose_onto int // number of frame to compose onto
|
|
Delay_ms int32 // negative for gapless frame, zero ignored, positive is number of ms
|
|
Is_opaque bool
|
|
Img image.Image
|
|
}
|
|
|
|
type SerializableImageFrame struct {
|
|
Width, Height, Left, Top int
|
|
Number int // 1-based number
|
|
Compose_onto int // number of frame to compose onto
|
|
Delay_ms int // negative for gapless frame, zero ignored, positive is number of ms
|
|
Is_opaque bool
|
|
Size int
|
|
}
|
|
|
|
func (s *ImageFrame) Serialize() SerializableImageFrame {
|
|
return SerializableImageFrame{
|
|
Width: s.Width, Height: s.Height, Left: s.Left, Top: s.Top,
|
|
Number: s.Number, Compose_onto: s.Compose_onto, Delay_ms: int(s.Delay_ms),
|
|
Is_opaque: s.Is_opaque,
|
|
}
|
|
}
|
|
|
|
func (self *ImageFrame) DataAsSHM(pattern string) (ans shm.MMap, err error) {
|
|
bytes_per_pixel := 4
|
|
if self.Is_opaque {
|
|
bytes_per_pixel = 3
|
|
}
|
|
ans, err = shm.CreateTemp(pattern, uint64(self.Width*self.Height*bytes_per_pixel))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
switch img := self.Img.(type) {
|
|
case *NRGB:
|
|
if bytes_per_pixel == 3 {
|
|
copy(ans.Slice(), img.Pix)
|
|
return
|
|
}
|
|
case *image.NRGBA:
|
|
if bytes_per_pixel == 4 {
|
|
copy(ans.Slice(), img.Pix)
|
|
return
|
|
}
|
|
}
|
|
dest_rect := image.Rect(0, 0, self.Width, self.Height)
|
|
var final_img image.Image
|
|
switch bytes_per_pixel {
|
|
case 3:
|
|
rgb := &NRGB{Pix: ans.Slice(), Stride: bytes_per_pixel * self.Width, Rect: dest_rect}
|
|
final_img = rgb
|
|
case 4:
|
|
rgba := &image.NRGBA{Pix: ans.Slice(), Stride: bytes_per_pixel * self.Width, Rect: dest_rect}
|
|
final_img = rgba
|
|
}
|
|
ctx := Context{}
|
|
ctx.PasteCenter(final_img, self.Img, nil)
|
|
return
|
|
|
|
}
|
|
|
|
func (self *ImageFrame) Data() (ans []byte) {
|
|
bytes_per_pixel := 4
|
|
if self.Is_opaque {
|
|
bytes_per_pixel = 3
|
|
}
|
|
switch img := self.Img.(type) {
|
|
case *NRGB:
|
|
if bytes_per_pixel == 3 {
|
|
return img.Pix
|
|
}
|
|
case *image.NRGBA:
|
|
if bytes_per_pixel == 4 {
|
|
return img.Pix
|
|
}
|
|
}
|
|
dest_rect := image.Rect(0, 0, self.Width, self.Height)
|
|
var final_img image.Image
|
|
switch bytes_per_pixel {
|
|
case 3:
|
|
rgb := NewNRGB(dest_rect)
|
|
final_img = rgb
|
|
ans = rgb.Pix
|
|
case 4:
|
|
rgba := image.NewNRGBA(dest_rect)
|
|
final_img = rgba
|
|
ans = rgba.Pix
|
|
}
|
|
ctx := Context{}
|
|
ctx.PasteCenter(final_img, self.Img, nil)
|
|
return
|
|
}
|
|
|
|
func ImageFrameFromSerialized(s SerializableImageFrame, data []byte) (*ImageFrame, error) {
|
|
ans := ImageFrame{
|
|
Width: s.Width, Height: s.Height, Left: s.Left, Top: s.Top,
|
|
Number: s.Number, Compose_onto: s.Compose_onto, Delay_ms: int32(s.Delay_ms),
|
|
Is_opaque: s.Is_opaque,
|
|
}
|
|
r := image.Rect(0, 0, s.Width, s.Height)
|
|
if s.Is_opaque {
|
|
if len(data) != 3*r.Dx()*r.Dy() {
|
|
return nil, fmt.Errorf("serialized image data has size: %d != %d", len(data), 3*r.Dy()*r.Dx())
|
|
}
|
|
ans.Img = &NRGB{Pix: data, Stride: 3 * r.Dx(), Rect: r}
|
|
} else {
|
|
if len(data) != 4*r.Dx()*r.Dy() {
|
|
return nil, fmt.Errorf("serialized image data has size: %d != %d", len(data), 4*r.Dy()*r.Dx())
|
|
}
|
|
ans.Img = &image.NRGBA{Pix: data, Stride: 4 * r.Dx(), Rect: r}
|
|
}
|
|
return &ans, nil
|
|
}
|
|
|
|
type ImageData struct {
|
|
Width, Height int
|
|
Format_uppercase string
|
|
Frames []*ImageFrame
|
|
}
|
|
|
|
type SerializableImageMetadata struct {
|
|
Version int
|
|
Width, Height int
|
|
Format_uppercase string
|
|
Frames []SerializableImageFrame
|
|
}
|
|
|
|
const SERIALIZE_VERSION = 1
|
|
|
|
func (self *ImageData) Serialize() (SerializableImageMetadata, [][]byte) {
|
|
m := SerializableImageMetadata{Version: SERIALIZE_VERSION, Width: self.Width, Height: self.Height, Format_uppercase: self.Format_uppercase}
|
|
data := make([][]byte, len(self.Frames))
|
|
for i, f := range self.Frames {
|
|
m.Frames = append(m.Frames, f.Serialize())
|
|
data[i] = f.Data()
|
|
m.Frames[len(m.Frames)-1].Size = len(data[i])
|
|
}
|
|
return m, data
|
|
}
|
|
|
|
func ImageFromSerialized(m SerializableImageMetadata, data [][]byte) (*ImageData, error) {
|
|
if m.Version > SERIALIZE_VERSION {
|
|
return nil, fmt.Errorf("serialized image data has unsupported version: %d", m.Version)
|
|
}
|
|
if len(m.Frames) != len(data) {
|
|
return nil, fmt.Errorf("serialized image data has %d frames in metadata but have data for: %d", len(m.Frames), len(data))
|
|
}
|
|
ans := ImageData{
|
|
Width: m.Width, Height: m.Height, Format_uppercase: m.Format_uppercase,
|
|
}
|
|
for i, f := range m.Frames {
|
|
if ff, err := ImageFrameFromSerialized(f, data[i]); err != nil {
|
|
return nil, err
|
|
} else {
|
|
ans.Frames = append(ans.Frames, ff)
|
|
}
|
|
}
|
|
return &ans, nil
|
|
}
|
|
|
|
func (self *ImageFrame) Resize(x_frac, y_frac float64) *ImageFrame {
|
|
b := self.Img.Bounds()
|
|
left, top, width, height := b.Min.X, b.Min.Y, b.Dx(), b.Dy()
|
|
ans := *self
|
|
ans.Width = int(x_frac * float64(width))
|
|
ans.Height = int(y_frac * float64(height))
|
|
ans.Img = imaging.Resize(self.Img, ans.Width, ans.Height, imaging.Lanczos)
|
|
ans.Left = int(x_frac * float64(left))
|
|
ans.Top = int(y_frac * float64(top))
|
|
return &ans
|
|
|
|
}
|
|
|
|
func (self *ImageData) Resize(x_frac, y_frac float64) *ImageData {
|
|
ans := *self
|
|
ans.Frames = utils.Map(func(f *ImageFrame) *ImageFrame { return f.Resize(x_frac, y_frac) }, self.Frames)
|
|
if len(ans.Frames) > 0 {
|
|
ans.Width, ans.Height = ans.Frames[0].Width, ans.Frames[0].Height
|
|
}
|
|
return &ans
|
|
}
|
|
|
|
func CalcMinimumGIFGap(gaps []int) int {
|
|
// Some broken GIF images have all zero gaps, browsers with their usual
|
|
// idiot ideas render these with a default 100ms gap https://bugzilla.mozilla.org/show_bug.cgi?id=125137
|
|
// Browsers actually force a 100ms gap at any zero gap frame, but that
|
|
// just means it is impossible to deliberately use zero gap frames for
|
|
// sophisticated blending, so we dont do that.
|
|
max_gap := utils.Max(0, gaps...)
|
|
min_gap := 0
|
|
if max_gap <= 0 {
|
|
min_gap = 10
|
|
}
|
|
return min_gap
|
|
}
|
|
|
|
func SetGIFFrameDisposal(number, anchor_frame int, disposal byte) (int, int) {
|
|
compose_onto := 0
|
|
if number > 1 {
|
|
switch disposal {
|
|
case gif.DisposalNone:
|
|
compose_onto = number - 1
|
|
anchor_frame = number
|
|
case gif.DisposalBackground:
|
|
// see https://github.com/golang/go/issues/20694
|
|
anchor_frame = number
|
|
case gif.DisposalPrevious:
|
|
compose_onto = anchor_frame
|
|
}
|
|
}
|
|
return anchor_frame, compose_onto
|
|
}
|
|
|
|
func MakeTempDir(template string) (ans string, err error) {
|
|
if template == "" {
|
|
template = "kitty-img-*"
|
|
}
|
|
if shm.SHM_DIR != "" {
|
|
ans, err = os.MkdirTemp(shm.SHM_DIR, template)
|
|
if err == nil {
|
|
return
|
|
}
|
|
}
|
|
return os.MkdirTemp("", template)
|
|
}
|
|
|
|
func check_resize(frame *ImageFrame, filename string) error {
|
|
// ImageMagick sometimes generates RGBA images smaller than the specified
|
|
// size. See https://github.com/kovidgoyal/kitty/issues/276 for examples
|
|
s, err := os.Stat(filename)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
sz := int(s.Size())
|
|
bytes_per_pixel := 4
|
|
if frame.Is_opaque {
|
|
bytes_per_pixel = 3
|
|
}
|
|
expected_size := bytes_per_pixel * frame.Width * frame.Height
|
|
if sz < expected_size {
|
|
missing := expected_size - sz
|
|
if missing%(bytes_per_pixel*frame.Width) != 0 {
|
|
return fmt.Errorf("ImageMagick failed to resize correctly. It generated %d < %d of data (w=%d h=%d bpp=%d)", sz, expected_size, frame.Width, frame.Height, bytes_per_pixel)
|
|
}
|
|
frame.Height -= missing / (bytes_per_pixel * frame.Width)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (frame *ImageFrame) set_delay(min_gap, delay int) {
|
|
frame.Delay_ms = int32(max(min_gap, delay) * 10)
|
|
if frame.Delay_ms == 0 {
|
|
frame.Delay_ms = -1 // gapless frame in the graphics protocol
|
|
}
|
|
}
|
|
|
|
func open_native_gif(f io.Reader, ans *ImageData) error {
|
|
gif_frames, err := gif.DecodeAll(f)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
min_gap := CalcMinimumGIFGap(gif_frames.Delay)
|
|
anchor_frame := 1
|
|
for i, paletted_img := range gif_frames.Image {
|
|
b := paletted_img.Bounds()
|
|
frame := ImageFrame{Img: paletted_img, Left: b.Min.X, Top: b.Min.Y, Width: b.Dx(), Height: b.Dy(), Number: len(ans.Frames) + 1, Is_opaque: paletted_img.Opaque()}
|
|
frame.set_delay(min_gap, gif_frames.Delay[i])
|
|
anchor_frame, frame.Compose_onto = SetGIFFrameDisposal(frame.Number, anchor_frame, gif_frames.Disposal[i])
|
|
ans.Frames = append(ans.Frames, &frame)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func OpenNativeImageFromReader(f io.ReadSeeker) (ans *ImageData, err error) {
|
|
c, fmt, err := image.DecodeConfig(f)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
_, _ = f.Seek(0, io.SeekStart)
|
|
ans = &ImageData{Width: c.Width, Height: c.Height, Format_uppercase: strings.ToUpper(fmt)}
|
|
|
|
if ans.Format_uppercase == "GIF" {
|
|
err = open_native_gif(f, ans)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
} else {
|
|
img, _, err := exiffix.Decode(f)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b := img.Bounds()
|
|
ans.Frames = []*ImageFrame{{Img: img, Left: b.Min.X, Top: b.Min.Y, Width: b.Dx(), Height: b.Dy()}}
|
|
ans.Frames[0].Is_opaque = c.ColorModel == color.YCbCrModel || c.ColorModel == color.GrayModel || c.ColorModel == color.Gray16Model || c.ColorModel == color.CMYKModel || ans.Format_uppercase == "JPEG" || ans.Format_uppercase == "JPG" || IsOpaque(img)
|
|
}
|
|
return
|
|
}
|
|
|
|
// ImageMagick {{{
|
|
var MagickExe = sync.OnceValue(func() string {
|
|
return utils.FindExe("magick")
|
|
})
|
|
|
|
func RunMagick(path string, cmd []string) ([]byte, error) {
|
|
if MagickExe() != "magick" {
|
|
cmd = append([]string{MagickExe()}, cmd...)
|
|
}
|
|
c := exec.Command(cmd[0], cmd[1:]...)
|
|
output, err := c.Output()
|
|
if err != nil {
|
|
var exit_err *exec.ExitError
|
|
if errors.As(err, &exit_err) {
|
|
return nil, fmt.Errorf("Running the command: %s\nFailed with error:\n%s", strings.Join(cmd, " "), string(exit_err.Stderr))
|
|
}
|
|
return nil, fmt.Errorf("Could not find the program: %#v. Is ImageMagick installed and in your PATH?", cmd[0])
|
|
}
|
|
return output, nil
|
|
}
|
|
|
|
type IdentifyOutput struct {
|
|
Fmt, Canvas, Transparency, Gap, Index, Size, Dpi, Dispose, Orientation string
|
|
}
|
|
|
|
type IdentifyRecord struct {
|
|
Fmt_uppercase string
|
|
Gap int
|
|
Canvas struct{ Width, Height, Left, Top int }
|
|
Width, Height int
|
|
Dpi struct{ X, Y float64 }
|
|
Index int
|
|
Is_opaque bool
|
|
Needs_blend bool
|
|
Disposal int
|
|
Dimensions_swapped bool
|
|
}
|
|
|
|
func parse_identify_record(ans *IdentifyRecord, raw *IdentifyOutput) (err error) {
|
|
ans.Fmt_uppercase = strings.ToUpper(raw.Fmt)
|
|
if raw.Gap != "" {
|
|
ans.Gap, err = strconv.Atoi(raw.Gap)
|
|
if err != nil {
|
|
return fmt.Errorf("Invalid gap value in identify output: %s", raw.Gap)
|
|
}
|
|
ans.Gap = max(0, ans.Gap)
|
|
}
|
|
area, pos, found := strings.Cut(raw.Canvas, "+")
|
|
ok := false
|
|
if found {
|
|
w, h, found := strings.Cut(area, "x")
|
|
if found {
|
|
ans.Canvas.Width, err = strconv.Atoi(w)
|
|
if err == nil {
|
|
ans.Canvas.Height, err = strconv.Atoi(h)
|
|
if err == nil {
|
|
x, y, found := strings.Cut(pos, "+")
|
|
if found {
|
|
ans.Canvas.Left, err = strconv.Atoi(x)
|
|
if err == nil {
|
|
if ans.Canvas.Top, err = strconv.Atoi(y); err == nil {
|
|
ok = true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if !ok {
|
|
return fmt.Errorf("Invalid canvas value in identify output: %s", raw.Canvas)
|
|
}
|
|
w, h, found := strings.Cut(raw.Size, "x")
|
|
ok = false
|
|
if found {
|
|
ans.Width, err = strconv.Atoi(w)
|
|
if err == nil {
|
|
if ans.Height, err = strconv.Atoi(h); err == nil {
|
|
ok = true
|
|
}
|
|
}
|
|
}
|
|
if !ok {
|
|
return fmt.Errorf("Invalid size value in identify output: %s", raw.Size)
|
|
}
|
|
x, y, found := strings.Cut(raw.Dpi, "x")
|
|
ok = false
|
|
if found {
|
|
ans.Dpi.X, err = strconv.ParseFloat(x, 64)
|
|
if err == nil {
|
|
if ans.Dpi.Y, err = strconv.ParseFloat(y, 64); err == nil {
|
|
ok = true
|
|
}
|
|
}
|
|
}
|
|
if !ok {
|
|
return fmt.Errorf("Invalid dpi value in identify output: %s", raw.Dpi)
|
|
}
|
|
ans.Index, err = strconv.Atoi(raw.Index)
|
|
if err != nil {
|
|
return fmt.Errorf("Invalid index value in identify output: %s", raw.Index)
|
|
}
|
|
q := strings.ToLower(raw.Transparency)
|
|
if q == "blend" || q == "true" {
|
|
ans.Is_opaque = false
|
|
} else {
|
|
ans.Is_opaque = true
|
|
}
|
|
ans.Needs_blend = q == "blend"
|
|
switch strings.ToLower(raw.Dispose) {
|
|
case "undefined":
|
|
ans.Disposal = 0
|
|
case "none":
|
|
ans.Disposal = gif.DisposalNone
|
|
case "background":
|
|
ans.Disposal = gif.DisposalBackground
|
|
case "previous":
|
|
ans.Disposal = gif.DisposalPrevious
|
|
default:
|
|
return fmt.Errorf("Invalid value for dispose: %s", raw.Dispose)
|
|
}
|
|
switch raw.Orientation {
|
|
case "5", "6", "7", "8":
|
|
ans.Dimensions_swapped = true
|
|
}
|
|
if ans.Dimensions_swapped {
|
|
ans.Canvas.Width, ans.Canvas.Height = ans.Canvas.Height, ans.Canvas.Width
|
|
ans.Width, ans.Height = ans.Height, ans.Width
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func IdentifyWithMagick(path string) (ans []IdentifyRecord, err error) {
|
|
cmd := []string{"identify"}
|
|
q := `{"fmt":"%m","canvas":"%g","transparency":"%A","gap":"%T","index":"%p","size":"%wx%h",` +
|
|
`"dpi":"%xx%y","dispose":"%D","orientation":"%[EXIF:Orientation]"},`
|
|
cmd = append(cmd, "-format", q, "--", path)
|
|
output, err := RunMagick(path, cmd)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to identify image at path: %s with error: %w", path, err)
|
|
}
|
|
output = bytes.TrimRight(bytes.TrimSpace(output), ",")
|
|
raw_json := make([]byte, 0, len(output)+2)
|
|
raw_json = append(raw_json, '[')
|
|
raw_json = append(raw_json, output...)
|
|
raw_json = append(raw_json, ']')
|
|
var records []IdentifyOutput
|
|
err = json.Unmarshal(raw_json, &records)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("The ImageMagick identify program returned malformed output for the image at path: %s, with error: %w", path, err)
|
|
}
|
|
ans = make([]IdentifyRecord, len(records))
|
|
for i, rec := range records {
|
|
err = parse_identify_record(&ans[i], &rec)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
return ans, nil
|
|
}
|
|
|
|
type RenderOptions struct {
|
|
RemoveAlpha *NRGBColor
|
|
Flip, Flop bool
|
|
ResizeTo image.Point
|
|
OnlyFirstFrame bool
|
|
TempfilenameTemplate string
|
|
}
|
|
|
|
func RenderWithMagick(path string, ro *RenderOptions, frames []IdentifyRecord) (ans []*ImageFrame, fmap map[int]string, err error) {
|
|
cmd := []string{"convert"}
|
|
ans = make([]*ImageFrame, 0, len(frames))
|
|
fmap = make(map[int]string, len(frames))
|
|
|
|
defer func() {
|
|
if err != nil {
|
|
for _, f := range fmap {
|
|
os.Remove(f)
|
|
}
|
|
}
|
|
}()
|
|
|
|
if ro.RemoveAlpha != nil {
|
|
cmd = append(cmd, "-background", ro.RemoveAlpha.AsSharp(), "-alpha", "remove")
|
|
} else {
|
|
cmd = append(cmd, "-background", "none")
|
|
}
|
|
if ro.Flip {
|
|
cmd = append(cmd, "-flip")
|
|
}
|
|
if ro.Flop {
|
|
cmd = append(cmd, "-flop")
|
|
}
|
|
cpath := path
|
|
if ro.OnlyFirstFrame {
|
|
cpath += "[0]"
|
|
}
|
|
has_multiple_frames := len(frames) > 1
|
|
get_multiple_frames := has_multiple_frames && !ro.OnlyFirstFrame
|
|
cmd = append(cmd, "--", cpath, "-auto-orient")
|
|
if ro.ResizeTo.X > 0 {
|
|
rcmd := []string{"-resize", fmt.Sprintf("%dx%d!", ro.ResizeTo.X, ro.ResizeTo.Y)}
|
|
if get_multiple_frames {
|
|
cmd = append(cmd, "-coalesce")
|
|
cmd = append(cmd, rcmd...)
|
|
cmd = append(cmd, "-deconstruct")
|
|
} else {
|
|
cmd = append(cmd, rcmd...)
|
|
}
|
|
}
|
|
cmd = append(cmd, "-depth", "8", "-set", "filename:f", "%w-%h-%g-%p")
|
|
if get_multiple_frames {
|
|
cmd = append(cmd, "+adjoin")
|
|
}
|
|
tdir, err := MakeTempDir(ro.TempfilenameTemplate)
|
|
if err != nil {
|
|
err = fmt.Errorf("Failed to create temporary directory to hold ImageMagick output with error: %w", err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(tdir)
|
|
mode := "rgba"
|
|
if frames[0].Is_opaque {
|
|
mode = "rgb"
|
|
}
|
|
cmd = append(cmd, filepath.Join(tdir, "im-%[filename:f]."+mode))
|
|
_, err = RunMagick(path, cmd)
|
|
if err != nil {
|
|
return
|
|
}
|
|
entries, err := os.ReadDir(tdir)
|
|
if err != nil {
|
|
err = fmt.Errorf("Failed to read temp dir used to store ImageMagick output with error: %w", err)
|
|
return
|
|
}
|
|
base_dir := filepath.Dir(tdir)
|
|
gaps := make([]int, len(frames))
|
|
for i, frame := range frames {
|
|
gaps[i] = frame.Gap
|
|
}
|
|
min_gap := CalcMinimumGIFGap(gaps)
|
|
for _, entry := range entries {
|
|
fname := entry.Name()
|
|
p, _, _ := strings.Cut(fname, ".")
|
|
parts := strings.Split(p, "-")
|
|
if len(parts) < 5 {
|
|
continue
|
|
}
|
|
index, cerr := strconv.Atoi(parts[len(parts)-1])
|
|
if cerr != nil || index < 0 || index >= len(frames) {
|
|
continue
|
|
}
|
|
width, cerr := strconv.Atoi(parts[1])
|
|
if cerr != nil {
|
|
continue
|
|
}
|
|
height, cerr := strconv.Atoi(parts[2])
|
|
if cerr != nil {
|
|
continue
|
|
}
|
|
_, pos, found := strings.Cut(parts[3], "+")
|
|
if !found {
|
|
continue
|
|
}
|
|
px, py, found := strings.Cut(pos, "+")
|
|
if !found {
|
|
continue
|
|
}
|
|
x, cerr := strconv.Atoi(px)
|
|
if cerr != nil {
|
|
continue
|
|
}
|
|
y, cerr := strconv.Atoi(py)
|
|
if cerr != nil {
|
|
continue
|
|
}
|
|
identify_data := frames[index]
|
|
df, cerr := os.CreateTemp(base_dir, TempTemplate+"."+mode)
|
|
if cerr != nil {
|
|
err = fmt.Errorf("Failed to create a temporary file in %s with error: %w", base_dir, cerr)
|
|
return
|
|
}
|
|
err = os.Rename(filepath.Join(tdir, fname), df.Name())
|
|
if err != nil {
|
|
err = fmt.Errorf("Failed to rename a temporary file in %s with error: %w", tdir, err)
|
|
return
|
|
}
|
|
df.Close()
|
|
fmap[index+1] = df.Name()
|
|
frame := ImageFrame{
|
|
Number: index + 1, Width: width, Height: height, Left: x, Top: y, Is_opaque: identify_data.Is_opaque,
|
|
}
|
|
frame.set_delay(min_gap, identify_data.Gap)
|
|
err = check_resize(&frame, df.Name())
|
|
if err != nil {
|
|
return
|
|
}
|
|
ans = append(ans, &frame)
|
|
}
|
|
if len(ans) < len(frames) {
|
|
err = fmt.Errorf("Failed to render %d out of %d frames", len(frames)-len(ans), len(frames))
|
|
return
|
|
}
|
|
slices.SortFunc(ans, func(a, b *ImageFrame) int { return a.Number - b.Number })
|
|
anchor_frame := 1
|
|
for i, frame := range ans {
|
|
anchor_frame, frame.Compose_onto = SetGIFFrameDisposal(frame.Number, anchor_frame, byte(frames[i].Disposal))
|
|
}
|
|
return
|
|
}
|
|
|
|
func OpenImageFromPathWithMagick(path string) (ans *ImageData, err error) {
|
|
identify_records, err := IdentifyWithMagick(path)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to identify image at %#v with error: %w", path, err)
|
|
}
|
|
frames, filenames, err := RenderWithMagick(path, &RenderOptions{}, identify_records)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to render image at %#v with error: %w", path, err)
|
|
}
|
|
defer func() {
|
|
for _, f := range filenames {
|
|
os.Remove(f)
|
|
}
|
|
}()
|
|
|
|
for _, frame := range frames {
|
|
filename := filenames[frame.Number]
|
|
data, err := os.ReadFile(filename)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to read temp file for image %#v at %#v with error: %w", path, filename, err)
|
|
}
|
|
dest_rect := image.Rect(0, 0, frame.Width, frame.Height)
|
|
if frame.Is_opaque {
|
|
frame.Img = &NRGB{Pix: data, Stride: frame.Width * 3, Rect: dest_rect}
|
|
} else {
|
|
frame.Img = &image.NRGBA{Pix: data, Stride: frame.Width * 4, Rect: dest_rect}
|
|
}
|
|
}
|
|
ans = &ImageData{
|
|
Width: frames[0].Width, Height: frames[0].Height, Format_uppercase: identify_records[0].Fmt_uppercase, Frames: frames,
|
|
}
|
|
return ans, nil
|
|
}
|
|
|
|
// }}}
|
|
|
|
func OpenImageFromPath(path string) (ans *ImageData, err error) {
|
|
mt := utils.GuessMimeType(path)
|
|
if DecodableImageTypes[mt] {
|
|
f, err := os.Open(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer f.Close()
|
|
ans, err = OpenNativeImageFromReader(f)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to load image at %#v with error: %w", path, err)
|
|
}
|
|
} else {
|
|
return OpenImageFromPathWithMagick(path)
|
|
}
|
|
return
|
|
}
|