2024-10-18 16:12:35 -07:00
|
|
|
package imageproc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"image"
|
|
|
|
"image/png"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestAspectRatios(t *testing.T) {
|
|
|
|
type aspectCase struct {
|
|
|
|
MaxTiles int
|
|
|
|
Expected []image.Point
|
|
|
|
}
|
|
|
|
|
|
|
|
cases := []aspectCase{
|
|
|
|
{
|
|
|
|
MaxTiles: 1,
|
|
|
|
Expected: []image.Point{{1, 1}},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
MaxTiles: 2,
|
|
|
|
Expected: []image.Point{{1, 1}, {1, 2}, {2, 1}},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
MaxTiles: 3,
|
|
|
|
Expected: []image.Point{{1, 1}, {1, 2}, {1, 3}, {2, 1}, {3, 1}},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
MaxTiles: 4,
|
|
|
|
Expected: []image.Point{{1, 1}, {1, 2}, {1, 3}, {1, 4}, {2, 1}, {2, 2}, {3, 1}, {4, 1}},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range cases {
|
|
|
|
actual := GetSupportedAspectRatios(c.MaxTiles)
|
|
|
|
|
|
|
|
if diff := cmp.Diff(actual, c.Expected); diff != "" {
|
|
|
|
t.Errorf("mismatch (-got +want):\n%s", diff)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetImageSizeFitToCanvas(t *testing.T) {
|
|
|
|
type imageSizeCase struct {
|
|
|
|
ImageRect image.Point
|
|
|
|
CanvasRect image.Point
|
|
|
|
TileSize int
|
|
|
|
Expected image.Point
|
|
|
|
}
|
|
|
|
|
|
|
|
cases := []imageSizeCase{
|
|
|
|
{
|
|
|
|
ImageRect: image.Point{400, 400},
|
|
|
|
CanvasRect: image.Point{640, 480},
|
|
|
|
TileSize: 200,
|
|
|
|
Expected: image.Point{400, 400},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageRect: image.Point{1024, 768},
|
|
|
|
CanvasRect: image.Point{640, 480},
|
|
|
|
TileSize: 200,
|
|
|
|
Expected: image.Point{640, 480},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageRect: image.Point{500, 500},
|
|
|
|
CanvasRect: image.Point{1000, 1000},
|
|
|
|
TileSize: 750,
|
|
|
|
Expected: image.Point{750, 750},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageRect: image.Point{500, 1000},
|
|
|
|
CanvasRect: image.Point{2000, 2000},
|
|
|
|
TileSize: 2000,
|
|
|
|
Expected: image.Point{1000, 2000},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageRect: image.Point{4000, 3000},
|
|
|
|
CanvasRect: image.Point{2000, 1000},
|
|
|
|
TileSize: 1000,
|
|
|
|
Expected: image.Point{1333, 1000},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageRect: image.Point{667, 1000},
|
|
|
|
CanvasRect: image.Point{1000, 1000},
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{667, 1000},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range cases {
|
|
|
|
actual := getImageSizeFitToCanvas(c.ImageRect, c.CanvasRect, c.TileSize)
|
|
|
|
|
|
|
|
if actual != c.Expected {
|
|
|
|
t.Errorf("incorrect image rect: '%#v'. expected: '%#v'", actual, c.Expected)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetOptimalTiledCanvas(t *testing.T) {
|
|
|
|
type tiledCanvasSizeCase struct {
|
|
|
|
ImageSize image.Point
|
|
|
|
MaxImageTiles int
|
|
|
|
TileSize int
|
|
|
|
Expected image.Point
|
|
|
|
}
|
|
|
|
|
|
|
|
cases := []tiledCanvasSizeCase{
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{1024, 768},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 1000,
|
|
|
|
Expected: image.Point{2000, 1000},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{1024, 768},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{1120, 1120},
|
2024-10-29 16:28:02 -07:00
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{800, 600},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{1120, 1120},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{640, 480},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{1120, 560},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{320, 200},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{560, 560},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{1320, 200},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{1680, 560},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{2000, 200},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{2240, 560},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{10000, 200},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{2240, 560},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{480, 640},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{560, 1120},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{200, 320},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{560, 560},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{200, 1320},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{560, 1680},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{200, 2000},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{560, 2240},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{200, 10000},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{560, 2240},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ImageSize: image.Point{10000, 10000},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
TileSize: 560,
|
|
|
|
Expected: image.Point{1120, 1120},
|
2024-10-18 16:12:35 -07:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range cases {
|
|
|
|
actual := getOptimalTiledCanvas(c.ImageSize, c.MaxImageTiles, c.TileSize)
|
|
|
|
|
|
|
|
if actual != c.Expected {
|
|
|
|
t.Errorf("incorrect tiled canvas: '%#v'. expected: '%#v'", actual, c.Expected)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSplitToTiles(t *testing.T) {
|
|
|
|
type splitCase struct {
|
|
|
|
TestImage image.Image
|
|
|
|
NumTilesSize image.Point
|
|
|
|
Expected []image.Image
|
|
|
|
}
|
|
|
|
|
|
|
|
cases := []splitCase{
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 1024, 768)),
|
|
|
|
NumTilesSize: image.Point{1, 1},
|
|
|
|
Expected: []image.Image{image.NewRGBA(image.Rect(0, 0, 1024, 768))},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 1000, 500)),
|
|
|
|
NumTilesSize: image.Point{2, 1},
|
|
|
|
Expected: []image.Image{
|
|
|
|
image.NewRGBA(image.Rect(0, 0, 500, 500)),
|
|
|
|
image.NewRGBA(image.Rect(500, 0, 1000, 500)),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 1000, 1000)),
|
|
|
|
NumTilesSize: image.Point{2, 2},
|
|
|
|
Expected: []image.Image{
|
|
|
|
image.NewRGBA(image.Rect(0, 0, 500, 500)),
|
|
|
|
image.NewRGBA(image.Rect(500, 0, 1000, 500)),
|
|
|
|
image.NewRGBA(image.Rect(0, 500, 500, 1000)),
|
|
|
|
image.NewRGBA(image.Rect(500, 500, 1000, 1000)),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range cases {
|
|
|
|
actual := splitToTiles(c.TestImage, c.NumTilesSize)
|
|
|
|
|
|
|
|
if len(actual) != len(c.Expected) {
|
|
|
|
t.Errorf("incorrect number of images '%d': expected: '%d'", len(actual), len(c.Expected))
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := range actual {
|
|
|
|
if actual[i].Bounds() != c.Expected[i].Bounds() {
|
|
|
|
t.Errorf("image size incorrect: '%#v': expected: '%#v'", actual[i].Bounds(), c.Expected[i].Bounds())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestResize(t *testing.T) {
|
|
|
|
type resizeCase struct {
|
|
|
|
TestImage image.Image
|
|
|
|
OutputSize image.Point
|
|
|
|
MaxImageTiles int
|
|
|
|
ExpectedImage image.Image
|
|
|
|
ExpectedAspectRatio image.Point
|
|
|
|
}
|
|
|
|
|
|
|
|
cases := []resizeCase{
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 200, 200)),
|
|
|
|
OutputSize: image.Point{100, 100},
|
|
|
|
MaxImageTiles: 1,
|
|
|
|
ExpectedImage: image.NewRGBA(image.Rect(0, 0, 100, 100)),
|
|
|
|
ExpectedAspectRatio: image.Point{1, 1},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 200, 200)),
|
|
|
|
OutputSize: image.Point{100, 100},
|
|
|
|
MaxImageTiles: 2,
|
|
|
|
ExpectedImage: image.NewRGBA(image.Rect(0, 0, 100, 100)),
|
|
|
|
ExpectedAspectRatio: image.Point{1, 1},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 10, 10)),
|
|
|
|
OutputSize: image.Point{560, 560},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
ExpectedImage: image.NewRGBA(image.Rect(0, 0, 560, 560)),
|
|
|
|
ExpectedAspectRatio: image.Point{1, 1},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 2560, 1920)),
|
|
|
|
OutputSize: image.Point{560, 560},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
ExpectedImage: image.NewRGBA(image.Rect(0, 0, 1120, 840)),
|
|
|
|
ExpectedAspectRatio: image.Point{2, 2},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 1024, 768)),
|
|
|
|
OutputSize: image.Point{560, 560},
|
|
|
|
MaxImageTiles: 4,
|
|
|
|
ExpectedImage: image.NewRGBA(image.Rect(0, 0, 1024, 768)),
|
|
|
|
ExpectedAspectRatio: image.Point{2, 2},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range cases {
|
|
|
|
actualImage, actualAspectRatio := ResizeImage(c.TestImage, "png", c.OutputSize, c.MaxImageTiles)
|
|
|
|
|
|
|
|
if actualImage.Bounds() != c.ExpectedImage.Bounds() {
|
|
|
|
t.Errorf("image size incorrect: '%#v': expected: '%#v'", actualImage.Bounds(), c.ExpectedImage.Bounds())
|
|
|
|
}
|
|
|
|
|
|
|
|
if actualAspectRatio != c.ExpectedAspectRatio {
|
|
|
|
t.Errorf("aspect ratio incorrect: '%#v': expected: '%#v'", actualAspectRatio, c.ExpectedAspectRatio)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPad(t *testing.T) {
|
|
|
|
type padCase struct {
|
|
|
|
TestImage image.Image
|
|
|
|
OutputSize image.Point
|
|
|
|
AspectRatio image.Point
|
|
|
|
Expected image.Image
|
|
|
|
}
|
|
|
|
|
|
|
|
cases := []padCase{
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 1000, 667)),
|
|
|
|
OutputSize: image.Point{560, 560},
|
|
|
|
AspectRatio: image.Point{2, 2},
|
|
|
|
Expected: image.NewRGBA(image.Rect(0, 0, 1120, 1120)),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range cases {
|
|
|
|
actual := PadImage(c.TestImage, c.OutputSize, c.AspectRatio)
|
|
|
|
|
|
|
|
if actual.Bounds() != c.Expected.Bounds() {
|
|
|
|
t.Errorf("image size incorrect: '%#v': expected: '%#v'", actual.Bounds(), c.Expected.Bounds())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPackImages(t *testing.T) {
|
|
|
|
type packCase struct {
|
|
|
|
TestImage image.Image
|
|
|
|
AspectRatio image.Point
|
|
|
|
ExpectedVals int
|
|
|
|
}
|
|
|
|
|
|
|
|
mean := [3]float32{0.48145466, 0.4578275, 0.40821073}
|
|
|
|
std := [3]float32{0.26862954, 0.26130258, 0.27577711}
|
|
|
|
|
|
|
|
cases := []packCase{
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 1120, 1120)),
|
|
|
|
AspectRatio: image.Point{2, 2},
|
|
|
|
ExpectedVals: 2 * 2 * 3 * 560 * 560,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 560, 560)),
|
|
|
|
AspectRatio: image.Point{1, 1},
|
|
|
|
ExpectedVals: 1 * 1 * 3 * 560 * 560,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 1120, 560)),
|
|
|
|
AspectRatio: image.Point{1, 2},
|
|
|
|
ExpectedVals: 1 * 2 * 3 * 560 * 560,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range cases {
|
|
|
|
actualVals := PackImages(c.TestImage, c.AspectRatio, mean, std)
|
|
|
|
if len(actualVals) != c.ExpectedVals {
|
|
|
|
t.Errorf("packed image size incorrect: '%d': expected: '%d'", len(actualVals), c.ExpectedVals)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPreprocess(t *testing.T) {
|
|
|
|
type preprocessCase struct {
|
|
|
|
TestImage image.Image
|
|
|
|
ExpectedVals int
|
|
|
|
ExpectedAspectRatioID int
|
|
|
|
}
|
|
|
|
|
|
|
|
cases := []preprocessCase{
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 10, 10)),
|
|
|
|
ExpectedVals: 0,
|
|
|
|
ExpectedAspectRatioID: 1,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TestImage: image.NewRGBA(image.Rect(0, 0, 1024, 768)),
|
|
|
|
ExpectedVals: 0,
|
|
|
|
ExpectedAspectRatioID: 6,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range cases {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
err := png.Encode(&buf, c.TestImage)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
imgData, aspectRatioID, err := Preprocess(buf.Bytes())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("error processing: %q", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(imgData) == 0 {
|
|
|
|
t.Errorf("no image data returned")
|
|
|
|
}
|
|
|
|
|
|
|
|
if aspectRatioID != c.ExpectedAspectRatioID {
|
|
|
|
t.Errorf("aspect ratio incorrect: '%d': expected: '%d'", aspectRatioID, c.ExpectedAspectRatioID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|