x = $x; $this->y = $y; $this->z = $z; } public static function clearVectors(){ Vector3::$nextVector = 0; Vector3::$vectorList = []; } public static function clearVectorList(){ if(Vector3::$nextVector > 65536){ Vector3::clearVectors(); }else{ Vector3::$nextVector = 0; } } /** * @param $x * @param $y * @param $z * * @return Vector3 */ public static function createVector($x, $y, $z){ if(Vector3::$nextVector >= count(Vector3::$vectorList)){ Vector3::$vectorList[] = new Vector3(0, 0, 0); } return Vector3::$vectorList[Vector3::$nextVector++]->setComponents($x, $y, $z); } /** * @param Vector3 $vector * * @return Vector3 */ public static function cloneVector(Vector3 $vector){ if(Vector3::$nextVector >= count(Vector3::$vectorList)){ Vector3::$vectorList[] = new Vector3(0, 0, 0); } return Vector3::$vectorList[Vector3::$nextVector++]->setComponents($vector->x, $vector->y, $vector->z); } public function getX(){ return $this->x; } public function getY(){ return $this->y; } public function getZ(){ return $this->z; } public function getFloorX(){ return (int) $this->x; } public function getFloorY(){ return (int) $this->y; } public function getFloorZ(){ return (int) $this->z; } public function getRight(){ return $this->x; } public function getUp(){ return $this->y; } public function getForward(){ return $this->z; } public function getSouth(){ return $this->x; } public function getWest(){ return $this->z; } /** * @param Vector3|int $x * @param int $y * @param int $z * * @return Vector3 */ public function add($x, $y = 0, $z = 0){ if($x instanceof Vector3){ return Vector3::createVector($this->x + $x->x, $this->y + $x->y, $this->z + $x->z); }else{ return Vector3::createVector($this->x + $x, $this->y + $y, $this->z + $z); } } /** * @param Vector3|int $x * @param int $y * @param int $z * * @return Vector3 */ public function subtract($x = 0, $y = 0, $z = 0){ if($x instanceof Vector3){ return $this->add(-$x->x, -$x->y, -$x->z); }else{ return $this->add(-$x, -$y, -$z); } } public function multiply($number){ return Vector3::createVector($this->x * $number, $this->y * $number, $this->z * $number); } public function divide($number){ return Vector3::createVector($this->x / $number, $this->y / $number, $this->z / $number); } public function ceil(){ return Vector3::createVector((int) ($this->x + 1), (int) ($this->y + 1), (int) ($this->z + 1)); } public function floor(){ $x = (int) $this->x; $y = (int) $this->y; $z = (int) $this->z; return Vector3::createVector($this->x >= $x ? $x : $x - 1, $this->y >= $y ? $y : $y - 1, $this->z >= $z ? $z : $z - 1); } public function round(){ return Vector3::createVector(round($this->x), round($this->y), round($this->z)); } public function abs(){ return Vector3::createVector(abs($this->x), abs($this->y), abs($this->z)); } public function getSide($side, $step = 1){ switch((int) $side){ case Vector3::SIDE_DOWN: return Vector3::createVector($this->x, $this->y - $step, $this->z); case Vector3::SIDE_UP: return Vector3::createVector($this->x, $this->y + $step, $this->z); case Vector3::SIDE_NORTH: return Vector3::createVector($this->x, $this->y, $this->z - $step); case Vector3::SIDE_SOUTH: return Vector3::createVector($this->x, $this->y, $this->z + $step); case Vector3::SIDE_WEST: return Vector3::createVector($this->x - $step, $this->y, $this->z); case Vector3::SIDE_EAST: return Vector3::createVector($this->x + $step, $this->y, $this->z); default: return $this; } } public static function getOppositeSide($side){ switch((int) $side){ case Vector3::SIDE_DOWN: return Vector3::SIDE_UP; case Vector3::SIDE_UP: return Vector3::SIDE_DOWN; case Vector3::SIDE_NORTH: return Vector3::SIDE_SOUTH; case Vector3::SIDE_SOUTH: return Vector3::SIDE_NORTH; case Vector3::SIDE_WEST: return Vector3::SIDE_EAST; case Vector3::SIDE_EAST: return Vector3::SIDE_WEST; default: return -1; } } public function distance(Vector3 $pos){ return sqrt($this->distanceSquared($pos)); } public function distanceSquared(Vector3 $pos){ return pow($this->x - $pos->x, 2) + pow($this->y - $pos->y, 2) + pow($this->z - $pos->z, 2); } public function maxPlainDistance($x = 0, $z = 0){ if($x instanceof Vector3){ return $this->maxPlainDistance($x->x, $x->z); }elseif($x instanceof Vector2){ return $this->maxPlainDistance($x->x, $x->y); }else{ return max(abs($this->x - $x), abs($this->z - $z)); } } public function length(){ return sqrt($this->lengthSquared()); } public function lengthSquared(){ return $this->x ** 2 + $this->y ** 2 + $this->z ** 2; } /** * @return Vector3 */ public function normalize(){ $len = $this->length(); if($len != 0){ return $this->divide($len); } return Vector3::createVector(0, 0, 0); } public function dot(Vector3 $v){ return $this->x * $v->x + $this->y * $v->y + $this->z * $v->z; } public function cross(Vector3 $v){ return Vector3::createVector( $this->y * $v->z - $this->z * $v->y, $this->z * $v->x - $this->x * $v->z, $this->x * $v->y - $this->y * $v->x ); } /** * Returns a new vector with x value equal to the second parameter, along the line between this vector and the * passed in vector, or null if not possible. * * @param Vector3 $v * @param float $x * * @return Vector3 */ public function getIntermediateWithXValue(Vector3 $v, $x){ $xDiff = $v->x - $this->x; $yDiff = $v->y - $this->y; $zDiff = $v->z - $this->z; if(($xDiff ** 2) < 1){ return null; } $f = ($x - $this->x) / $xDiff; if($f < 0 or $f > 1){ return null; }else{ return Vector3::createVector($this->x + $xDiff * $f, $this->y + $yDiff * $f, $this->z + $zDiff * $f); } } /** * Returns a new vector with y value equal to the second parameter, along the line between this vector and the * passed in vector, or null if not possible. * * @param Vector3 $v * @param float $y * * @return Vector3 */ public function getIntermediateWithYValue(Vector3 $v, $y){ $xDiff = $v->x - $this->x; $yDiff = $v->y - $this->y; $zDiff = $v->z - $this->z; if(($yDiff ** 2) < 1){ return null; } $f = ($y - $this->y) / $yDiff; if($f < 0 or $f > 1){ return null; }else{ return Vector3::createVector($this->x + $xDiff * $f, $this->y + $yDiff * $f, $this->z + $zDiff * $f); } } /** * Returns a new vector with z value equal to the second parameter, along the line between this vector and the * passed in vector, or null if not possible. * * @param Vector3 $v * @param float $z * * @return Vector3 */ public function getIntermediateWithZValue(Vector3 $v, $z){ $xDiff = $v->x - $this->x; $yDiff = $v->y - $this->y; $zDiff = $v->z - $this->z; if(($zDiff ** 2) < 1){ return null; } $f = ($z - $this->z) / $zDiff; if($f < 0 or $f > 1){ return null; }else{ return Vector3::createVector($this->x + $xDiff * $f, $this->y + $yDiff * $f, $this->z + $zDiff * $f); } } /** * @param $x * @param $y * @param $z * * @return Vector3 */ public function setComponents($x, $y, $z){ $this->x = $x; $this->y = $y; $this->z = $z; return $this; } public function __toString(){ return "Vector3(x=" . $this->x . ",y=" . $this->y . ",z=" . $this->z . ")"; } }