2014-12-29 12:17:40 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @package dev
|
|
|
|
|
* @author David Molineus <david.molineus@netzmacht.de>
|
|
|
|
|
* @copyright 2014 netzmacht creative David Molineus
|
|
|
|
|
* @license LGPL 3.0
|
|
|
|
|
* @filesource
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
namespace Netzmacht\Contao\Leaflet\Mapper;
|
|
|
|
|
|
|
|
|
|
use Netzmacht\Contao\Leaflet\Event\BuildDefinitionEvent;
|
2015-01-23 13:04:04 +01:00
|
|
|
use Netzmacht\Contao\Leaflet\Event\ConvertToGeoJsonEvent;
|
2015-01-12 09:55:59 +01:00
|
|
|
use Netzmacht\Contao\Leaflet\Event\GetHashEvent;
|
2015-01-24 21:37:25 +01:00
|
|
|
use Netzmacht\Contao\Leaflet\Filter\Filter;
|
2014-12-29 12:17:40 +01:00
|
|
|
use Netzmacht\LeafletPHP\Definition;
|
2015-01-27 14:52:20 +01:00
|
|
|
use Netzmacht\LeafletPHP\Value\GeoJson\ConvertsToGeoJsonFeature;
|
|
|
|
|
use Netzmacht\LeafletPHP\Value\GeoJson\Feature;
|
|
|
|
|
use Netzmacht\LeafletPHP\Value\GeoJson\FeatureCollection;
|
|
|
|
|
use Netzmacht\LeafletPHP\Value\GeoJson\GeoJsonFeature;
|
2015-01-27 12:41:11 +01:00
|
|
|
use Netzmacht\LeafletPHP\Value\LatLngBounds;
|
2014-12-29 12:17:40 +01:00
|
|
|
use Symfony\Component\EventDispatcher\EventDispatcherInterface as EventDispatcher;
|
|
|
|
|
|
|
|
|
|
/**
|
2015-01-22 10:01:30 +01:00
|
|
|
* Class DefinitionMapper is the main mapper instance which contains all other mappers as children.
|
2014-12-29 12:17:40 +01:00
|
|
|
*
|
2015-01-22 10:01:30 +01:00
|
|
|
* @package Netzmacht\Contao\Leaflet\Mapper
|
2014-12-29 12:17:40 +01:00
|
|
|
*/
|
|
|
|
|
class DefinitionMapper
|
|
|
|
|
{
|
|
|
|
|
/**
|
2015-01-12 19:03:29 +01:00
|
|
|
* Lit of all registered mappers.
|
2014-12-29 12:17:40 +01:00
|
|
|
*
|
2015-01-12 19:03:29 +01:00
|
|
|
* @var Mapper[][]
|
2014-12-29 12:17:40 +01:00
|
|
|
*/
|
2015-01-22 10:01:30 +01:00
|
|
|
private $mappers = array();
|
2014-12-29 12:17:40 +01:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* The event dispatcher.
|
|
|
|
|
*
|
|
|
|
|
* @var EventDispatcher
|
|
|
|
|
*/
|
|
|
|
|
private $eventDispatcher;
|
|
|
|
|
|
2015-01-05 14:58:37 +01:00
|
|
|
/**
|
2015-01-12 19:03:29 +01:00
|
|
|
* Cache of mapped definitions.
|
|
|
|
|
*
|
2015-01-05 14:58:37 +01:00
|
|
|
* @var array
|
|
|
|
|
*/
|
|
|
|
|
private $mapped = array();
|
|
|
|
|
|
2014-12-29 12:17:40 +01:00
|
|
|
/**
|
|
|
|
|
* Construct.
|
|
|
|
|
*
|
|
|
|
|
* @param EventDispatcher $eventDispatcher The event dispatcher.
|
|
|
|
|
*/
|
|
|
|
|
public function __construct($eventDispatcher)
|
|
|
|
|
{
|
|
|
|
|
$this->eventDispatcher = $eventDispatcher;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2015-01-22 10:01:30 +01:00
|
|
|
* Add a mapper.
|
2014-12-29 12:17:40 +01:00
|
|
|
*
|
2015-01-22 13:34:22 +01:00
|
|
|
* @param Mapper $mapper The mapper.
|
2014-12-29 12:17:40 +01:00
|
|
|
* @param int $priority The priority. The higher priorities get called first.
|
|
|
|
|
*
|
|
|
|
|
* @return $this
|
|
|
|
|
*/
|
2015-01-22 10:01:30 +01:00
|
|
|
public function register(Mapper $mapper, $priority = 0)
|
2014-12-29 12:17:40 +01:00
|
|
|
{
|
2015-01-22 10:01:30 +01:00
|
|
|
$this->mappers[$priority][] = $mapper;
|
2014-12-29 12:17:40 +01:00
|
|
|
|
2015-01-22 10:01:30 +01:00
|
|
|
krsort($this->mappers);
|
2015-01-06 14:55:53 +01:00
|
|
|
|
2014-12-29 12:17:40 +01:00
|
|
|
return $this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Build a model.
|
|
|
|
|
*
|
2015-01-20 16:38:23 +01:00
|
|
|
* @param mixed $model The definition model.
|
2015-01-24 21:37:25 +01:00
|
|
|
* @param Filter $filter Optional request filter.
|
2015-01-20 16:38:23 +01:00
|
|
|
* @param string $elementId Optional element id. If none given the mapId or alias is used.
|
|
|
|
|
* @param Definition|null $parent Optional pass the parent object.
|
2014-12-29 12:17:40 +01:00
|
|
|
*
|
2015-01-12 19:03:29 +01:00
|
|
|
* @return Definition|null
|
|
|
|
|
*
|
|
|
|
|
* @throws \RuntimeException If model could not be mapped to a definition.
|
2014-12-29 12:17:40 +01:00
|
|
|
*/
|
2015-01-24 21:37:25 +01:00
|
|
|
public function handle($model, Filter $filter = null, $elementId = null, $parent = null)
|
2014-12-29 12:17:40 +01:00
|
|
|
{
|
2015-01-22 10:01:30 +01:00
|
|
|
$hash = $this->hash($model, $elementId);
|
2015-01-05 14:58:37 +01:00
|
|
|
|
2015-01-22 10:01:30 +01:00
|
|
|
if (!isset($this->mapped[$hash])) {
|
|
|
|
|
$mapper = $this->getMapper($model);
|
2015-01-24 21:37:25 +01:00
|
|
|
$definition = $mapper->handle($model, $this, $filter, $elementId, $parent);
|
2015-01-05 14:58:37 +01:00
|
|
|
|
2015-01-22 10:01:30 +01:00
|
|
|
if ($definition) {
|
2015-01-24 21:37:25 +01:00
|
|
|
$event = new BuildDefinitionEvent($definition, $model, $filter);
|
2015-01-22 10:01:30 +01:00
|
|
|
$this->eventDispatcher->dispatch($event::NAME, $event);
|
2014-12-29 12:17:40 +01:00
|
|
|
}
|
2015-01-22 10:01:30 +01:00
|
|
|
|
|
|
|
|
$this->mapped[$hash] = $definition;
|
2014-12-29 12:17:40 +01:00
|
|
|
}
|
|
|
|
|
|
2015-01-22 10:01:30 +01:00
|
|
|
return $this->mapped[$hash];
|
2014-12-29 12:17:40 +01:00
|
|
|
}
|
2015-01-06 14:55:53 +01:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Build a model.
|
|
|
|
|
*
|
2015-01-24 21:37:25 +01:00
|
|
|
* @param mixed $model The definition model.
|
|
|
|
|
* @param Filter $filter Optional request filter.
|
2015-01-12 19:03:29 +01:00
|
|
|
*
|
|
|
|
|
* @return FeatureCollection|Feature|null
|
2015-01-06 14:55:53 +01:00
|
|
|
*
|
2015-01-12 19:03:29 +01:00
|
|
|
* @throws \RuntimeException If a model could not be mapped to the GeoJSON representation.
|
2015-01-06 14:55:53 +01:00
|
|
|
*/
|
2015-01-24 21:37:25 +01:00
|
|
|
public function handleGeoJson($model, Filter $filter = null)
|
2015-01-06 14:55:53 +01:00
|
|
|
{
|
2015-01-22 10:01:30 +01:00
|
|
|
$mapper = $this->getMapper($model);
|
2015-01-06 14:55:53 +01:00
|
|
|
|
2015-01-22 10:01:30 +01:00
|
|
|
if ($mapper instanceof GeoJsonMapper) {
|
2015-01-24 21:37:25 +01:00
|
|
|
return $mapper->handleGeoJson($model, $this, $filter);
|
2015-01-06 14:55:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw new \RuntimeException(
|
|
|
|
|
sprintf(
|
2015-01-22 10:01:30 +01:00
|
|
|
'Mapper for model "%s::%s" is not a GeoJsonMapper',
|
2015-01-06 14:55:53 +01:00
|
|
|
$model->getTable(),
|
|
|
|
|
$model->{$model->getPk()}
|
|
|
|
|
)
|
|
|
|
|
);
|
|
|
|
|
}
|
2015-01-12 09:55:59 +01:00
|
|
|
|
2015-01-23 13:04:04 +01:00
|
|
|
/**
|
|
|
|
|
* Convert a definition to a geo json feature.
|
|
|
|
|
*
|
|
|
|
|
* @param Definition $definition The leaflet definition object.
|
|
|
|
|
* @param mixed $model The corresponding definition model.
|
|
|
|
|
*
|
|
|
|
|
* @return GeoJsonFeature
|
2015-01-23 14:11:43 +01:00
|
|
|
* @throws \RuntimeException If a definition type is not supported.
|
2015-01-23 13:04:04 +01:00
|
|
|
*/
|
|
|
|
|
public function convertToGeoJsonFeature(Definition $definition, $model)
|
|
|
|
|
{
|
|
|
|
|
if ($definition instanceof GeoJsonFeature) {
|
|
|
|
|
$feature = $definition;
|
|
|
|
|
} elseif ($definition instanceof ConvertsToGeoJsonFeature) {
|
|
|
|
|
$feature = $definition->toGeoJsonFeature();
|
|
|
|
|
} else {
|
2015-01-23 14:11:43 +01:00
|
|
|
throw new \RuntimeException(
|
|
|
|
|
sprintf(
|
|
|
|
|
'Definition of class "%s" could not be converted to a geo json feature.',
|
|
|
|
|
get_class($definition)
|
|
|
|
|
)
|
|
|
|
|
);
|
2015-01-23 13:04:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$event = new ConvertToGeoJsonEvent($definition, $feature, $model);
|
|
|
|
|
$this->eventDispatcher->dispatch($event::NAME, $event);
|
|
|
|
|
|
|
|
|
|
return $feature;
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-12 09:55:59 +01:00
|
|
|
/**
|
2015-01-12 19:03:29 +01:00
|
|
|
* Get the hash of a model.
|
|
|
|
|
*
|
|
|
|
|
* @param mixed $model The definition model.
|
|
|
|
|
* @param string|null $elementId Optional defined extra element id.
|
2015-01-12 09:55:59 +01:00
|
|
|
*
|
|
|
|
|
* @return string
|
2015-01-12 19:03:29 +01:00
|
|
|
*
|
|
|
|
|
* @throws \RuntimeException If no hash was created.
|
2015-01-12 09:55:59 +01:00
|
|
|
*/
|
2015-01-22 10:01:30 +01:00
|
|
|
private function hash($model, $elementId = null)
|
2015-01-12 09:55:59 +01:00
|
|
|
{
|
|
|
|
|
$event = new GetHashEvent($model);
|
|
|
|
|
$this->eventDispatcher->dispatch($event::NAME, $event);
|
|
|
|
|
$hash = $event->getHash();
|
|
|
|
|
|
|
|
|
|
if (!$hash) {
|
|
|
|
|
throw new \RuntimeException('Could not create a hash');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ($elementId) {
|
|
|
|
|
$hash .= '.' . $elementId;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $hash;
|
|
|
|
|
}
|
2015-01-22 10:01:30 +01:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Get the mapper for a definition model.
|
|
|
|
|
*
|
|
|
|
|
* @param mixed $model The data model.
|
|
|
|
|
*
|
|
|
|
|
* @return Mapper
|
2015-01-22 13:34:22 +01:00
|
|
|
* @throws \RuntimeException If the mapper could not be found.
|
2015-01-22 10:01:30 +01:00
|
|
|
*/
|
|
|
|
|
private function getMapper($model)
|
|
|
|
|
{
|
|
|
|
|
foreach ($this->mappers as $mappers) {
|
|
|
|
|
foreach ($mappers as $mapper) {
|
|
|
|
|
if ($mapper->match($model)) {
|
|
|
|
|
return $mapper;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw new \RuntimeException(
|
|
|
|
|
sprintf(
|
2015-01-22 13:34:22 +01:00
|
|
|
'Could not build model "%s". No matching mappers found.',
|
2015-01-22 10:01:30 +01:00
|
|
|
$this->hash($model)
|
|
|
|
|
)
|
|
|
|
|
);
|
|
|
|
|
}
|
2014-12-29 12:17:40 +01:00
|
|
|
}
|