package thumbnail import ( "image" "image/color" "image/draw" "image/gif" "strings" "github.com/kovidgoyal/imaging" "github.com/owncloud/ocis/v2/services/thumbnails/pkg/errors" ) // Generator generates a web friendly file version. type Generator interface { Generate(size image.Rectangle, img interface{}, processor Processor) (interface{}, error) } // SimpleGenerator is the default image generator and is used for all image types expect gif. type SimpleGenerator struct{} // Generate generates a alternative image version. func (g SimpleGenerator) Generate(size image.Rectangle, img interface{}, processor Processor) (interface{}, error) { m, ok := img.(image.Image) if !ok { return nil, errors.ErrInvalidType } return processor.Process(m, size.Dx(), size.Dy(), imaging.Lanczos), nil } // GifGenerator is used to create a web friendly version of the provided gif image. type GifGenerator struct{} // Generate generates a alternative gif version. func (g GifGenerator) Generate(size image.Rectangle, img interface{}, processor Processor) (interface{}, error) { // Code inspired by https://github.com/willnorris/gifresize/blob/db93a7e1dcb1c279f7eeb99cc6d90b9e2e23e871/gifresize.go m, ok := img.(*gif.GIF) if !ok { return nil, errors.ErrInvalidType } // Create a new RGBA image to hold the incremental frames. srcX, srcY := m.Config.Width, m.Config.Height b := image.Rect(0, 0, srcX, srcY) tmp := image.NewRGBA(b) for i, frame := range m.Image { bounds := frame.Bounds() prev := tmp draw.Draw(tmp, bounds, frame, bounds.Min, draw.Over) processed := processor.Process(tmp, size.Dx(), size.Dy(), imaging.Lanczos) m.Image[i] = g.imageToPaletted(processed, frame.Palette) switch m.Disposal[i] { case gif.DisposalBackground: tmp = image.NewRGBA(b) case gif.DisposalPrevious: tmp = prev } } m.Config.Width = size.Dx() m.Config.Height = size.Dy() return m, nil } func (g GifGenerator) imageToPaletted(img image.Image, p color.Palette) *image.Paletted { b := img.Bounds() pm := image.NewPaletted(b, p) draw.FloydSteinberg.Draw(pm, b, img, image.Point{}) return pm } // GeneratorForType returns the generator for a given file type // or nil if the type is not supported. func GeneratorForType(fileType string) (Generator, error) { switch strings.ToLower(fileType) { case typePng, typeJpg, typeJpeg, typeGgs: return SimpleGenerator{}, nil case typeGif: return GifGenerator{}, nil default: return nil, errors.ErrNoEncoderForType } }