250 lines
5.2 KiB
Go
250 lines
5.2 KiB
Go
package bitmap_test
|
|
|
|
import (
|
|
"fmt"
|
|
"math/rand"
|
|
"os"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/iliadenisov/galaxy/pkg/bitmap"
|
|
)
|
|
|
|
func TestBitVectorSize(t *testing.T) {
|
|
type testCase struct {
|
|
width, height uint32
|
|
expectSize int
|
|
}
|
|
for _, tc := range []testCase{
|
|
{
|
|
width: 10,
|
|
height: 10,
|
|
expectSize: 4,
|
|
},
|
|
{
|
|
width: 1,
|
|
height: 1,
|
|
expectSize: 1,
|
|
},
|
|
{
|
|
width: 32,
|
|
height: 32,
|
|
expectSize: 32,
|
|
},
|
|
} {
|
|
t.Run(fmt.Sprintf("w=%d h=%d s=%d", tc.width, tc.height, tc.expectSize), func(t *testing.T) {
|
|
bm := bitmap.NewBitmap(tc.width, tc.height)
|
|
l := len(bitmap.Value(bm))
|
|
if tc.expectSize != l {
|
|
t.Errorf("expected bitmap size: %d, got: %d", tc.expectSize, l)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestSetPixel(t *testing.T) {
|
|
type coord struct {
|
|
x, y int
|
|
}
|
|
type testCase struct {
|
|
width, height uint32
|
|
pixels []coord
|
|
bits []int
|
|
}
|
|
asMap := func(bits []int) map[int]bool {
|
|
result := make(map[int]bool)
|
|
for i := range bits {
|
|
if _, ok := result[bits[i]]; ok {
|
|
t.Fatalf("source bits duplicate at idx=%d", i)
|
|
} else {
|
|
result[bits[i]] = true
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
asUint32 := func(v bool) uint32 { return map[bool]uint32{true: 1, false: 0}[v] }
|
|
for i, tc := range []testCase{
|
|
{
|
|
width: 5,
|
|
height: 5,
|
|
pixels: []coord{{0, 0}},
|
|
bits: []int{0},
|
|
},
|
|
{
|
|
width: 5,
|
|
height: 5,
|
|
pixels: []coord{{1, 0}},
|
|
bits: []int{1},
|
|
},
|
|
{
|
|
width: 5,
|
|
height: 5,
|
|
pixels: []coord{{2, 0}},
|
|
bits: []int{2},
|
|
},
|
|
{
|
|
width: 5,
|
|
height: 5,
|
|
pixels: []coord{{0, 1}},
|
|
bits: []int{5},
|
|
},
|
|
{
|
|
width: 5,
|
|
height: 5,
|
|
pixels: []coord{{4, 4}},
|
|
bits: []int{24},
|
|
},
|
|
{
|
|
width: 8,
|
|
height: 8,
|
|
pixels: []coord{{7, 7}},
|
|
bits: []int{63},
|
|
},
|
|
} {
|
|
t.Run(fmt.Sprintf("tc#%d", i), func(t *testing.T) {
|
|
bm := bitmap.NewBitmap(tc.width, tc.height)
|
|
for _, c := range tc.pixels {
|
|
if bm.IsSet(c.x, c.y) {
|
|
t.Errorf("expected pixel to be clear at x=%d y=%d", c.x, c.y)
|
|
}
|
|
bm.Set(c.x, c.y)
|
|
if !bm.IsSet(c.x, c.y) {
|
|
t.Errorf("expected pixel to be set at x=%d y=%d", c.x, c.y)
|
|
}
|
|
}
|
|
bitVector := bitmap.Value(bm)
|
|
bitNum := 0
|
|
expected := asMap(tc.bits)
|
|
for bi := range bitVector {
|
|
for ; bitNum < (bi+1)*32; bitNum++ {
|
|
if (bitVector[bi]>>(bitNum%32))&1 != asUint32(expected[bitNum]) {
|
|
t.Errorf("expected: bit #%d to be %t, got %v", bitNum, expected[bitNum], uint32(1<<(bitNum%32)))
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFreeCount(t *testing.T) {
|
|
bm := bitmap.NewBitmap(10, 10)
|
|
type testCase struct {
|
|
x, y, expect int
|
|
}
|
|
for _, tc := range []testCase{
|
|
{0, 0, 99},
|
|
{5, 5, 98},
|
|
{9, 9, 97},
|
|
{10, 10, 97},
|
|
{15, 6, 96},
|
|
{9, 9, 96},
|
|
{3, 8, 95},
|
|
} {
|
|
t.Run(fmt.Sprintf("x=%d,y=%d", tc.x, tc.y), func(t *testing.T) {
|
|
bm.Set(tc.x, tc.y)
|
|
count := bm.FreeCount()
|
|
if tc.expect != count {
|
|
t.Errorf("expected: %d, actual %d", tc.expect, count)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestSetFreeN(t *testing.T) {
|
|
bm := bitmap.NewBitmap(5, 5)
|
|
type testCase struct {
|
|
x, y, number int
|
|
}
|
|
for i, tc := range []testCase{
|
|
{0, 0, 0},
|
|
{1, 0, 0},
|
|
{4, 0, 2},
|
|
{2, 2, 9},
|
|
{1, 1, 3},
|
|
{3, 1, 4},
|
|
{3, 2, 7},
|
|
{4, 4, 17},
|
|
{3, 0, 1},
|
|
} {
|
|
t.Run(fmt.Sprintf("tc#%d", i), func(t *testing.T) {
|
|
if x, y, err := bm.GetFreeN(tc.number); err != nil {
|
|
t.Errorf("get free by number=%d: %s", tc.number, err)
|
|
} else if tc.x != x || tc.y != y {
|
|
t.Fatalf("expected: x=%d, y=%d by number=%d, got: x=%d, y=%d", tc.x, tc.y, tc.number, x, y)
|
|
}
|
|
if err := bm.SetFreeN(tc.number); err != nil {
|
|
t.Errorf("set free by number=%d: %s", tc.number, err)
|
|
} else if !bm.IsSet(tc.x, tc.y) {
|
|
t.Errorf("expected to be set: free_number=%d @ x=%d,y=%d bitmap:\n%s", tc.number, tc.x, tc.y, bm)
|
|
}
|
|
})
|
|
}
|
|
|
|
bm = bitmap.NewBitmap(2, 2)
|
|
bm.Set(0, 0)
|
|
bm.Set(1, 0)
|
|
bm.Set(0, 1)
|
|
if err := bm.SetFreeN(1); err == nil {
|
|
t.Errorf("expected: error when free pixel number greater than free pixels count")
|
|
}
|
|
if _, _, err := bm.GetFreeN(1); err == nil {
|
|
t.Errorf("expected: error when free pixel number greater than free pixels count")
|
|
}
|
|
bm.Set(1, 1)
|
|
if err := bm.SetFreeN(0); err == nil {
|
|
t.Errorf("expected: error when no free pixels left")
|
|
}
|
|
}
|
|
|
|
func TestClear(t *testing.T) {
|
|
var bm = bitmap.NewBitmap(10, 10)
|
|
for range 50 {
|
|
bm.Set(rand.Intn(10), rand.Intn(10))
|
|
}
|
|
var acc uint32
|
|
for _, holder := range bitmap.Value(bm) {
|
|
acc |= holder
|
|
}
|
|
if acc == 0 {
|
|
t.Errorf("some pixels should be set")
|
|
}
|
|
bm.Clear()
|
|
acc = 0
|
|
for _, holder := range bitmap.Value(bm) {
|
|
acc |= holder
|
|
}
|
|
if acc != 0 {
|
|
t.Errorf("all pixels should be clear")
|
|
}
|
|
}
|
|
|
|
func TestCircle(t *testing.T) {
|
|
type testCase struct {
|
|
x, y int
|
|
r float64
|
|
filled bool
|
|
}
|
|
for i, tc := range []testCase{
|
|
{2, 2, 2, false},
|
|
{0, 2, 2, true},
|
|
{5, 5, 5, true},
|
|
{5, 0, 5, false},
|
|
{7, 7, 6.6, false},
|
|
} {
|
|
exampleFile := fmt.Sprintf("assets_test/circle_case_%02d.txt", i)
|
|
t.Run(exampleFile, func(t *testing.T) {
|
|
size := uint32(tc.r*2) + 2
|
|
bm := bitmap.NewBitmap(size, size)
|
|
bm.Circle(tc.x, tc.y, tc.r, tc.filled)
|
|
b, err := os.ReadFile(exampleFile)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
expect := strings.TrimSpace(string(b))
|
|
if expect != strings.TrimSpace(bm.String()) {
|
|
t.Errorf("expect:\n%s\ngot:\n%s\n", expect, bm)
|
|
}
|
|
})
|
|
}
|
|
}
|