github.com/cybriq/giocore@v0.0.7-0.20210703034601-cfb9cb5f3900/gpu/internal/rendertest/util_test.go (about)

     1  // SPDX-License-Identifier: Unlicense OR MIT
     2  
     3  package rendertest
     4  
     5  import (
     6  	"bytes"
     7  	"flag"
     8  	"fmt"
     9  	"image"
    10  	"image/color"
    11  	"image/draw"
    12  	"image/png"
    13  	"io/ioutil"
    14  	"path/filepath"
    15  	"strconv"
    16  	"testing"
    17  
    18  	"golang.org/x/image/colornames"
    19  
    20  	"github.com/cybriq/giocore/f32"
    21  	"github.com/cybriq/giocore/gpu/headless"
    22  	"github.com/cybriq/giocore/internal/f32color"
    23  	"github.com/cybriq/giocore/op"
    24  	"github.com/cybriq/giocore/op/paint"
    25  )
    26  
    27  var (
    28  	dumpImages   = flag.Bool("saveimages", false, "save test images")
    29  	squares      paint.ImageOp
    30  	smallSquares paint.ImageOp
    31  )
    32  
    33  var (
    34  	red         = f32color.RGBAToNRGBA(colornames.Red)
    35  	green       = f32color.RGBAToNRGBA(colornames.Green)
    36  	blue        = f32color.RGBAToNRGBA(colornames.Blue)
    37  	magenta     = f32color.RGBAToNRGBA(colornames.Magenta)
    38  	black       = f32color.RGBAToNRGBA(colornames.Black)
    39  	white       = f32color.RGBAToNRGBA(colornames.White)
    40  	transparent = color.RGBA{}
    41  )
    42  
    43  func init() {
    44  	squares = buildSquares(512)
    45  	smallSquares = buildSquares(50)
    46  }
    47  
    48  func buildSquares(size int) paint.ImageOp {
    49  	sub := size / 4
    50  	im := image.NewNRGBA(image.Rect(0, 0, size, size))
    51  	c1, c2 := image.NewUniform(colornames.Green), image.NewUniform(colornames.Blue)
    52  	for r := 0; r < 4; r++ {
    53  		for c := 0; c < 4; c++ {
    54  			c1, c2 = c2, c1
    55  			draw.Draw(im, image.Rect(r*sub, c*sub, r*sub+sub, c*sub+sub), c1, image.Point{}, draw.Over)
    56  		}
    57  		c1, c2 = c2, c1
    58  	}
    59  	return paint.NewImageOp(im)
    60  }
    61  
    62  func drawImage(t *testing.T, size int, ops *op.Ops, draw func(o *op.Ops)) (im *image.RGBA, err error) {
    63  	sz := image.Point{X: size, Y: size}
    64  	w := newWindow(t, sz.X, sz.Y)
    65  	draw(ops)
    66  	if err := w.Frame(ops); err != nil {
    67  		return nil, err
    68  	}
    69  	return w.Screenshot()
    70  }
    71  
    72  func run(t *testing.T, f func(o *op.Ops), c func(r result)) {
    73  	// Draw a few times and check that it is correct each time, to
    74  	// ensure any caching effects still generate the correct images.
    75  	var img *image.RGBA
    76  	var err error
    77  	ops := new(op.Ops)
    78  	for i := 0; i < 3; i++ {
    79  		ops.Reset()
    80  		img, err = drawImage(t, 128, ops, f)
    81  		if err != nil {
    82  			t.Error("error rendering:", err)
    83  			return
    84  		}
    85  		// Check for a reference image and make sure it is identical.
    86  		if !verifyRef(t, img, 0) {
    87  			name := fmt.Sprintf("%s-%d-bad.png", t.Name(), i)
    88  			if err := saveImage(name, img); err != nil {
    89  				t.Error(err)
    90  			}
    91  		}
    92  		c(result{t: t, img: img})
    93  	}
    94  
    95  	if *dumpImages {
    96  		if err := saveImage(t.Name()+".png", img); err != nil {
    97  			t.Error(err)
    98  		}
    99  	}
   100  }
   101  
   102  func frame(f func(o *op.Ops), c func(r result)) frameT {
   103  	return frameT{f: f, c: c}
   104  }
   105  
   106  type frameT struct {
   107  	f func(o *op.Ops)
   108  	c func(r result)
   109  }
   110  
   111  // multiRun is used to run test cases over multiple frames, typically
   112  // to test caching interactions.
   113  func multiRun(t *testing.T, frames ...frameT) {
   114  	// draw a few times and check that it is correct each time, to
   115  	// ensure any caching effects still generate the correct images.
   116  	var img *image.RGBA
   117  	var err error
   118  	sz := image.Point{X: 128, Y: 128}
   119  	w := newWindow(t, sz.X, sz.Y)
   120  	ops := new(op.Ops)
   121  	for i := range frames {
   122  		ops.Reset()
   123  		frames[i].f(ops)
   124  		if err := w.Frame(ops); err != nil {
   125  			t.Errorf("rendering failed: %v", err)
   126  			continue
   127  		}
   128  		img, err = w.Screenshot()
   129  		if err != nil {
   130  			t.Errorf("screenshot failed: %v", err)
   131  			continue
   132  		}
   133  		// Check for a reference image and make sure they are identical.
   134  		ok := verifyRef(t, img, i)
   135  		if frames[i].c != nil {
   136  			frames[i].c(result{t: t, img: img})
   137  		}
   138  		if *dumpImages || !ok {
   139  			name := t.Name() + ".png"
   140  			if i != 0 {
   141  				name = t.Name() + "_" + strconv.Itoa(i) + ".png"
   142  			}
   143  			if err := saveImage(name, img); err != nil {
   144  				t.Error(err)
   145  			}
   146  		}
   147  	}
   148  
   149  }
   150  
   151  func verifyRef(t *testing.T, img *image.RGBA, frame int) (ok bool) {
   152  	// ensure identical to ref data
   153  	path := filepath.Join("refs", t.Name()+".png")
   154  	if frame != 0 {
   155  		path = filepath.Join("refs", t.Name()+"_"+strconv.Itoa(frame)+".png")
   156  	}
   157  	b, err := ioutil.ReadFile(path)
   158  	if err != nil {
   159  		t.Error("could not open ref:", err)
   160  		return
   161  	}
   162  	r, err := png.Decode(bytes.NewReader(b))
   163  	if err != nil {
   164  		t.Error("could not decode ref:", err)
   165  		return
   166  	}
   167  	if img.Bounds() != r.Bounds() {
   168  		t.Errorf("reference image is %v, expected %v", r.Bounds(), img.Bounds())
   169  		return false
   170  	}
   171  	var ref *image.RGBA
   172  	switch r := r.(type) {
   173  	case *image.RGBA:
   174  		ref = r
   175  	case *image.NRGBA:
   176  		ref = image.NewRGBA(r.Bounds())
   177  		bnd := r.Bounds()
   178  		for x := bnd.Min.X; x < bnd.Max.X; x++ {
   179  			for y := bnd.Min.Y; y < bnd.Max.Y; y++ {
   180  				ref.SetRGBA(x, y, f32color.NRGBAToRGBA(r.NRGBAAt(x, y)))
   181  			}
   182  		}
   183  	default:
   184  		t.Fatalf("reference image is a %T, expected *image.NRGBA or *image.RGBA", r)
   185  	}
   186  	bnd := img.Bounds()
   187  	for x := bnd.Min.X; x < bnd.Max.X; x++ {
   188  		for y := bnd.Min.Y; y < bnd.Max.Y; y++ {
   189  			exp := ref.RGBAAt(x, y)
   190  			got := img.RGBAAt(x, y)
   191  			if !colorsClose(exp, got) {
   192  				t.Error("not equal to ref at", x, y, " ", got, exp)
   193  				return false
   194  			}
   195  		}
   196  	}
   197  	return true
   198  }
   199  
   200  func colorsClose(c1, c2 color.RGBA) bool {
   201  	const delta = 0.01 // magic value obtained from experimentation.
   202  	return yiqEqApprox(c1, c2, delta)
   203  }
   204  
   205  // yiqEqApprox compares the colors of 2 pixels, in the NTSC YIQ color space,
   206  // as described in:
   207  //
   208  //   Measuring perceived color difference using YIQ NTSC
   209  //   transmission color space in mobile applications.
   210  //   Yuriy Kotsarenko, Fernando Ramos.
   211  //
   212  // An electronic version is available at:
   213  //
   214  // - http://www.progmat.uaem.mx:8080/artVol2Num2/Articulo3Vol2Num2.pdf
   215  func yiqEqApprox(c1, c2 color.RGBA, d2 float64) bool {
   216  	const max = 35215.0 // difference between 2 maximally different pixels.
   217  
   218  	var (
   219  		r1 = float64(c1.R)
   220  		g1 = float64(c1.G)
   221  		b1 = float64(c1.B)
   222  
   223  		r2 = float64(c2.R)
   224  		g2 = float64(c2.G)
   225  		b2 = float64(c2.B)
   226  
   227  		y1 = r1*0.29889531 + g1*0.58662247 + b1*0.11448223
   228  		i1 = r1*0.59597799 - g1*0.27417610 - b1*0.32180189
   229  		q1 = r1*0.21147017 - g1*0.52261711 + b1*0.31114694
   230  
   231  		y2 = r2*0.29889531 + g2*0.58662247 + b2*0.11448223
   232  		i2 = r2*0.59597799 - g2*0.27417610 - b2*0.32180189
   233  		q2 = r2*0.21147017 - g2*0.52261711 + b2*0.31114694
   234  
   235  		y = y1 - y2
   236  		i = i1 - i2
   237  		q = q1 - q2
   238  
   239  		diff = 0.5053*y*y + 0.299*i*i + 0.1957*q*q
   240  	)
   241  	return diff <= max*d2
   242  }
   243  
   244  func (r result) expect(x, y int, col color.RGBA) {
   245  	r.t.Helper()
   246  	if r.img == nil {
   247  		return
   248  	}
   249  	c := r.img.RGBAAt(x, y)
   250  	if !colorsClose(c, col) {
   251  		r.t.Error("expected ", col, " at ", "(", x, ",", y, ") but got ", c)
   252  	}
   253  }
   254  
   255  type result struct {
   256  	t   *testing.T
   257  	img *image.RGBA
   258  }
   259  
   260  func saveImage(file string, img *image.RGBA) error {
   261  	// Only NRGBA images are losslessly encoded by png.Encode.
   262  	nrgba := image.NewNRGBA(img.Bounds())
   263  	bnd := img.Bounds()
   264  	for x := bnd.Min.X; x < bnd.Max.X; x++ {
   265  		for y := bnd.Min.Y; y < bnd.Max.Y; y++ {
   266  			nrgba.SetNRGBA(x, y, f32color.RGBAToNRGBA(img.RGBAAt(x, y)))
   267  		}
   268  	}
   269  	var buf bytes.Buffer
   270  	if err := png.Encode(&buf, nrgba); err != nil {
   271  		return err
   272  	}
   273  	return ioutil.WriteFile(file, buf.Bytes(), 0666)
   274  }
   275  
   276  func newWindow(t testing.TB, width, height int) *headless.Window {
   277  	w, err := headless.NewWindow(width, height)
   278  	if err != nil {
   279  		t.Skipf("failed to create headless window, skipping: %v", err)
   280  	}
   281  	t.Cleanup(w.Release)
   282  	return w
   283  }
   284  
   285  func scale(sx, sy float32) op.TransformOp {
   286  	return op.Affine(f32.Affine2D{}.Scale(f32.Point{}, f32.Pt(sx, sy)))
   287  }