2013-12-07 20:05:18 +00:00
|
|
|
/*
|
|
|
|
Copyright 2013 The Camlistore Authors
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package resize
|
|
|
|
|
|
|
|
import (
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"image"
|
|
|
|
"image/color"
|
|
|
|
"image/draw"
|
|
|
|
"image/png"
|
|
|
|
"io"
|
|
|
|
"math"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// psnrThreshold is the threshold over which images must match to consider
|
|
|
|
// HalveInplace equivalent to Resize. It is in terms of dB and 60-80 is
|
|
|
|
// good for RGB.
|
|
|
|
psnrThreshold = 50.0
|
|
|
|
|
|
|
|
maxPixelDiffPercentage = 10
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
output = flag.String("output", "", "If non-empty, the directory to save comparison images.")
|
|
|
|
|
|
|
|
orig = image.Rect(0, 0, 1024, 1024)
|
|
|
|
thumb = image.Rect(0, 0, 64, 64)
|
|
|
|
)
|
|
|
|
|
2013-12-16 18:19:55 +00:00
|
|
|
var somePalette = []color.Color{
|
2014-03-20 19:29:45 +00:00
|
|
|
color.RGBA{0x00, 0x00, 0x00, 0xff},
|
|
|
|
color.RGBA{0x00, 0x00, 0x44, 0xff},
|
|
|
|
color.RGBA{0x00, 0x00, 0x88, 0xff},
|
|
|
|
color.RGBA{0x00, 0x00, 0xcc, 0xff},
|
|
|
|
color.RGBA{0x00, 0x44, 0x00, 0xff},
|
|
|
|
color.RGBA{0x00, 0x44, 0x44, 0xff},
|
|
|
|
color.RGBA{0x00, 0x44, 0x88, 0xff},
|
|
|
|
color.RGBA{0x00, 0x44, 0xcc, 0xff},
|
2013-12-16 18:19:55 +00:00
|
|
|
}
|
|
|
|
|
2013-12-07 20:05:18 +00:00
|
|
|
func makeImages(r image.Rectangle) []image.Image {
|
|
|
|
return []image.Image{
|
|
|
|
image.NewGray(r),
|
|
|
|
image.NewGray16(r),
|
|
|
|
image.NewNRGBA(r),
|
|
|
|
image.NewNRGBA64(r),
|
2013-12-16 18:19:55 +00:00
|
|
|
image.NewPaletted(r, somePalette),
|
2013-12-07 20:05:18 +00:00
|
|
|
image.NewRGBA(r),
|
|
|
|
image.NewRGBA64(r),
|
|
|
|
image.NewYCbCr(r, image.YCbCrSubsampleRatio444),
|
|
|
|
image.NewYCbCr(r, image.YCbCrSubsampleRatio422),
|
|
|
|
image.NewYCbCr(r, image.YCbCrSubsampleRatio420),
|
|
|
|
image.NewYCbCr(r, image.YCbCrSubsampleRatio440),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestResize(t *testing.T) {
|
|
|
|
for i, im := range makeImages(orig) {
|
|
|
|
m := Resize(im, orig, thumb.Dx(), thumb.Dy())
|
|
|
|
got, want := m.Bounds(), thumb
|
|
|
|
if !got.Eq(want) {
|
|
|
|
t.Error(i, "Want bounds", want, "got", got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestResampleInplace(t *testing.T) {
|
|
|
|
for i, im := range makeImages(orig) {
|
|
|
|
m := ResampleInplace(im, orig, thumb.Dx(), thumb.Dy())
|
|
|
|
got, want := m.Bounds(), thumb
|
|
|
|
if !got.Eq(want) {
|
|
|
|
t.Error(i, "Want bounds", want, "got", got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestResample(t *testing.T) {
|
|
|
|
for i, im := range makeImages(orig) {
|
|
|
|
m := Resample(im, orig, thumb.Dx(), thumb.Dy())
|
|
|
|
got, want := m.Bounds(), thumb
|
|
|
|
if !got.Eq(want) {
|
|
|
|
t.Error(i, "Want bounds", want, "got", got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, d := range []struct {
|
|
|
|
wantFn string
|
|
|
|
r image.Rectangle
|
|
|
|
w, h int
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
// Generated with imagemagick:
|
|
|
|
// $ convert -crop 128x128+320+160 -resize 64x64 -filter point \
|
|
|
|
// testdata/test.png testdata/test-resample-128x128-64x64.png
|
|
|
|
wantFn: "test-resample-128x128-64x64.png",
|
|
|
|
r: image.Rect(320, 160, 320+128, 160+128),
|
|
|
|
w: 64,
|
|
|
|
h: 64,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
// Generated with imagemagick:
|
|
|
|
// $ convert -resize 128x128 -filter point testdata/test.png \
|
|
|
|
// testdata/test-resample-768x576-128x96.png
|
|
|
|
wantFn: "test-resample-768x576-128x96.png",
|
|
|
|
r: image.Rect(0, 0, 768, 576),
|
|
|
|
w: 128,
|
|
|
|
h: 96,
|
|
|
|
},
|
|
|
|
} {
|
|
|
|
m := image.NewRGBA(testIm.Bounds())
|
|
|
|
fillTestImage(m)
|
|
|
|
r, err := os.Open(filepath.Join("testdata", d.wantFn))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer r.Close()
|
|
|
|
want, err := png.Decode(r)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
got := Resample(m, d.r, d.w, d.h)
|
|
|
|
res := compareImages(got, want)
|
|
|
|
t.Logf("PSNR %.4f", res.psnr)
|
|
|
|
s := got.Bounds().Size()
|
|
|
|
tot := s.X * s.Y
|
|
|
|
per := float32(100*res.diffCnt) / float32(tot)
|
|
|
|
t.Logf("Resample not the same %d pixels different %.2f%%", res.diffCnt, per)
|
|
|
|
if *output != "" {
|
|
|
|
err = savePng(t, want, fmt.Sprintf("Resample.%s->%dx%d.want.png",
|
|
|
|
d.r, d.w, d.h))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
err = savePng(t, got, fmt.Sprintf("Resample.%s->%dx%d.got.png",
|
|
|
|
d.r, d.w, d.h))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
err = savePng(t, res.diffIm,
|
|
|
|
fmt.Sprintf("Resample.%s->%dx%d.diff.png", d.r, d.w, d.h))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestHalveInplace(t *testing.T) {
|
|
|
|
for i, im := range makeImages(orig) {
|
|
|
|
m := HalveInplace(im)
|
|
|
|
b := im.Bounds()
|
|
|
|
got, want := m.Bounds(), image.Rectangle{
|
|
|
|
Min: b.Min,
|
|
|
|
Max: b.Min.Add(b.Max.Div(2)),
|
|
|
|
}
|
|
|
|
if !got.Eq(want) {
|
|
|
|
t.Error(i, "Want bounds", want, "got", got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type results struct {
|
|
|
|
diffCnt int
|
|
|
|
psnr float64
|
|
|
|
diffIm *image.Gray
|
|
|
|
}
|
|
|
|
|
|
|
|
func compareImages(m1, m2 image.Image) results {
|
|
|
|
b := m1.Bounds()
|
|
|
|
s := b.Size()
|
|
|
|
res := results{}
|
|
|
|
mse := uint32(0)
|
|
|
|
for y := b.Min.Y; y < b.Max.Y; y++ {
|
|
|
|
for x := b.Min.X; x < b.Max.X; x++ {
|
|
|
|
r1, g1, b1, a1 := m1.At(x, y).RGBA()
|
|
|
|
r2, g2, b2, a2 := m2.At(x, y).RGBA()
|
|
|
|
|
|
|
|
mse += ((r1-r2)*(r1-r2) + (g1-g2)*(g1-g2) + (b1-b2)*(b1-b2)) / 3
|
|
|
|
if r1 != r2 || g1 != g2 || b1 != b2 || a1 != a2 {
|
|
|
|
if res.diffIm == nil {
|
|
|
|
res.diffIm = image.NewGray(m1.Bounds())
|
|
|
|
}
|
|
|
|
res.diffCnt++
|
|
|
|
res.diffIm.Set(x, y, color.White)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mse = mse / uint32(s.X*s.Y)
|
|
|
|
res.psnr = 20*math.Log10(1<<16) - 10*math.Log10(float64(mse))
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
var testIm image.Image
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
r, err := os.Open(filepath.Join("testdata", "test.png"))
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
defer r.Close()
|
|
|
|
testIm, err = png.Decode(r)
|
|
|
|
}
|
|
|
|
|
|
|
|
func fillTestImage(im image.Image) {
|
|
|
|
b := im.Bounds()
|
|
|
|
if !b.Eq(testIm.Bounds()) {
|
|
|
|
panic("Requested target image dimensions not equal reference image.")
|
|
|
|
}
|
|
|
|
src := testIm
|
|
|
|
if dst, ok := im.(*image.YCbCr); ok {
|
|
|
|
b := testIm.Bounds()
|
|
|
|
for y := b.Min.Y; y < b.Max.Y; y++ {
|
|
|
|
for x := b.Min.X; x < b.Max.X; x++ {
|
|
|
|
r, g, b, _ := src.At(x, y).RGBA()
|
|
|
|
yp, cb, cr := color.RGBToYCbCr(uint8(r), uint8(g), uint8(b))
|
|
|
|
|
|
|
|
dst.Y[dst.YOffset(x, y)] = yp
|
|
|
|
off := dst.COffset(x, y)
|
|
|
|
dst.Cb[off] = cb
|
|
|
|
dst.Cr[off] = cr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
draw.Draw(im.(draw.Image), b, testIm, b.Min, draw.Src)
|
|
|
|
}
|
|
|
|
|
|
|
|
func savePng(t *testing.T, m image.Image, fn string) error {
|
|
|
|
fn = filepath.Join(*output, fn)
|
|
|
|
t.Log("Saving", fn)
|
|
|
|
f, err := os.Create(fn)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
return png.Encode(f, m)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getFilename(im image.Image, method string) string {
|
|
|
|
imgType := fmt.Sprintf("%T", im)
|
|
|
|
imgType = imgType[strings.Index(imgType, ".")+1:]
|
|
|
|
if m, ok := im.(*image.YCbCr); ok {
|
|
|
|
imgType += "." + m.SubsampleRatio.String()
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%s.%s.png", imgType, method)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCompareResizeToHavleInplace(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("Skipping TestCompareResizeToHavleInplace in short mode.")
|
|
|
|
}
|
|
|
|
images1, images2 := []image.Image{}, []image.Image{}
|
|
|
|
for _, im := range makeImages(testIm.Bounds()) {
|
|
|
|
fillTestImage(im)
|
|
|
|
images1 = append(images1, HalveInplace(im))
|
|
|
|
}
|
|
|
|
for _, im := range makeImages(testIm.Bounds()) {
|
|
|
|
fillTestImage(im)
|
|
|
|
s := im.Bounds().Size()
|
|
|
|
images2 = append(images2, Resize(im, im.Bounds(), s.X/2, s.Y/2))
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
f io.WriteCloser
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
if *output != "" {
|
|
|
|
os.Mkdir(*output, os.FileMode(0777))
|
|
|
|
f, err = os.Create(filepath.Join(*output, "index.html"))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
fmt.Fprintf(f, `<!DOCTYPE html>
|
|
|
|
<html lang="en">
|
|
|
|
<head>
|
|
|
|
<meta charset="utf-8">
|
|
|
|
<title>Image comparison for TestCompareResizeToHavleInplace</title>
|
|
|
|
</head>
|
|
|
|
<body style="background-color: grey">
|
|
|
|
<table>
|
|
|
|
`)
|
|
|
|
}
|
|
|
|
for i, im1 := range images1 {
|
|
|
|
im2 := images2[i]
|
|
|
|
res := compareImages(im1, im2)
|
|
|
|
if *output != "" {
|
|
|
|
fmt.Fprintf(f, "<tr>")
|
|
|
|
fn := getFilename(im1, "halve")
|
|
|
|
err := savePng(t, im1, fn)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(f, `<td><img src="%s"><br>%s`, fn, fn)
|
|
|
|
|
|
|
|
fn = getFilename(im1, "resize")
|
|
|
|
err = savePng(t, im2, fn)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(f, `<td><img src="%s"><br>%s`, fn, fn)
|
|
|
|
|
|
|
|
if res.diffIm != nil {
|
|
|
|
fn = getFilename(im1, "diff")
|
|
|
|
err = savePng(t, res.diffIm, fn)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(f, `<td><img src="%s"><br>%s`, fn, fn)
|
|
|
|
}
|
|
|
|
fmt.Fprintln(f)
|
|
|
|
}
|
|
|
|
|
|
|
|
if res.psnr < psnrThreshold {
|
|
|
|
t.Errorf("%T PSNR too low %.4f", im1, res.psnr)
|
|
|
|
} else {
|
|
|
|
t.Logf("%T PSNR %.4f", im1, res.psnr)
|
|
|
|
}
|
|
|
|
s := im1.Bounds().Size()
|
|
|
|
tot := s.X * s.Y
|
|
|
|
if per := float32(100*res.diffCnt) / float32(tot); per > maxPixelDiffPercentage {
|
|
|
|
t.Errorf("%T not the same %d pixels different %.2f%%", im1, res.diffCnt, per)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|