mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-12 20:18:59 +01:00
examples/2048: Pop animation
This commit is contained in:
parent
c44f2819af
commit
941c24e9fd
@ -75,7 +75,7 @@ func (b *Board) Move(dir Dir) error {
|
|||||||
}
|
}
|
||||||
b.tasks = append(b.tasks, func() error {
|
b.tasks = append(b.tasks, func() error {
|
||||||
for t := range b.tiles {
|
for t := range b.tiles {
|
||||||
if t.IsAnimating() {
|
if t.IsMoving() {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -84,7 +84,7 @@ func (b *Board) Move(dir Dir) error {
|
|||||||
b.tasks = append(b.tasks, func() error {
|
b.tasks = append(b.tasks, func() error {
|
||||||
nextTiles := map[*Tile]struct{}{}
|
nextTiles := map[*Tile]struct{}{}
|
||||||
for t := range b.tiles {
|
for t := range b.tiles {
|
||||||
if t.IsAnimating() {
|
if t.IsMoving() {
|
||||||
panic("not reach")
|
panic("not reach")
|
||||||
}
|
}
|
||||||
if t.next.value != 0 {
|
if t.next.value != 0 {
|
||||||
@ -101,6 +101,14 @@ func (b *Board) Move(dir Dir) error {
|
|||||||
}
|
}
|
||||||
return taskTerminated
|
return taskTerminated
|
||||||
})
|
})
|
||||||
|
b.tasks = append(b.tasks, func() error {
|
||||||
|
for t := range b.tiles {
|
||||||
|
if t.isPopping() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return taskTerminated
|
||||||
|
})
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -131,7 +139,7 @@ func (b *Board) Draw(boardImage *ebiten.Image) error {
|
|||||||
animatingTiles := map[*Tile]struct{}{}
|
animatingTiles := map[*Tile]struct{}{}
|
||||||
nonAnimatingTiles := map[*Tile]struct{}{}
|
nonAnimatingTiles := map[*Tile]struct{}{}
|
||||||
for t := range b.tiles {
|
for t := range b.tiles {
|
||||||
if t.IsAnimating() {
|
if t.IsMoving() {
|
||||||
animatingTiles[t] = struct{}{}
|
animatingTiles[t] = struct{}{}
|
||||||
} else {
|
} else {
|
||||||
nonAnimatingTiles[t] = struct{}{}
|
nonAnimatingTiles[t] = struct{}{}
|
||||||
|
@ -34,7 +34,8 @@ type TileData struct {
|
|||||||
type Tile struct {
|
type Tile struct {
|
||||||
current TileData
|
current TileData
|
||||||
next TileData
|
next TileData
|
||||||
animationCount int
|
moveCount int
|
||||||
|
popCount int
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewTile(value int, x, y int) *Tile {
|
func NewTile(value int, x, y int) *Tile {
|
||||||
@ -63,8 +64,12 @@ func (t *Tile) NextPos() (int, int) {
|
|||||||
return t.next.x, t.next.y
|
return t.next.x, t.next.y
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *Tile) IsAnimating() bool {
|
func (t *Tile) IsMoving() bool {
|
||||||
return 0 < t.animationCount
|
return 0 < t.moveCount
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *Tile) isPopping() bool {
|
||||||
|
return 0 < t.popCount
|
||||||
}
|
}
|
||||||
|
|
||||||
func tileAt(tiles map[*Tile]struct{}, x, y int) *Tile {
|
func tileAt(tiles map[*Tile]struct{}, x, y int) *Tile {
|
||||||
@ -84,7 +89,7 @@ func tileAt(tiles map[*Tile]struct{}, x, y int) *Tile {
|
|||||||
func nextTileAt(tiles map[*Tile]struct{}, x, y int) *Tile {
|
func nextTileAt(tiles map[*Tile]struct{}, x, y int) *Tile {
|
||||||
var result *Tile
|
var result *Tile
|
||||||
for t := range tiles {
|
for t := range tiles {
|
||||||
if 0 < t.animationCount {
|
if 0 < t.moveCount {
|
||||||
if t.next.x != x || t.next.y != y || t.next.value == 0 {
|
if t.next.x != x || t.next.y != y || t.next.value == 0 {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@ -102,7 +107,8 @@ func nextTileAt(tiles map[*Tile]struct{}, x, y int) *Tile {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const (
|
const (
|
||||||
maxAnimationCount = 10
|
maxMovingCount = 5
|
||||||
|
maxPoppingCount = 6
|
||||||
)
|
)
|
||||||
|
|
||||||
func MoveTiles(tiles map[*Tile]struct{}, size int, dir Dir) bool {
|
func MoveTiles(tiles map[*Tile]struct{}, size int, dir Dir) bool {
|
||||||
@ -130,7 +136,7 @@ func MoveTiles(tiles map[*Tile]struct{}, size int, dir Dir) bool {
|
|||||||
if t.next != (TileData{}) {
|
if t.next != (TileData{}) {
|
||||||
panic("not reach")
|
panic("not reach")
|
||||||
}
|
}
|
||||||
if t.IsAnimating() {
|
if t.IsMoving() {
|
||||||
panic("not reach")
|
panic("not reach")
|
||||||
}
|
}
|
||||||
ii := i
|
ii := i
|
||||||
@ -151,7 +157,7 @@ func MoveTiles(tiles map[*Tile]struct{}, size int, dir Dir) bool {
|
|||||||
if t.current.value != tt.current.value {
|
if t.current.value != tt.current.value {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
if 0 < tt.animationCount && tt.current.value != tt.next.value {
|
if 0 < tt.moveCount && tt.current.value != tt.next.value {
|
||||||
// already merged
|
// already merged
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
@ -167,20 +173,20 @@ func MoveTiles(tiles map[*Tile]struct{}, size int, dir Dir) bool {
|
|||||||
tt.next.value = 0
|
tt.next.value = 0
|
||||||
tt.next.x = ii
|
tt.next.x = ii
|
||||||
tt.next.y = jj
|
tt.next.y = jj
|
||||||
tt.animationCount = maxAnimationCount
|
tt.moveCount = maxMovingCount
|
||||||
}
|
}
|
||||||
next.x = ii
|
next.x = ii
|
||||||
next.y = jj
|
next.y = jj
|
||||||
if t.current != next {
|
if t.current != next {
|
||||||
t.next = next
|
t.next = next
|
||||||
t.animationCount = maxAnimationCount
|
t.moveCount = maxMovingCount
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !moved {
|
if !moved {
|
||||||
for t := range tiles {
|
for t := range tiles {
|
||||||
t.next = TileData{}
|
t.next = TileData{}
|
||||||
t.animationCount = 0
|
t.moveCount = 0
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return moved
|
return moved
|
||||||
@ -189,7 +195,7 @@ func MoveTiles(tiles map[*Tile]struct{}, size int, dir Dir) bool {
|
|||||||
func addRandomTile(tiles map[*Tile]struct{}, size int) error {
|
func addRandomTile(tiles map[*Tile]struct{}, size int) error {
|
||||||
cells := make([]bool, size*size)
|
cells := make([]bool, size*size)
|
||||||
for t := range tiles {
|
for t := range tiles {
|
||||||
if t.IsAnimating() {
|
if t.IsMoving() {
|
||||||
panic("not reach")
|
panic("not reach")
|
||||||
}
|
}
|
||||||
i := t.current.x + t.current.y*size
|
i := t.current.x + t.current.y*size
|
||||||
@ -218,17 +224,19 @@ func addRandomTile(tiles map[*Tile]struct{}, size int) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (t *Tile) Update() error {
|
func (t *Tile) Update() error {
|
||||||
if t.animationCount == 0 {
|
switch {
|
||||||
if t.next != (TileData{}) {
|
case 0 < t.moveCount:
|
||||||
panic("not reach")
|
t.moveCount--
|
||||||
|
if t.moveCount == 0 {
|
||||||
|
if t.current.value != t.next.value && 0 < t.next.value {
|
||||||
|
t.popCount = maxPoppingCount
|
||||||
}
|
}
|
||||||
return nil
|
|
||||||
}
|
|
||||||
t.animationCount--
|
|
||||||
if t.animationCount == 0 {
|
|
||||||
t.current = t.next
|
t.current = t.next
|
||||||
t.next = TileData{}
|
t.next = TileData{}
|
||||||
}
|
}
|
||||||
|
case 0 < t.popCount:
|
||||||
|
t.popCount--
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -248,7 +256,11 @@ func colorToScale(clr color.Color) (float64, float64, float64, float64) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func mean(a, b int, rate float64) int {
|
func mean(a, b int, rate float64) int {
|
||||||
return int(float64(a)*rate + float64(b)*(1-rate))
|
return int(float64(a)*(1-rate) + float64(b)*rate)
|
||||||
|
}
|
||||||
|
|
||||||
|
func meanF(a, b float64, rate float64) float64 {
|
||||||
|
return a*(1-rate) + b*rate
|
||||||
}
|
}
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@ -283,11 +295,26 @@ func (t *Tile) Draw(boardImage *ebiten.Image) error {
|
|||||||
y := j*tileSize + (j+1)*tileMargin
|
y := j*tileSize + (j+1)*tileMargin
|
||||||
nx := ni*tileSize + (ni+1)*tileMargin
|
nx := ni*tileSize + (ni+1)*tileMargin
|
||||||
ny := nj*tileSize + (nj+1)*tileMargin
|
ny := nj*tileSize + (nj+1)*tileMargin
|
||||||
if 0 < t.animationCount {
|
if 0 < t.moveCount {
|
||||||
rate := float64(t.animationCount) / maxAnimationCount
|
rate := 1 - float64(t.moveCount)/maxMovingCount
|
||||||
x = mean(x, nx, rate)
|
x = mean(x, nx, rate)
|
||||||
y = mean(y, ny, rate)
|
y = mean(y, ny, rate)
|
||||||
}
|
}
|
||||||
|
if 0 < t.popCount {
|
||||||
|
const maxScale = 1.2
|
||||||
|
rate := 0.0
|
||||||
|
if maxPoppingCount*2/3 <= t.popCount {
|
||||||
|
// 0 to 1
|
||||||
|
rate = 1 - float64(t.popCount-2*maxPoppingCount/3)/float64(maxPoppingCount/3)
|
||||||
|
} else {
|
||||||
|
// 1 to 0
|
||||||
|
rate = float64(t.popCount) / float64(maxPoppingCount*2/3)
|
||||||
|
}
|
||||||
|
scale := meanF(1.0, maxScale, rate)
|
||||||
|
op.GeoM.Translate(float64(-tileSize/2), float64(-tileSize/2))
|
||||||
|
op.GeoM.Scale(scale, scale)
|
||||||
|
op.GeoM.Translate(float64(tileSize/2), float64(tileSize/2))
|
||||||
|
}
|
||||||
op.GeoM.Translate(float64(x), float64(y))
|
op.GeoM.Translate(float64(x), float64(y))
|
||||||
r, g, b, a := colorToScale(tileBackgroundColor(v))
|
r, g, b, a := colorToScale(tileBackgroundColor(v))
|
||||||
op.ColorM.Scale(r, g, b, a)
|
op.ColorM.Scale(r, g, b, a)
|
||||||
|
@ -42,7 +42,7 @@ func tilesToCells(tiles map[*Tile]struct{}, size int) ([]int, []int) {
|
|||||||
for t := range tiles {
|
for t := range tiles {
|
||||||
x, y := t.Pos()
|
x, y := t.Pos()
|
||||||
cells[x+y*size] = t.Value()
|
cells[x+y*size] = t.Value()
|
||||||
if t.IsAnimating() {
|
if t.IsMoving() {
|
||||||
if t.NextValue() == 0 {
|
if t.NextValue() == 0 {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user