You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
367 lines
11 KiB
367 lines
11 KiB
<?php |
|
/** |
|
* Handles actions related to GIS GEOMETRYCOLLECTION objects |
|
*/ |
|
|
|
declare(strict_types=1); |
|
|
|
namespace PhpMyAdmin\Gis; |
|
|
|
use PhpMyAdmin\Image\ImageWrapper; |
|
use TCPDF; |
|
|
|
use function array_merge; |
|
use function count; |
|
use function mb_strpos; |
|
use function mb_substr; |
|
use function str_split; |
|
|
|
/** |
|
* Handles actions related to GIS GEOMETRYCOLLECTION objects |
|
*/ |
|
class GisGeometryCollection extends GisGeometry |
|
{ |
|
/** @var self */ |
|
private static $instance; |
|
|
|
/** |
|
* A private constructor; prevents direct creation of object. |
|
*/ |
|
private function __construct() |
|
{ |
|
} |
|
|
|
/** |
|
* Returns the singleton. |
|
* |
|
* @return GisGeometryCollection the singleton |
|
*/ |
|
public static function singleton() |
|
{ |
|
if (! isset(self::$instance)) { |
|
self::$instance = new GisGeometryCollection(); |
|
} |
|
|
|
return self::$instance; |
|
} |
|
|
|
/** |
|
* Scales each row. |
|
* |
|
* @param string $spatial spatial data of a row |
|
* |
|
* @return array array containing the min, max values for x and y coordinates |
|
*/ |
|
public function scaleRow($spatial) |
|
{ |
|
$min_max = []; |
|
|
|
// Trim to remove leading 'GEOMETRYCOLLECTION(' and trailing ')' |
|
$goem_col = mb_substr($spatial, 19, -1); |
|
|
|
// Split the geometry collection object to get its constituents. |
|
$sub_parts = $this->explodeGeomCol($goem_col); |
|
|
|
foreach ($sub_parts as $sub_part) { |
|
$type_pos = mb_strpos($sub_part, '('); |
|
if ($type_pos === false) { |
|
continue; |
|
} |
|
|
|
$type = mb_substr($sub_part, 0, $type_pos); |
|
|
|
$gis_obj = GisFactory::factory($type); |
|
if (! $gis_obj) { |
|
continue; |
|
} |
|
|
|
$scale_data = $gis_obj->scaleRow($sub_part); |
|
|
|
// Update minimum/maximum values for x and y coordinates. |
|
$c_maxX = (float) $scale_data['maxX']; |
|
if (! isset($min_max['maxX']) || $c_maxX > $min_max['maxX']) { |
|
$min_max['maxX'] = $c_maxX; |
|
} |
|
|
|
$c_minX = (float) $scale_data['minX']; |
|
if (! isset($min_max['minX']) || $c_minX < $min_max['minX']) { |
|
$min_max['minX'] = $c_minX; |
|
} |
|
|
|
$c_maxY = (float) $scale_data['maxY']; |
|
if (! isset($min_max['maxY']) || $c_maxY > $min_max['maxY']) { |
|
$min_max['maxY'] = $c_maxY; |
|
} |
|
|
|
$c_minY = (float) $scale_data['minY']; |
|
if (isset($min_max['minY']) && $c_minY >= $min_max['minY']) { |
|
continue; |
|
} |
|
|
|
$min_max['minY'] = $c_minY; |
|
} |
|
|
|
return $min_max; |
|
} |
|
|
|
/** |
|
* Adds to the PNG image object, the data related to a row in the GIS dataset. |
|
* |
|
* @param string $spatial GIS POLYGON object |
|
* @param string|null $label Label for the GIS POLYGON object |
|
* @param string $color Color for the GIS POLYGON object |
|
* @param array $scale_data Array containing data related to scaling |
|
*/ |
|
public function prepareRowAsPng( |
|
$spatial, |
|
?string $label, |
|
$color, |
|
array $scale_data, |
|
ImageWrapper $image |
|
): ImageWrapper { |
|
// Trim to remove leading 'GEOMETRYCOLLECTION(' and trailing ')' |
|
$goem_col = mb_substr($spatial, 19, -1); |
|
// Split the geometry collection object to get its constituents. |
|
$sub_parts = $this->explodeGeomCol($goem_col); |
|
|
|
foreach ($sub_parts as $sub_part) { |
|
$type_pos = mb_strpos($sub_part, '('); |
|
if ($type_pos === false) { |
|
continue; |
|
} |
|
|
|
$type = mb_substr($sub_part, 0, $type_pos); |
|
|
|
$gis_obj = GisFactory::factory($type); |
|
if (! $gis_obj) { |
|
continue; |
|
} |
|
|
|
$image = $gis_obj->prepareRowAsPng($sub_part, $label, $color, $scale_data, $image); |
|
} |
|
|
|
return $image; |
|
} |
|
|
|
/** |
|
* Adds to the TCPDF instance, the data related to a row in the GIS dataset. |
|
* |
|
* @param string $spatial GIS GEOMETRYCOLLECTION object |
|
* @param string|null $label label for the GIS GEOMETRYCOLLECTION object |
|
* @param string $color color for the GIS GEOMETRYCOLLECTION object |
|
* @param array $scale_data array containing data related to scaling |
|
* @param TCPDF $pdf TCPDF instance |
|
* |
|
* @return TCPDF the modified TCPDF instance |
|
*/ |
|
public function prepareRowAsPdf($spatial, ?string $label, $color, array $scale_data, $pdf) |
|
{ |
|
// Trim to remove leading 'GEOMETRYCOLLECTION(' and trailing ')' |
|
$goem_col = mb_substr($spatial, 19, -1); |
|
// Split the geometry collection object to get its constituents. |
|
$sub_parts = $this->explodeGeomCol($goem_col); |
|
|
|
foreach ($sub_parts as $sub_part) { |
|
$type_pos = mb_strpos($sub_part, '('); |
|
if ($type_pos === false) { |
|
continue; |
|
} |
|
|
|
$type = mb_substr($sub_part, 0, $type_pos); |
|
|
|
$gis_obj = GisFactory::factory($type); |
|
if (! $gis_obj) { |
|
continue; |
|
} |
|
|
|
$pdf = $gis_obj->prepareRowAsPdf($sub_part, $label, $color, $scale_data, $pdf); |
|
} |
|
|
|
return $pdf; |
|
} |
|
|
|
/** |
|
* Prepares and returns the code related to a row in the GIS dataset as SVG. |
|
* |
|
* @param string $spatial GIS GEOMETRYCOLLECTION object |
|
* @param string $label label for the GIS GEOMETRYCOLLECTION object |
|
* @param string $color color for the GIS GEOMETRYCOLLECTION object |
|
* @param array $scale_data array containing data related to scaling |
|
* |
|
* @return string the code related to a row in the GIS dataset |
|
*/ |
|
public function prepareRowAsSvg($spatial, $label, $color, array $scale_data) |
|
{ |
|
$row = ''; |
|
|
|
// Trim to remove leading 'GEOMETRYCOLLECTION(' and trailing ')' |
|
$goem_col = mb_substr($spatial, 19, -1); |
|
// Split the geometry collection object to get its constituents. |
|
$sub_parts = $this->explodeGeomCol($goem_col); |
|
|
|
foreach ($sub_parts as $sub_part) { |
|
$type_pos = mb_strpos($sub_part, '('); |
|
if ($type_pos === false) { |
|
continue; |
|
} |
|
|
|
$type = mb_substr($sub_part, 0, $type_pos); |
|
|
|
$gis_obj = GisFactory::factory($type); |
|
if (! $gis_obj) { |
|
continue; |
|
} |
|
|
|
$row .= $gis_obj->prepareRowAsSvg($sub_part, $label, $color, $scale_data); |
|
} |
|
|
|
return $row; |
|
} |
|
|
|
/** |
|
* Prepares JavaScript related to a row in the GIS dataset |
|
* to visualize it with OpenLayers. |
|
* |
|
* @param string $spatial GIS GEOMETRYCOLLECTION object |
|
* @param int $srid spatial reference ID |
|
* @param string $label label for the GIS GEOMETRYCOLLECTION object |
|
* @param array $color color for the GIS GEOMETRYCOLLECTION object |
|
* @param array $scale_data array containing data related to scaling |
|
* |
|
* @return string JavaScript related to a row in the GIS dataset |
|
*/ |
|
public function prepareRowAsOl($spatial, int $srid, $label, $color, array $scale_data) |
|
{ |
|
$row = ''; |
|
|
|
// Trim to remove leading 'GEOMETRYCOLLECTION(' and trailing ')' |
|
$goem_col = mb_substr($spatial, 19, -1); |
|
// Split the geometry collection object to get its constituents. |
|
$sub_parts = $this->explodeGeomCol($goem_col); |
|
|
|
foreach ($sub_parts as $sub_part) { |
|
$type_pos = mb_strpos($sub_part, '('); |
|
if ($type_pos === false) { |
|
continue; |
|
} |
|
|
|
$type = mb_substr($sub_part, 0, $type_pos); |
|
|
|
$gis_obj = GisFactory::factory($type); |
|
if (! $gis_obj) { |
|
continue; |
|
} |
|
|
|
$row .= $gis_obj->prepareRowAsOl($sub_part, $srid, $label, $color, $scale_data); |
|
} |
|
|
|
return $row; |
|
} |
|
|
|
/** |
|
* Splits the GEOMETRYCOLLECTION object and get its constituents. |
|
* |
|
* @param string $geom_col geometry collection string |
|
* |
|
* @return array the constituents of the geometry collection object |
|
*/ |
|
private function explodeGeomCol($geom_col) |
|
{ |
|
$sub_parts = []; |
|
$br_count = 0; |
|
$start = 0; |
|
$count = 0; |
|
foreach (str_split($geom_col) as $char) { |
|
if ($char === '(') { |
|
$br_count++; |
|
} elseif ($char === ')') { |
|
$br_count--; |
|
if ($br_count == 0) { |
|
$sub_parts[] = mb_substr($geom_col, $start, $count + 1 - $start); |
|
$start = $count + 2; |
|
} |
|
} |
|
|
|
$count++; |
|
} |
|
|
|
return $sub_parts; |
|
} |
|
|
|
/** |
|
* Generates the WKT with the set of parameters passed by the GIS editor. |
|
* |
|
* @param array $gis_data GIS data |
|
* @param int $index index into the parameter object |
|
* @param string|null $empty value for empty points |
|
* |
|
* @return string WKT with the set of parameters passed by the GIS editor |
|
*/ |
|
public function generateWkt(array $gis_data, $index, $empty = '') |
|
{ |
|
$geom_count = $gis_data['GEOMETRYCOLLECTION']['geom_count'] ?? 1; |
|
$wkt = 'GEOMETRYCOLLECTION('; |
|
for ($i = 0; $i < $geom_count; $i++) { |
|
if (! isset($gis_data[$i]['gis_type'])) { |
|
continue; |
|
} |
|
|
|
$type = $gis_data[$i]['gis_type']; |
|
$gis_obj = GisFactory::factory($type); |
|
if (! $gis_obj) { |
|
continue; |
|
} |
|
|
|
$wkt .= $gis_obj->generateWkt($gis_data, $i, $empty) . ','; |
|
} |
|
|
|
if (isset($gis_data[0]['gis_type'])) { |
|
$wkt = mb_substr($wkt, 0, -1); |
|
} |
|
|
|
return $wkt . ')'; |
|
} |
|
|
|
/** |
|
* Generates parameters for the GIS data editor from the value of the GIS column. |
|
* |
|
* @param string $value of the GIS column |
|
* |
|
* @return array parameters for the GIS editor from the value of the GIS column |
|
*/ |
|
public function generateParams($value) |
|
{ |
|
$params = []; |
|
$data = GisGeometry::generateParams($value); |
|
$params['srid'] = $data['srid']; |
|
$wkt = $data['wkt']; |
|
|
|
// Trim to remove leading 'GEOMETRYCOLLECTION(' and trailing ')' |
|
$goem_col = mb_substr($wkt, 19, -1); |
|
// Split the geometry collection object to get its constituents. |
|
$sub_parts = $this->explodeGeomCol($goem_col); |
|
$params['GEOMETRYCOLLECTION']['geom_count'] = count($sub_parts); |
|
|
|
$i = 0; |
|
foreach ($sub_parts as $sub_part) { |
|
$type_pos = mb_strpos($sub_part, '('); |
|
if ($type_pos === false) { |
|
continue; |
|
} |
|
|
|
$type = mb_substr($sub_part, 0, $type_pos); |
|
/** |
|
* @var GisMultiPolygon|GisPolygon|GisMultiPoint|GisPoint|GisMultiLineString|GisLineString $gis_obj |
|
*/ |
|
$gis_obj = GisFactory::factory($type); |
|
if (! $gis_obj) { |
|
continue; |
|
} |
|
|
|
$params = array_merge($params, $gis_obj->generateParams($sub_part, $i)); |
|
$i++; |
|
} |
|
|
|
return $params; |
|
} |
|
}
|
|
|