<?php

/**
 * PHP Shapefile - PHP library to read and write ESRI Shapefiles, compatible with WKT and GeoJSON
 *
 * @package Shapefile
 * @author  Gaspare Sganga
 * @version 3.3.0
 * @license MIT
 * @link    https://gasparesganga.com/labs/php-shapefile/
 */

namespace Shapefile\Geometry;

use Shapefile\Shapefile;
use Shapefile\ShapefileException;

/**
 * Linestring Geometry.
 *
 *  - Array: [
 *      [numpoints] => int
 *      [points]    => [
 *          [
 *              [x] => float
 *              [y] => float
 *              [z] => float
 *              [m] => float/bool
 *          ]
 *      ]
 *  ]
 *
 *  - WKT:
 *      LINESTRING [Z][M] (x y z m, x y z m)
 *
 *  - GeoJSON:
 *      {
 *          "type": "LineString" / "LineStringM"
 *          "coordinates": [
 *              [x, y, z] / [x, y, m] / [x, y, z, m]
 *          ]
 *      }
 */
class Linestring extends MultiPoint
{
    /**
     * WKT and GeoJSON basetypes, collection class type
     */
    const WKT_BASETYPE      = 'LINESTRING';
    const GEOJSON_BASETYPE  = 'LineString';
    const COLLECTION_CLASS  = 'Point';
    
    
    /////////////////////////////// PUBLIC ///////////////////////////////
    /**
     * Checks whether the linestring is a closed ring or not.
     * A closed ring has at least 4 vertices and the first and last ones must be the same.
     *
     * @return  bool
     */
    public function isClosedRing()
    {
        return $this->getNumPoints() >= 4 && $this->getPoint(0) == $this->getPoint($this->getNumPoints() - 1);
    }
    
    /**
     * Forces the linestring to be a closed ring.
     *
     * @return  self    Returns $this to provide a fluent interface.
     */
    public function forceClosedRing()
    {
        if (!$this->checkRingNumPoints()) {
            throw new ShapefileException(Shapefile::ERR_GEOM_RING_NOT_ENOUGH_VERTICES);
        }
        if (!$this->isClosedRing()) {
            $this->addPoint($this->getPoint(0));
        }
        return $this;
    }
    
    
    /**
     * Checks whether a ring is clockwise or not (it works with open rings too).
     *
     * Throws an exception if ring area is too small and cannot determine its orientation.
     * Returns Shapefile::UNDEFINED or throw an exception if there are not enough points.
     *
     * @param   bool    $flag_throw_exception   Optional flag to throw an exception if there are not enough points.
     *
     * @return  bool|Shapefile::UNDEFINED
     */
    public function isClockwise($flag_throw_exception = false)
    {
        if ($this->isEmpty()) {
            return Shapefile::UNDEFINED;
        }
        
        if (!$this->checkRingNumPoints()) {
            if ($flag_throw_exception) {
                throw new ShapefileException(Shapefile::ERR_GEOM_RING_NOT_ENOUGH_VERTICES);
            }
            return Shapefile::UNDEFINED;
        }
        
        $area = $this->computeGaussArea($this->getArray()['points']);
        if (!$area) {
            throw new ShapefileException(Shapefile::ERR_GEOM_RING_AREA_TOO_SMALL);
        }
        
        // Negative area means clockwise direction
        return $area < 0;
    }
    
    /**
     * Forces the ring to be in clockwise direction (it works with open rings too).
     * Throws an exception if direction is undefined.
     *
     * @return  self    Returns $this to provide a fluent interface.
     */
    public function forceClockwise()
    {
        if ($this->isClockwise(true) === false) {
            $this->reverseGeometries();
        }
        return $this;
    }
    
    /**
     * Forces the ring to be in counterclockwise direction (it works with open rings too).
     * Throws an exception if direction is undefined.
     *
     * @return  self    Returns $this to provide a fluent interface.
     */
    public function forceCounterClockwise()
    {
        if ($this->isClockwise(true) === true) {
            $this->reverseGeometries();
        }
        return $this;
    }
    
    
    
    public function getSHPBasetype()
    {
        return Shapefile::SHAPE_TYPE_POLYLINE;
    }
    
    
    /////////////////////////////// PROTECTED ///////////////////////////////
    protected function getWKTBasetype()
    {
        return static::WKT_BASETYPE;
    }
    
    protected function getGeoJSONBasetype()
    {
        return static::GEOJSON_BASETYPE;
    }
    
    protected function getCollectionClass()
    {
        return __NAMESPACE__ . '\\' . static::COLLECTION_CLASS;
    }
    
    
    /////////////////////////////// PRIVATE ///////////////////////////////
    /**
     * Checks if the linestring has enough points to be a ring.
     */
    private function checkRingNumPoints()
    {
        return $this->getNumPoints() >= 3;
    }
    
    
    /**
     * Computes ring area using a Gauss-like formula.
     * The target is to determine whether it is positive or negative, not the exact area.
     *
     * An optional $exp parameter is used to deal with very small areas.
     *
     * @param   array   $points     Array of points. Each element must have "x" and "y" members.
     * @param   int     $exp        Optional exponent to deal with small areas (coefficient = 10^exponent).
     *
     * @return  float
     */
    private function computeGaussArea($points, $exp = 0)
    {
        // If a coefficient of 10^9 is not enough, give up!
        if ($exp > 9) {
            return 0;
        }
        $coef = pow(10, $exp);
        
        // At least 3 points (in case of an open ring) are needed to compute the area
        $num_points = count($points);
        if ($num_points < 3) {
            return 0;
        }
        
        // Use Gauss's area formula (no need to be strict here, hence no 1/2 coefficient and no check for closed rings)
        $num_points--;
        $tot = 0;
        for ($i = 0; $i < $num_points; ++$i) {
            $tot += ($coef * $points[$i]['x'] * $points[$i + 1]['y']) - ($coef * $points[$i]['y'] * $points[$i + 1]['x']);
        }
        $tot += ($coef * $points[$num_points]['x'] * $points[0]['y']) - ($coef * $points[$num_points]['y'] * $points[0]['x']);
        
        // If area is too small, increase coefficient exponent and retry
        if ($tot == 0) {
            return $this->computeGaussArea($points, $exp + 3);
        }
        
        return $tot;
    }
}