| // Copyright 2016 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package iconvg |
| |
| import ( |
| "bytes" |
| "fmt" |
| "image" |
| "image/color" |
| "image/draw" |
| "image/png" |
| "os" |
| "path/filepath" |
| "runtime" |
| "strings" |
| "testing" |
| |
| "golang.org/x/image/math/f32" |
| ) |
| |
| // disassemble returns a disassembly of an encoded IconVG graphic. Users of |
| // this package aren't expected to want to do this, so it lives in a _test.go |
| // file, but it can be useful for debugging. |
| func disassemble(src []byte) ([]byte, error) { |
| w := new(bytes.Buffer) |
| p := func(b []byte, format string, args ...interface{}) { |
| const hex = "0123456789abcdef" |
| var buf [14]byte |
| for i := range buf { |
| buf[i] = ' ' |
| } |
| for i, x := range b { |
| buf[3*i+0] = hex[x>>4] |
| buf[3*i+1] = hex[x&0x0f] |
| } |
| w.Write(buf[:]) |
| fmt.Fprintf(w, format, args...) |
| } |
| m := Metadata{} |
| if err := decode(nil, p, &m, false, buffer(src), nil); err != nil { |
| return nil, err |
| } |
| return w.Bytes(), nil |
| } |
| |
| var ( |
| _ Destination = (*Encoder)(nil) |
| _ Destination = (*Rasterizer)(nil) |
| ) |
| |
| func encodePNG(dstFilename string, src image.Image) error { |
| f, err := os.Create(dstFilename) |
| if err != nil { |
| return err |
| } |
| encErr := png.Encode(f, src) |
| closeErr := f.Close() |
| if encErr != nil { |
| return encErr |
| } |
| return closeErr |
| } |
| |
| func decodePNG(srcFilename string) (image.Image, error) { |
| f, err := os.Open(srcFilename) |
| if err != nil { |
| return nil, err |
| } |
| defer f.Close() |
| return png.Decode(f) |
| } |
| |
| func checkApproxEqual(m0, m1 image.Image) error { |
| diff := func(a, b uint32) uint32 { |
| if a < b { |
| return b - a |
| } |
| return a - b |
| } |
| |
| bounds0 := m0.Bounds() |
| bounds1 := m1.Bounds() |
| if bounds0 != bounds1 { |
| return fmt.Errorf("bounds differ: got %v, want %v", bounds0, bounds1) |
| } |
| for y := bounds0.Min.Y; y < bounds0.Max.Y; y++ { |
| for x := bounds0.Min.X; x < bounds0.Max.X; x++ { |
| r0, g0, b0, a0 := m0.At(x, y).RGBA() |
| r1, g1, b1, a1 := m1.At(x, y).RGBA() |
| |
| // TODO: be more principled in picking this magic threshold, other |
| // than what the difference is, in practice, in x/image/vector's |
| // fixed and floating point rasterizer? |
| const D = 0xffff * 12 / 100 // Diff threshold of 12%. |
| |
| if diff(r0, r1) > D || diff(g0, g1) > D || diff(b0, b1) > D || diff(a0, a1) > D { |
| return fmt.Errorf("at (%d, %d):\n"+ |
| "got RGBA %#04x, %#04x, %#04x, %#04x\n"+ |
| "want RGBA %#04x, %#04x, %#04x, %#04x", |
| x, y, r0, g0, b0, a0, r1, g1, b1, a1) |
| } |
| } |
| } |
| return nil |
| } |
| |
| func diffLines(t *testing.T, got, want string) { |
| gotLines := strings.Split(got, "\n") |
| wantLines := strings.Split(want, "\n") |
| for i := 1; ; i++ { |
| if len(gotLines) == 0 { |
| t.Errorf("line %d:\ngot %q\nwant %q", i, "", wantLines[0]) |
| return |
| } |
| if len(wantLines) == 0 { |
| t.Errorf("line %d:\ngot %q\nwant %q", i, gotLines[0], "") |
| return |
| } |
| g, w := gotLines[0], wantLines[0] |
| gotLines = gotLines[1:] |
| wantLines = wantLines[1:] |
| if g != w { |
| t.Errorf("line %d:\ngot %q\nwant %q", i, g, w) |
| return |
| } |
| } |
| } |
| |
| var testdataTestCases = []struct { |
| filename string |
| variants string |
| }{ |
| {"testdata/action-info.lores", ""}, |
| {"testdata/action-info.hires", ""}, |
| {"testdata/arcs", ""}, |
| {"testdata/blank", ""}, |
| {"testdata/cowbell", ""}, |
| {"testdata/elliptical", ""}, |
| {"testdata/favicon", ";pink"}, |
| {"testdata/gradient", ""}, |
| {"testdata/lod-polygon", ";64"}, |
| {"testdata/video-005.primitive", ""}, |
| } |
| |
| func TestDisassembly(t *testing.T) { |
| for _, tc := range testdataTestCases { |
| ivgData, err := os.ReadFile(filepath.FromSlash(tc.filename) + ".ivg") |
| if err != nil { |
| t.Errorf("%s: ReadFile: %v", tc.filename, err) |
| continue |
| } |
| got, err := disassemble(ivgData) |
| if err != nil { |
| t.Errorf("%s: disassemble: %v", tc.filename, err) |
| continue |
| } |
| wantFilename := filepath.FromSlash(tc.filename) + ".ivg.disassembly" |
| if *updateFlag { |
| if err := os.WriteFile(filepath.FromSlash(wantFilename), got, 0666); err != nil { |
| t.Errorf("%s: WriteFile: %v", tc.filename, err) |
| } |
| continue |
| } |
| want, err := os.ReadFile(wantFilename) |
| if err != nil { |
| t.Errorf("%s: ReadFile: %v", tc.filename, err) |
| continue |
| } |
| if !bytes.Equal(got, want) { |
| t.Errorf("%s: got:\n%s\nwant:\n%s", tc.filename, got, want) |
| diffLines(t, string(got), string(want)) |
| } |
| } |
| } |
| |
| // The IconVG decoder and encoder are expected to be completely deterministic, |
| // so check that we get the original bytes after a decode + encode round-trip. |
| func TestDecodeEncodeRoundTrip(t *testing.T) { |
| for _, tc := range testdataTestCases { |
| ivgData, err := os.ReadFile(filepath.FromSlash(tc.filename) + ".ivg") |
| if err != nil { |
| t.Errorf("%s: ReadFile: %v", tc.filename, err) |
| continue |
| } |
| var e resolutionPreservingEncoder |
| e.HighResolutionCoordinates = strings.HasSuffix(tc.filename, ".hires") |
| if err := Decode(&e, ivgData, nil); err != nil { |
| t.Errorf("%s: Decode: %v", tc.filename, err) |
| continue |
| } |
| got, err := e.Bytes() |
| if err != nil { |
| t.Errorf("%s: Encoder.Bytes: %v", tc.filename, err) |
| continue |
| } |
| if want := ivgData; !bytes.Equal(got, want) { |
| t.Errorf("%s:\ngot %d bytes (on GOOS=%s GOARCH=%s, using compiler %q):\n% x\nwant %d bytes:\n% x", |
| tc.filename, len(got), runtime.GOOS, runtime.GOARCH, runtime.Compiler, got, len(want), want) |
| gotDisasm, err1 := disassemble(got) |
| wantDisasm, err2 := disassemble(want) |
| if err1 == nil && err2 == nil { |
| diffLines(t, string(gotDisasm), string(wantDisasm)) |
| } |
| } |
| } |
| } |
| |
| // resolutionPreservingEncoder is an Encoder |
| // whose Reset method keeps prior resolution. |
| type resolutionPreservingEncoder struct { |
| Encoder |
| } |
| |
| // Reset resets the Encoder for the given Metadata. |
| // |
| // Unlike Encoder.Reset, it leaves the value |
| // of e.HighResolutionCoordinates unmodified. |
| func (e *resolutionPreservingEncoder) Reset(m Metadata) { |
| orig := e.HighResolutionCoordinates |
| e.Encoder.Reset(m) |
| e.HighResolutionCoordinates = orig |
| } |
| |
| func TestDecodeAndRasterize(t *testing.T) { |
| for _, tc := range testdataTestCases { |
| ivgData, err := os.ReadFile(filepath.FromSlash(tc.filename) + ".ivg") |
| if err != nil { |
| t.Errorf("%s: ReadFile: %v", tc.filename, err) |
| continue |
| } |
| md, err := DecodeMetadata(ivgData) |
| if err != nil { |
| t.Errorf("%s: DecodeMetadata: %v", tc.filename, err) |
| continue |
| } |
| |
| for _, variant := range strings.Split(tc.variants, ";") { |
| length := 256 |
| if variant == "64" { |
| length = 64 |
| } |
| width, height := length, length |
| if dx, dy := md.ViewBox.AspectRatio(); dx < dy { |
| width = int(float32(length) * dx / dy) |
| } else { |
| height = int(float32(length) * dy / dx) |
| } |
| |
| opts := &DecodeOptions{} |
| if variant == "pink" { |
| pal := DefaultPalette |
| pal[0] = color.RGBA{0xfe, 0x76, 0xea, 0xff} |
| opts.Palette = &pal |
| } |
| |
| got := image.NewRGBA(image.Rect(0, 0, width, height)) |
| var z Rasterizer |
| z.SetDstImage(got, got.Bounds(), draw.Src) |
| if err := Decode(&z, ivgData, opts); err != nil { |
| t.Errorf("%s %q variant: Decode: %v", tc.filename, variant, err) |
| continue |
| } |
| |
| wantFilename := filepath.FromSlash(tc.filename) |
| if variant != "" { |
| wantFilename += "." + variant |
| } |
| wantFilename += ".png" |
| if *updateFlag { |
| if err := encodePNG(filepath.FromSlash(wantFilename), got); err != nil { |
| t.Errorf("%s %q variant: encodePNG: %v", tc.filename, variant, err) |
| } |
| continue |
| } |
| want, err := decodePNG(wantFilename) |
| if err != nil { |
| t.Errorf("%s %q variant: decodePNG: %v", tc.filename, variant, err) |
| continue |
| } |
| if err := checkApproxEqual(got, want); err != nil { |
| t.Errorf("%s %q variant: %v", tc.filename, variant, err) |
| continue |
| } |
| } |
| } |
| } |
| |
| func TestInvalidAlphaPremultipliedColor(t *testing.T) { |
| // See http://golang.org/issue/39526 for some discussion. |
| |
| dst := image.NewRGBA(image.Rect(0, 0, 1, 1)) |
| var z Rasterizer |
| z.SetDstImage(dst, dst.Bounds(), draw.Over) |
| z.Reset(Metadata{ |
| ViewBox: Rectangle{ |
| Min: f32.Vec2{0.0, 0.0}, |
| Max: f32.Vec2{1.0, 1.0}, |
| }, |
| }) |
| |
| // Fill the unit square with red. |
| z.SetCReg(0, false, RGBAColor(color.RGBA{0x55, 0x00, 0x00, 0x66})) |
| z.StartPath(0, 0.0, 0.0) |
| z.AbsLineTo(1.0, 0.0) |
| z.AbsLineTo(1.0, 1.0) |
| z.AbsLineTo(0.0, 1.0) |
| z.ClosePathEndPath() |
| |
| // Fill the unit square with an invalid (non-gradient) alpha-premultiplied |
| // color (super-saturated green). This should be a no-op (and not crash). |
| z.SetCReg(0, false, RGBAColor(color.RGBA{0x00, 0x99, 0x00, 0x88})) |
| z.StartPath(0, 0.0, 0.0) |
| z.AbsLineTo(1.0, 0.0) |
| z.AbsLineTo(1.0, 1.0) |
| z.AbsLineTo(0.0, 1.0) |
| z.ClosePathEndPath() |
| |
| // We should see red. |
| got := dst.Pix |
| want := []byte{0x55, 0x00, 0x00, 0x66} |
| if !bytes.Equal(got, want) { |
| t.Errorf("got [% 02x], want [% 02x]", got, want) |
| } |
| } |
| |
| func TestBlendColor(t *testing.T) { |
| // This example comes from doc.go. Look for "orange" in the "Colors" |
| // section. |
| pal := Palette{ |
| 2: color.RGBA{0xff, 0xcc, 0x80, 0xff}, // "Material Design Orange 200". |
| } |
| cReg := [64]color.RGBA{} |
| got := BlendColor(0x40, 0x7f, 0x82).Resolve(&pal, &cReg) |
| want := color.RGBA{0x40, 0x33, 0x20, 0x40} // 25% opaque "Orange 200", alpha-premultiplied. |
| if got != want { |
| t.Errorf("\ngot %x\nwant %x", got, want) |
| } |
| } |