2016-06-22 15:54:16 -05:00
|
|
|
// Copyright 2013 Google Inc. All rights reserved.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2014-12-04 13:34:26 -05:00
|
|
|
package imageproxy
|
|
|
|
|
|
|
|
import (
|
2014-12-04 20:30:01 -05:00
|
|
|
"bytes"
|
2014-12-04 13:34:26 -05:00
|
|
|
"image"
|
|
|
|
"image/color"
|
|
|
|
"image/draw"
|
2014-12-04 20:30:01 -05:00
|
|
|
"image/gif"
|
|
|
|
"image/jpeg"
|
|
|
|
"image/png"
|
|
|
|
"io"
|
2014-12-04 13:34:26 -05:00
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/disintegration/imaging"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
red = color.NRGBA{255, 0, 0, 255}
|
|
|
|
green = color.NRGBA{0, 255, 0, 255}
|
|
|
|
blue = color.NRGBA{0, 0, 255, 255}
|
|
|
|
yellow = color.NRGBA{255, 255, 0, 255}
|
|
|
|
)
|
|
|
|
|
|
|
|
// newImage creates a new NRGBA image with the specified dimensions and pixel
|
|
|
|
// color data. If the length of pixels is 1, the entire image is filled with
|
|
|
|
// that color.
|
|
|
|
func newImage(w, h int, pixels ...color.NRGBA) image.Image {
|
|
|
|
m := image.NewNRGBA(image.Rect(0, 0, w, h))
|
|
|
|
if len(pixels) == 1 {
|
|
|
|
draw.Draw(m, m.Bounds(), &image.Uniform{pixels[0]}, image.ZP, draw.Src)
|
|
|
|
} else {
|
|
|
|
for i, p := range pixels {
|
|
|
|
m.Set(i%w, i/w, p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
2015-10-14 11:15:57 -05:00
|
|
|
func TestResizeParams(t *testing.T) {
|
|
|
|
src := image.NewNRGBA(image.Rect(0, 0, 64, 128))
|
|
|
|
tests := []struct {
|
2015-11-26 15:05:13 -05:00
|
|
|
opt Options
|
|
|
|
w, h int
|
2015-10-14 11:25:48 -05:00
|
|
|
resize bool
|
2015-10-14 11:15:57 -05:00
|
|
|
}{
|
2015-10-14 11:25:48 -05:00
|
|
|
{Options{Width: 0.5}, 32, 0, true},
|
|
|
|
{Options{Height: 0.5}, 0, 64, true},
|
|
|
|
{Options{Width: 0.5, Height: 0.5}, 32, 64, true},
|
|
|
|
{Options{Width: 100, Height: 200}, 0, 0, false},
|
|
|
|
{Options{Width: 100, Height: 200, ScaleUp: true}, 100, 200, true},
|
|
|
|
{Options{Width: 64}, 0, 0, false},
|
|
|
|
{Options{Height: 128}, 0, 0, false},
|
2015-10-14 11:15:57 -05:00
|
|
|
}
|
2015-11-26 15:05:13 -05:00
|
|
|
for _, tt := range tests {
|
|
|
|
w, h, resize := resizeParams(src, tt.opt)
|
2015-10-14 11:25:48 -05:00
|
|
|
if w != tt.w || h != tt.h || resize != tt.resize {
|
2015-11-26 15:05:13 -05:00
|
|
|
t.Errorf("resizeParams(%v) returned (%d,%d,%t), want (%d,%d,%t)", tt.opt, w, h, resize, tt.w, tt.h, tt.resize)
|
2015-10-14 11:15:57 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-15 12:55:49 -05:00
|
|
|
func TestCropParams(t *testing.T) {
|
|
|
|
src := image.NewNRGBA(image.Rect(0, 0, 64, 128))
|
|
|
|
tests := []struct {
|
|
|
|
opt Options
|
|
|
|
x0, y0, x1, y1 int
|
|
|
|
crop bool
|
|
|
|
}{
|
|
|
|
{Options{CropHeight: 0, CropWidth: 10}, 0, 0, 0, 0, false},
|
|
|
|
{Options{CropHeight: 10, CropWidth: 0}, 0, 0, 0, 0, false},
|
|
|
|
{Options{CropHeight: -1, CropWidth: -1}, 0, 0, 0, 0, false},
|
|
|
|
{Options{CropWidth: 50, CropHeight: 100}, 0, 0, 50, 100, true},
|
|
|
|
{Options{CropWidth: 100, CropHeight: 100}, 0, 0, 64, 100, true},
|
|
|
|
{Options{CropX: 50, CropY: 100, CropWidth: 50, CropHeight: 100}, 14, 28, 64, 128, true},
|
|
|
|
{Options{CropX: 50, CropY: 100, CropWidth: 100, CropHeight: 150}, 0, 0, 64, 128, true},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
x0, y0, x1, y1, crop := cropParams(src, tt.opt)
|
|
|
|
if x0 != tt.x0 || y0 != tt.y0 || x1 != tt.x1 || y1 != tt.y1 || crop != tt.crop {
|
|
|
|
t.Errorf("cropParams(%v) returned (%d,%d,%d,%d,%t), want (%d,%d,%d,%d,%t)", tt.opt, x0, y0, x1, y1, crop, tt.x0, tt.y0, tt.x1, tt.y1, tt.crop)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-04 20:30:01 -05:00
|
|
|
func TestTransform(t *testing.T) {
|
|
|
|
src := newImage(2, 2, red, green, blue, yellow)
|
|
|
|
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
png.Encode(buf, src)
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
encode func(io.Writer, image.Image)
|
|
|
|
exactOutput bool // whether input and output should match exactly
|
|
|
|
}{
|
|
|
|
{"gif", func(w io.Writer, m image.Image) { gif.Encode(w, m, nil) }, true},
|
|
|
|
{"jpeg", func(w io.Writer, m image.Image) { jpeg.Encode(w, m, nil) }, false},
|
|
|
|
{"png", func(w io.Writer, m image.Image) { png.Encode(w, m) }, true},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
tt.encode(buf, src)
|
|
|
|
in := buf.Bytes()
|
|
|
|
|
|
|
|
out, err := Transform(in, emptyOptions)
|
|
|
|
if err != nil {
|
2015-02-12 17:21:26 -05:00
|
|
|
t.Errorf("Transform with encoder %s returned unexpected error: %v", tt.name, err)
|
2014-12-04 20:30:01 -05:00
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(in, out) {
|
2015-02-12 17:21:26 -05:00
|
|
|
t.Errorf("Transform with with encoder %s with empty options returned modified result", tt.name)
|
2014-12-04 20:30:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
out, err = Transform(in, Options{Width: -1, Height: -1})
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Transform with encoder %s returned unexpected error: %v", tt.name, err)
|
|
|
|
}
|
|
|
|
if len(out) == 0 {
|
|
|
|
t.Errorf("Transform with encoder %s returned empty bytes", tt.name)
|
|
|
|
}
|
|
|
|
if tt.exactOutput && !reflect.DeepEqual(in, out) {
|
|
|
|
t.Errorf("Transform with encoder %s with noop Options returned modified result", tt.name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := Transform([]byte{}, Options{Width: 1}); err == nil {
|
|
|
|
t.Errorf("Transform with invalid image input did not return expected err")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-04 13:34:26 -05:00
|
|
|
func TestTransformImage(t *testing.T) {
|
|
|
|
// ref is a 2x2 reference image containing four colors
|
|
|
|
ref := newImage(2, 2, red, green, blue, yellow)
|
|
|
|
|
2017-05-15 12:55:49 -05:00
|
|
|
// cropRef is a 4x4 image with four colors, each in 2x2 quarter
|
|
|
|
cropRef := newImage(4, 4, red, red, green, green, red, red, green, green, blue, blue, yellow, yellow, blue, blue, yellow, yellow)
|
|
|
|
|
2014-12-04 13:34:26 -05:00
|
|
|
// use simpler filter while testing that won't skew colors
|
|
|
|
resampleFilter = imaging.Box
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
src image.Image // source image to transform
|
|
|
|
opt Options // options to apply during transform
|
|
|
|
want image.Image // expected transformed image
|
|
|
|
}{
|
|
|
|
// no transformation
|
|
|
|
{ref, emptyOptions, ref},
|
|
|
|
|
|
|
|
// rotations
|
|
|
|
{ref, Options{Rotate: 45}, ref}, // invalid rotation is a noop
|
|
|
|
{ref, Options{Rotate: 90}, newImage(2, 2, green, yellow, red, blue)},
|
|
|
|
{ref, Options{Rotate: 180}, newImage(2, 2, yellow, blue, green, red)},
|
|
|
|
{ref, Options{Rotate: 270}, newImage(2, 2, blue, red, yellow, green)},
|
|
|
|
|
|
|
|
// flips
|
|
|
|
{
|
|
|
|
ref,
|
|
|
|
Options{FlipHorizontal: true},
|
|
|
|
newImage(2, 2, green, red, yellow, blue),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ref,
|
|
|
|
Options{FlipVertical: true},
|
|
|
|
newImage(2, 2, blue, yellow, red, green),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ref,
|
|
|
|
Options{FlipHorizontal: true, FlipVertical: true},
|
|
|
|
newImage(2, 2, yellow, blue, green, red),
|
|
|
|
},
|
|
|
|
|
|
|
|
// resizing
|
|
|
|
{ // can't resize larger than original image
|
|
|
|
ref,
|
|
|
|
Options{Width: 100, Height: 100},
|
|
|
|
ref,
|
|
|
|
},
|
2015-08-12 13:39:38 -05:00
|
|
|
{ // can resize larger than original image
|
|
|
|
ref,
|
|
|
|
Options{Width: 4, Height: 4, ScaleUp: true},
|
|
|
|
newImage(4, 4, red, red, green, green, red, red, green, green, blue, blue, yellow, yellow, blue, blue, yellow, yellow),
|
|
|
|
},
|
2014-12-04 13:34:26 -05:00
|
|
|
{ // invalid values
|
|
|
|
ref,
|
|
|
|
Options{Width: -1, Height: -1},
|
|
|
|
ref,
|
|
|
|
},
|
|
|
|
{ // absolute values
|
|
|
|
newImage(100, 100, red),
|
|
|
|
Options{Width: 1, Height: 1},
|
|
|
|
newImage(1, 1, red),
|
|
|
|
},
|
|
|
|
{ // percentage values
|
|
|
|
newImage(100, 100, red),
|
|
|
|
Options{Width: 0.50, Height: 0.25},
|
|
|
|
newImage(50, 25, red),
|
|
|
|
},
|
|
|
|
{ // only width specified, proportional height
|
|
|
|
newImage(100, 50, red),
|
|
|
|
Options{Width: 50},
|
|
|
|
newImage(50, 25, red),
|
|
|
|
},
|
|
|
|
{ // only height specified, proportional width
|
|
|
|
newImage(100, 50, red),
|
|
|
|
Options{Height: 25},
|
|
|
|
newImage(50, 25, red),
|
|
|
|
},
|
|
|
|
{ // resize in one dimenstion, with cropping
|
|
|
|
newImage(4, 2, red, red, blue, blue, red, red, blue, blue),
|
|
|
|
Options{Width: 4, Height: 1},
|
|
|
|
newImage(4, 1, red, red, blue, blue),
|
|
|
|
},
|
|
|
|
{ // resize in two dimensions, with cropping
|
|
|
|
newImage(4, 2, red, red, blue, blue, red, red, blue, blue),
|
|
|
|
Options{Width: 2, Height: 2},
|
|
|
|
newImage(2, 2, red, blue, red, blue),
|
|
|
|
},
|
|
|
|
{ // resize in two dimensions, fit option prevents cropping
|
|
|
|
newImage(4, 2, red, red, blue, blue, red, red, blue, blue),
|
|
|
|
Options{Width: 2, Height: 2, Fit: true},
|
|
|
|
newImage(2, 1, red, blue),
|
|
|
|
},
|
|
|
|
{ // scale image explicitly
|
|
|
|
newImage(4, 2, red, red, blue, blue, red, red, blue, blue),
|
|
|
|
Options{Width: 2, Height: 1},
|
|
|
|
newImage(2, 1, red, blue),
|
|
|
|
},
|
|
|
|
|
|
|
|
// combinations of options
|
|
|
|
{
|
|
|
|
newImage(4, 2, red, red, blue, blue, red, red, blue, blue),
|
|
|
|
Options{Width: 2, Height: 1, Fit: true, FlipHorizontal: true, Rotate: 90},
|
|
|
|
newImage(1, 2, red, blue),
|
|
|
|
},
|
2017-05-15 12:55:49 -05:00
|
|
|
|
|
|
|
// crop
|
|
|
|
{ // quarter ((0, 0), (2, 2)) -> red
|
|
|
|
cropRef,
|
|
|
|
Options{CropHeight: 2, CropWidth: 2},
|
|
|
|
newImage(2, 2, red, red, red, red),
|
|
|
|
},
|
|
|
|
{ // quarter ((2, 0), (4, 2)) -> green
|
|
|
|
cropRef,
|
|
|
|
Options{CropHeight: 2, CropWidth: 2, CropX: 2},
|
|
|
|
newImage(2, 2, green, green, green, green),
|
|
|
|
},
|
|
|
|
{ // quarter ((0, 2), (2, 4)) -> blue
|
|
|
|
cropRef,
|
|
|
|
Options{CropHeight: 2, CropWidth: 2, CropX: 0, CropY: 2},
|
|
|
|
newImage(2, 2, blue, blue, blue, blue),
|
|
|
|
},
|
|
|
|
{ // quarter ((2, 2), (4, 4)) -> yellow
|
|
|
|
cropRef,
|
|
|
|
Options{CropHeight: 2, CropWidth: 2, CropX: 2, CropY: 2},
|
|
|
|
newImage(2, 2, yellow, yellow, yellow, yellow),
|
|
|
|
},
|
2014-12-04 13:34:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
if got := transformImage(tt.src, tt.opt); !reflect.DeepEqual(got, tt.want) {
|
2017-05-15 12:55:49 -05:00
|
|
|
t.Errorf("transformImage(%v, %v) returned image %#v, want %#v", tt.src, tt.opt, got, tt.want)
|
2014-12-04 13:34:26 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|