83 lines
1.7 KiB
Go
83 lines
1.7 KiB
Go
|
|
package rules
|
||
|
|
|
||
|
|
import (
|
||
|
|
"errors"
|
||
|
|
)
|
||
|
|
|
||
|
|
type RoyaleRuleset struct {
|
||
|
|
StandardRuleset
|
||
|
|
|
||
|
|
Turn int32
|
||
|
|
ShrinkEveryNTurns int32
|
||
|
|
|
||
|
|
// Output
|
||
|
|
OutOfBounds []Point
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *RoyaleRuleset) CreateNextBoardState(prevState *BoardState, moves []SnakeMove) (*BoardState, error) {
|
||
|
|
if r.ShrinkEveryNTurns < 1 {
|
||
|
|
return nil, errors.New("royale game must shrink at least every 1 turn")
|
||
|
|
}
|
||
|
|
|
||
|
|
nextBoardState, err := r.StandardRuleset.CreateNextBoardState(prevState, moves)
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
|
||
|
|
// TODO: LOG?
|
||
|
|
err = r.populateOutOfBounds(nextBoardState)
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
|
||
|
|
// TODO: LOG?
|
||
|
|
err = r.eliminateOutOfBounds(nextBoardState)
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
|
||
|
|
return nextBoardState, nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *RoyaleRuleset) populateOutOfBounds(b *BoardState) error {
|
||
|
|
r.OutOfBounds = []Point{}
|
||
|
|
|
||
|
|
if r.ShrinkEveryNTurns < 1 {
|
||
|
|
return errors.New("royale game must shrink at least every 1 turn")
|
||
|
|
}
|
||
|
|
|
||
|
|
if r.Turn < r.ShrinkEveryNTurns {
|
||
|
|
return nil
|
||
|
|
}
|
||
|
|
|
||
|
|
numShrinks := r.Turn / r.ShrinkEveryNTurns
|
||
|
|
minX, maxX := numShrinks, b.Width-1-numShrinks
|
||
|
|
minY, maxY := numShrinks, b.Height-1-numShrinks
|
||
|
|
for x := int32(0); x < b.Width; x++ {
|
||
|
|
for y := int32(0); y < b.Height; y++ {
|
||
|
|
if x < minX || x > maxX || y < minY || y > maxY {
|
||
|
|
r.OutOfBounds = append(r.OutOfBounds, Point{x, y})
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
return nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func (r *RoyaleRuleset) eliminateOutOfBounds(b *BoardState) error {
|
||
|
|
for i := 0; i < len(b.Snakes); i++ {
|
||
|
|
snake := &b.Snakes[i]
|
||
|
|
if snake.EliminatedCause == NotEliminated {
|
||
|
|
head := snake.Body[0]
|
||
|
|
for _, p := range r.OutOfBounds {
|
||
|
|
if head == p {
|
||
|
|
// Snake is now out of bounds, eliminate it
|
||
|
|
snake.EliminatedCause = EliminatedByOutOfBounds
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
return nil
|
||
|
|
}
|