ebiten/examples/2048/2048/board.go

152 lines
3.3 KiB
Go
Raw Normal View History

2016-07-27 21:30:10 +02:00
// Copyright 2016 The Ebiten Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package twenty48
import (
2016-07-30 19:49:28 +02:00
"errors"
2016-07-27 21:30:10 +02:00
2020-10-03 19:35:13 +02:00
"github.com/hajimehoshi/ebiten/v2"
2016-07-27 21:30:10 +02:00
)
2016-07-30 19:49:28 +02:00
var taskTerminated = errors.New("twenty48: task terminated")
type task func() error
2017-06-04 11:00:47 +02:00
// Board represents the game board.
2016-07-27 21:30:10 +02:00
type Board struct {
size int
tiles map[*Tile]struct{}
2016-07-30 19:49:28 +02:00
tasks []task
2016-07-27 21:30:10 +02:00
}
2017-06-04 11:00:47 +02:00
// NewBoard generates a new Board with giving a size.
2016-08-02 18:02:05 +02:00
func NewBoard(size int) (*Board, error) {
2016-07-27 21:30:10 +02:00
b := &Board{
size: size,
tiles: map[*Tile]struct{}{},
}
for i := 0; i < 2; i++ {
2016-08-02 18:02:05 +02:00
if err := addRandomTile(b.tiles, b.size); err != nil {
return nil, err
}
2016-07-27 21:30:10 +02:00
}
2016-08-02 18:02:05 +02:00
return b, nil
2016-07-27 21:30:10 +02:00
}
2016-07-28 05:36:31 +02:00
func (b *Board) tileAt(x, y int) *Tile {
return tileAt(b.tiles, x, y)
}
2017-06-04 11:00:47 +02:00
// Update updates the board state.
func (b *Board) Update(input *Input) error {
2016-07-30 19:49:28 +02:00
for t := range b.tiles {
if err := t.Update(); err != nil {
return err
2016-07-30 10:54:05 +02:00
}
}
2016-07-30 19:49:28 +02:00
if 0 < len(b.tasks) {
t := b.tasks[0]
2016-07-31 18:23:40 +02:00
if err := t(); err == taskTerminated {
2016-07-30 19:49:28 +02:00
b.tasks = b.tasks[1:]
} else if err != nil {
2016-07-30 17:55:29 +02:00
return err
}
2016-07-30 19:49:28 +02:00
return nil
2016-07-30 17:55:29 +02:00
}
2016-07-30 19:49:28 +02:00
if dir, ok := input.Dir(); ok {
if err := b.Move(dir); err != nil {
return err
2016-07-30 17:55:29 +02:00
}
}
2016-07-30 10:54:05 +02:00
return nil
}
2017-06-04 11:00:47 +02:00
// Move enqueues tile moving tasks.
2016-07-30 10:23:39 +02:00
func (b *Board) Move(dir Dir) error {
2016-07-31 19:42:06 +02:00
for t := range b.tiles {
t.stopAnimation()
}
2016-07-30 19:49:28 +02:00
if !MoveTiles(b.tiles, b.size, dir) {
2016-07-30 10:23:39 +02:00
return nil
}
2016-07-30 19:49:28 +02:00
b.tasks = append(b.tasks, func() error {
for t := range b.tiles {
2016-07-31 10:25:39 +02:00
if t.IsMoving() {
2016-07-30 19:49:28 +02:00
return nil
}
}
return taskTerminated
})
b.tasks = append(b.tasks, func() error {
nextTiles := map[*Tile]struct{}{}
for t := range b.tiles {
2016-07-31 10:25:39 +02:00
if t.IsMoving() {
panic("not reach")
}
2016-07-30 19:49:28 +02:00
if t.next.value != 0 {
panic("not reach")
}
if t.current.value == 0 {
continue
}
nextTiles[t] = struct{}{}
}
b.tiles = nextTiles
if err := addRandomTile(b.tiles, b.size); err != nil {
return err
}
return taskTerminated
})
2016-07-30 10:23:39 +02:00
return nil
2016-07-27 21:30:10 +02:00
}
2017-06-04 11:00:47 +02:00
// Size returns the board size.
2016-07-28 18:26:44 +02:00
func (b *Board) Size() (int, int) {
x := b.size*tileSize + (b.size+1)*tileMargin
y := x
return x, y
}
2017-06-04 11:00:47 +02:00
// Draw draws the board to the given boardImage.
func (b *Board) Draw(boardImage *ebiten.Image) {
boardImage.Fill(frameColor)
2016-07-27 21:30:10 +02:00
for j := 0; j < b.size; j++ {
for i := 0; i < b.size; i++ {
2016-07-28 18:26:44 +02:00
v := 0
op := &ebiten.DrawImageOptions{}
x := i*tileSize + (i+1)*tileMargin
y := j*tileSize + (j+1)*tileMargin
op.GeoM.Translate(float64(x), float64(y))
op.ColorM.ScaleWithColor(tileBackgroundColor(v))
boardImage.DrawImage(tileImage, op)
2016-07-30 10:54:05 +02:00
}
}
2016-07-30 19:49:28 +02:00
animatingTiles := map[*Tile]struct{}{}
nonAnimatingTiles := map[*Tile]struct{}{}
2016-07-30 10:54:05 +02:00
for t := range b.tiles {
2016-07-31 10:25:39 +02:00
if t.IsMoving() {
2016-07-30 19:49:28 +02:00
animatingTiles[t] = struct{}{}
} else {
nonAnimatingTiles[t] = struct{}{}
}
}
for t := range nonAnimatingTiles {
t.Draw(boardImage)
2016-07-30 19:49:28 +02:00
}
for t := range animatingTiles {
t.Draw(boardImage)
2016-07-27 21:30:10 +02:00
}
}