func (e *GoImageEngine) TransformGIF(img *imagefile.ImageFile, width int, height int, options *Options, trans Transformation) ([]byte, error) { first, err := gif.Decode(bytes.NewReader(img.Source)) if err != nil { return nil, err } factor := scalingFactorImage(first, width, height) if factor > 1 && !options.Upscale { return img.Source, nil } g, err := gif.DecodeAll(bytes.NewReader(img.Source)) if err != nil { return nil, err } length := len(g.Image) done := make(chan *Result) images := make([]*image.Paletted, length) processed := 0 for i := range g.Image { go func(paletted *image.Paletted, width int, height int, position int, trans Transformation, options *Options) { img := e.Scale(paletted, width, height, options.Upscale, trans) bounds := img.Bounds() done <- &Result{ Image: img, Position: position, Paletted: image.NewPaletted(image.Rect(0, 0, bounds.Max.X, bounds.Max.Y), paletted.Palette), } }(g.Image[i], width, height, i, trans, options) } for { select { case result := <-done: bounds := result.Image.Bounds() draw.Draw(result.Paletted, bounds, result.Image, bounds.Min, draw.Src) images[result.Position] = result.Paletted processed++ case <-time.After(time.Second * 5): break } if processed == length { break } } close(done) g.Image = images buf := &bytes.Buffer{} err = gif.EncodeAll(buf, g) if err != nil { return nil, err } return buf.Bytes(), nil }