restorable: Bug fix: partial ReplacePixels broke the image state (#562)

This commit is contained in:
Hajime Hoshi 2018-03-20 00:37:02 +09:00
parent 9d37221743
commit be4e3ab801
2 changed files with 46 additions and 29 deletions

View File

@ -130,8 +130,12 @@ func (i *Image) ReplacePixels(pixels []byte, x, y, width, height int) {
// Copy the pixels so that this works even p is modified just after ReplacePixels. // Copy the pixels so that this works even p is modified just after ReplacePixels.
if i.basePixels == nil { if i.basePixels == nil {
w, h := i.image.Size() if x == 0 && y == 0 && width == w && height == h {
i.basePixels = make([]byte, 4*w*h) i.basePixels = make([]byte, 4*w*h)
} else {
i.makeStale()
return
}
} }
idx := 4 * (y*w + x) idx := 4 * (y*w + x)
for j := 0; j < height; j++ { for j := 0; j < height; j++ {

View File

@ -101,21 +101,6 @@ func TestRestore(t *testing.T) {
} }
} }
func vertices(sw, sh int, x, y int) []float32 {
swf := float32(sw)
shf := float32(sh)
tx := float32(x)
ty := float32(y)
// For the rule of values, see vertices.go.
return []float32{
0 + tx, 0 + ty, 0, 0, 1, 1,
swf + tx, 0 + ty, 1, 0, 0, 1,
0 + tx, shf + ty, 0, 1, 1, 0,
swf + tx, shf + ty, 1, 1, 0, 0,
}
}
func TestRestoreChain(t *testing.T) { func TestRestoreChain(t *testing.T) {
const num = 10 const num = 10
imgs := []*Image{} imgs := []*Image{}
@ -336,6 +321,7 @@ func TestRestoreRecursive(t *testing.T) {
base.Pix[1] = 0xff base.Pix[1] = 0xff
base.Pix[2] = 0xff base.Pix[2] = 0xff
base.Pix[3] = 0xff base.Pix[3] = 0xff
img0 := newImageFromImage(base) img0 := newImageFromImage(base)
img1 := NewImage(4, 1, false) img1 := NewImage(4, 1, false)
fill(img1, 0, 0, 0, 0) fill(img1, 0, 0, 0, 0)
@ -386,22 +372,23 @@ func TestReplacePixels(t *testing.T) {
w = 17 w = 17
h = 31 h = 31
) )
img := NewImage(17, 31, false) img := NewImage(17, 31, false)
defer img.Dispose()
pix := make([]byte, 4*4*4) pix := make([]byte, 4*4*4)
for i := range pix { for i := range pix {
pix[i] = 0xff pix[i] = 0xff
} }
img.ReplacePixels(pix, 5, 7, 4, 4) img.ReplacePixels(pix, 5, 7, 4, 4)
for j := 0; j < h; j++ { // Check the region (5, 7)-(9, 11). Outside state is indeterministic.
for i := 0; i < w; i++ { for j := 7; j < 11; j++ {
for i := 5; i < 9; i++ {
got, err := img.At(i, j) got, err := img.At(i, j)
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
want := color.RGBA{} want := color.RGBA{0xff, 0xff, 0xff, 0xff}
if 5 <= i && i < 9 && 7 <= j && j < 11 {
want = color.RGBA{0xff, 0xff, 0xff, 0xff}
}
if got != want { if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want) t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
} }
@ -413,16 +400,13 @@ func TestReplacePixels(t *testing.T) {
if err := Restore(); err != nil { if err := Restore(); err != nil {
t.Fatal(err) t.Fatal(err)
} }
for j := 0; j < h; j++ { for j := 7; j < 11; j++ {
for i := 0; i < w; i++ { for i := 5; i < 9; i++ {
got, err := img.At(i, j) got, err := img.At(i, j)
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
want := color.RGBA{} want := color.RGBA{0xff, 0xff, 0xff, 0xff}
if 5 <= i && i < 9 && 7 <= j && j < 11 {
want = color.RGBA{0xff, 0xff, 0xff, 0xff}
}
if got != want { if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want) t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
} }
@ -430,4 +414,33 @@ func TestReplacePixels(t *testing.T) {
} }
} }
func TestDrawImageAndReplacePixels(t *testing.T) {
base := image.NewRGBA(image.Rect(0, 0, 1, 1))
base.Pix[0] = 0xff
base.Pix[1] = 0xff
base.Pix[2] = 0xff
base.Pix[3] = 0xff
img0 := newImageFromImage(base)
defer img0.Dispose()
img1 := NewImage(2, 1, false)
defer img1.Dispose()
img1.DrawImage(img0, 0, 0, 1, 1, nil, nil, opengl.CompositeModeCopy, graphics.FilterNearest)
img1.ReplacePixels([]byte{0xff, 0xff, 0xff, 0xff}, 1, 0, 1, 1)
if err := ResolveStaleImages(); err != nil {
t.Fatal(err)
}
if err := Restore(); err != nil {
t.Fatal(err)
}
got, err := img1.At(0, 0)
if err != nil {
t.Fatal(err)
}
want := color.RGBA{0xff, 0xff, 0xff, 0xff}
if !sameColors(got, want, 1) {
t.Errorf("got: %v, want: %v", got, want)
}
}
// TODO: How about volatile/screen images? // TODO: How about volatile/screen images?