/**
* 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();
}
}