Utility classes

2019
/**
 * Grid cell with coordonates
 */
class GridPointer implements \JsonSerializable, \ArrayAccess
{
    public $xy, $history = false;

    /**
     * @param array $xy
     */
    public function __construct($xy = [0, 0])
    {
        $this->xy = array_map("intval", $xy);

        // Keep track of all the positions
        $this->history = new Set([]);
        $this->history[] = $this->xy;
    }

    // =============================================================================
    // > ARRAY ACCESS / INTERFACES
    // =============================================================================
    /** Impementation of ArrayAccess::offsetExists */
    public function offsetExists(mixed $offset): bool
    {
        return isset($this->xy[$offset]);
    }

    /** Impementation of ArrayAccess::offsetGet */
    public function offsetGet(mixed $offset): mixed
    {
        return $this->xy[$offset];
    }

    /** Impementation of ArrayAccess::offsetSet */
    public function offsetSet(mixed $offset, mixed $value): void
    {
        $this->xy[$offset] = $value;
    }

    /** Impementation of ArrayAccess::offsetUnset */
    public function offsetUnset(mixed $offset): void
    {
        unset($this->xy[$offset]);
    }

    /**
     * When debugging, only output the value
     *
     * @return array [x, y]
     */
    public function jsonSerialize(): mixed
    {
        return $this->xy;
    }

    /**
     * Dump the result of a model with all its fields loaded
     *
     * @return void
     */
    public function json()
    {
        \Syltaen\Log::json($this);
    }

    /**
     * Coordonates as a string
     *
     * @return string
     */
    public function __toString()
    {
        return implode(";", $this->xy);
    }

    // =============================================================================
    // > DIRECTIONS
    // =============================================================================
    /**
     * Normalize a movement direction into GridPointer
     *
     * @param string|array $direction (@see normalizeMove)
     * @return GridPointer [x, y]
     */
    public static function getMoveFromDirection($direction, $count = 1)
    {
        switch ($direction) {
            case "left":
                return [-$count, 0];
            case "right":
                return [$count, 0];
            case "up":
                return [0, -$count];
            case "down":
                return [0, $count];
            case "up-left":
                return [-$count, -$count];
            case "up-right":
                return [$count, -$count];
            case "down-left":
                return [-$count, $count];
            case "down-right":
                return [$count, $count];
            default:
                die("Direction not supported: {$direction}");
        }
    }

    /**
     * Set the XY position of the pointer
     *
     * @param array $xy [x, y]
     * @return self
     */
    public function setXY($xy)
    {
        $this->xy = [$xy[0], $xy[1]];
        if ($this->history) $this->history[] = $this->xy;
        return $this;
    }

    // =============================================================================
    // > MOVEMENTS
    // =============================================================================
    /**
     * Move the pointer
     *
     * @param array $xy The offsets to apply [x, y]
     * @return self
     */
    public function move($xy)
    {
        return $this->setXY([
            $this->xy[0] + $xy[0],
            $this->xy[1] + $xy[1],
        ]);
    }

    /**
     * Go back in the history
     *
     * @return self
     */
    public function back()
    {
        return $this->setXY($this->history->last(2));
    }

    /**
     * Move the pointer in a specific direction
     *
     * @param string $direction The direction to move (left, right, up, down)
     * @param int $count Number of steps
     * @param bool $one_by_one If true, move one step at a time
     * @return self
     */
    public function direction($direction, $count = 1)
    {
        return $this->move(static::getMoveFromDirection($direction, $count));
    }

    /**
     * Apply a serie of directions to the pointer
     *
     * @param array $directions
     * @return self
     */
    public function directions($directions)
    {
        foreach ($directions as $direction) {
            $direction = is_string($direction) ? [$direction, 1] : $direction;
            $this->direction(...$direction);
        }
        return $this;
    }

    // =============================================================================
    // > GETTERS
    // =============================================================================
    /**
     * Get the neighor of the pointer
     *
     * @param string $direction
     * @param Grid $wrap_grid Specify a grid to use as a wrap reference, if needed
     * @return array
     */
    public function getNeighbor($direction, $wrap_grid = false)
    {
        $move = static::getMoveFromDirection($direction);
        return [$this->xy[0] + $move[0], $this->xy[1] + $move[1]];
    }

    /**
     * Get the neighors of the pointer
     *
     * @return Set
     */
    public function getNeighbors($diagonals = false)
    {
        $directions = !$diagonals
            ? ["up", "right", "down", "left"]
            : ["up", "right", "down", "left", "up-right", "down-right", "down-left", "up-left"];

        if (is_string($diagonals)) {
            $directions = [
                "up"    => ["up-right", "up", "up-left"],
                "right" => ["up-right", "right", "down-right"],
                "down"  => ["down-right", "down", "down-left"],
                "left"  => ["down-left", "left", "up-left"],
            ][$diagonals];
        }

        return set($directions)->mapAssoc(fn ($i, $direction) => [$direction => $this->getNeighbor($direction)]);
    }

    /**
     * Get the vector to another points
     *
     * @param array $xy
     * @return array
     */
    public function getVector($xy)
    {
        $distance = Math::gcd($xy[0] - $this->xy[0], $xy[1] - $this->xy[1]);
        $vector = [($xy[0] - $this->xy[0]) / $distance, ($xy[1] - $this->xy[1]) / $distance];
        return [$vector, $distance];
    }

    // =============================================================================
    // > CALC
    // =============================================================================
    /**
     * Get the distance between the pointer and another point
     *
     * @param array $target [x, y]
     * @param integer $y
     * @return array [x, y]
     */
    public function getDistanceTo($target)
    {
        return [$target[0] - $this->xy[0], $target[1] - $this->xy[1]];
    }

    /**
     * Get the manhattan distance between the pointer and another point
     *
     * @param array $target [x, y]
     * @return int
     */
    public function manhattan($target)
    {
        return array_sum(array_map("abs", $this->getDistanceTo($target)));
    }

    /**
     * Get all the pointers between this point and another
     *
     * @param array $target [x, y]
     * @return Set
     */
    public function pathTo($target)
    {
        return static::getInbetween($this->xy, $target);
    }

    /**
     * Get all points between two others
     *
     * @param array|GridPointer $a
     * @param array|GridPointer $b
     * @return array
     */
    public static function getInbetween($a, $b)
    {
        $distance  = [$b[0] - $a[0], $b[1] - $a[1]];
        $inbetween = [$a];

        while ($a[0] != $b[0] || $a[1] != $b[1]) {
            $a[0] += $a[0] != $b[0] ? $distance[0] / abs($distance[0]) : 0;
            $a[1] += $a[1] != $b[1] ? $distance[1] / abs($distance[1]) : 0;
            $inbetween[] = $a;
        }

        return $inbetween;
    }

    /**
     * Get the full history with inbetween points
     *
     * @return Set
     */
    public function getFullHistory()
    {
        $path = [];

        for ($i = 0; $i < count($this->history) - 1; $i++) {
            $line = static::getInbetween($this->history[$i], $this->history[$i + 1]);
            array_pop($line);
            $path = array_merge($path, $line);
        }
        $path[] = $this->xy;

        return set($path);
    }

    /**
     * Get the number of unique points visited
     *
     * @return int
     */
    public function getVisitedCount()
    {
        return $this->history->map(fn ($xy) => implode(";", $xy))->unique()->count();
    }
}