2020-07-25 17:37:41 -07:00
|
|
|
package rules
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"errors"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type RoyaleRuleset struct {
|
|
|
|
|
StandardRuleset
|
|
|
|
|
|
|
|
|
|
Turn int32
|
|
|
|
|
ShrinkEveryNTurns int32
|
2020-07-27 10:59:52 -07:00
|
|
|
DamagePerTurn int32
|
2020-07-25 17:37:41 -07:00
|
|
|
|
|
|
|
|
// Output
|
|
|
|
|
OutOfBounds []Point
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (r *RoyaleRuleset) CreateNextBoardState(prevState *BoardState, moves []SnakeMove) (*BoardState, error) {
|
|
|
|
|
if r.ShrinkEveryNTurns < 1 {
|
2020-07-27 10:59:52 -07:00
|
|
|
return nil, errors.New("royale game must shrink at least every turn")
|
2020-07-25 17:37:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nextBoardState, err := r.StandardRuleset.CreateNextBoardState(prevState, moves)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-27 10:59:52 -07:00
|
|
|
// Algorithm:
|
|
|
|
|
// - Populate OOB for last turn
|
|
|
|
|
// - Apply damage to snake heads that are OOB
|
|
|
|
|
// - Re-populate OOB for this turn
|
|
|
|
|
// ---> This means damage on board shrinks doesn't hit until the following turn.
|
|
|
|
|
|
|
|
|
|
// TODO: LOG?
|
|
|
|
|
err = r.populateOutOfBounds(nextBoardState, r.Turn-1)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-25 17:37:41 -07:00
|
|
|
// TODO: LOG?
|
2020-07-27 10:59:52 -07:00
|
|
|
err = r.damageOutOfBounds(nextBoardState)
|
2020-07-25 17:37:41 -07:00
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO: LOG?
|
2020-07-27 10:59:52 -07:00
|
|
|
err = r.populateOutOfBounds(nextBoardState, r.Turn)
|
2020-07-25 17:37:41 -07:00
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nextBoardState, nil
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-27 10:59:52 -07:00
|
|
|
func (r *RoyaleRuleset) populateOutOfBounds(b *BoardState, turn int32) error {
|
2020-07-25 17:37:41 -07:00
|
|
|
r.OutOfBounds = []Point{}
|
|
|
|
|
|
|
|
|
|
if r.ShrinkEveryNTurns < 1 {
|
2020-07-27 10:59:52 -07:00
|
|
|
return errors.New("royale game must shrink at least every turn")
|
2020-07-25 17:37:41 -07:00
|
|
|
}
|
|
|
|
|
|
2020-07-27 10:59:52 -07:00
|
|
|
if turn < r.ShrinkEveryNTurns {
|
2020-07-25 17:37:41 -07:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-27 10:59:52 -07:00
|
|
|
numShrinks := turn / r.ShrinkEveryNTurns
|
2020-07-25 17:37:41 -07:00
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-27 10:59:52 -07:00
|
|
|
func (r *RoyaleRuleset) damageOutOfBounds(b *BoardState) error {
|
|
|
|
|
if r.DamagePerTurn < 1 {
|
|
|
|
|
return errors.New("royale damage per turn must be greater than zero")
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-25 17:37:41 -07:00
|
|
|
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 {
|
2020-07-27 10:59:52 -07:00
|
|
|
// Snake is now out of bounds, reduce health
|
|
|
|
|
snake.Health = snake.Health - r.DamagePerTurn
|
|
|
|
|
if snake.Health <= 0 {
|
|
|
|
|
snake.Health = 0
|
|
|
|
|
snake.EliminatedCause = EliminatedByStarvation
|
|
|
|
|
}
|
2020-07-25 17:37:41 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|