You are here

GeolocationMap.php in Geolocation Field 8.3

Same filename and directory in other branches
  1. 8.2 src/Element/GeolocationMap.php

File

src/Element/GeolocationMap.php
View source
<?php

namespace Drupal\geolocation\Element;

use Drupal\Core\Render\Element;
use Drupal\Core\Render\Element\RenderElement;
use Drupal\Component\Utility\SortArray;
use Drupal\Core\Template\Attribute;
use Drupal\Core\Render\BubbleableMetadata;

/**
 * Provides a render element to display a geolocation map.
 *
 * Usage example:
 * @code
 * $form['map'] = [
 *   '#type' => 'geolocation_map',
 *   '#prefix' => $this->t('Geolocation Map Render Element'),
 *   '#description' => $this->t('Render element type "geolocation_map"'),
 *   '#maptype' => 'leaflet,
 *   '#centre' => [],
 *   '#id' => 'thisisanid',
 * ];
 * @endcode
 *
 * @FormElement("geolocation_map")
 */
class GeolocationMap extends RenderElement {

  /**
   * Map Provider.
   *
   * @var \Drupal\geolocation\MapProviderManager
   */
  protected $mapProviderManager = NULL;

  /**
   * {@inheritdoc}
   */
  public function __construct(array $configuration, $plugin_id, $plugin_definition) {
    parent::__construct($configuration, $plugin_id, $plugin_definition);
    $this->mapProviderManager = \Drupal::service('plugin.manager.geolocation.mapprovider');
  }

  /**
   * {@inheritdoc}
   */
  public function getInfo() {
    $class = get_class($this);
    return [
      '#process' => [
        [
          $class,
          'processGroup',
        ],
      ],
      '#pre_render' => [
        [
          $class,
          'preRenderGroup',
        ],
        [
          $this,
          'preRenderMap',
        ],
      ],
      '#maptype' => NULL,
      '#centre' => NULL,
      '#id' => NULL,
      '#controls' => NULL,
      '#context' => [],
    ];
  }

  /**
   * Map element.
   *
   * @param array $render_array
   *   Element.
   *
   * @return array
   *   Renderable map.
   */
  public function preRenderMap(array $render_array) {
    $render_array['#theme'] = 'geolocation_map_wrapper';
    if (empty($render_array['#cache'])) {
      $render_array['#cache'] = [];
    }
    $render_array['#cache'] = array_merge_recursive($render_array['#cache'], [
      'contexts' => [
        'languages:language_interface',
      ],
    ]);
    if (empty($render_array['#attributes'])) {
      $render_array['#attributes'] = [];
    }
    if (empty($render_array['#attributes'])) {
      $render_array['#attributes'] = [];
    }
    if (empty($render_array['#id'])) {
      $render_array['#id'] = uniqid();
    }
    if (!empty($render_array['#controls'])) {
      uasort($render_array['#controls'], [
        SortArray::class,
        'sortByWeightProperty',
      ]);
    }
    if (!empty($render_array['#layers'])) {
      uasort($render_array['#layers'], [
        SortArray::class,
        'sortByWeightProperty',
      ]);
    }
    if (!empty($render_array['#children'])) {
      uasort($render_array['#children'], [
        SortArray::class,
        'sortByWeightProperty',
      ]);
    }
    if (empty($render_array['#maptype'])) {
      if (\Drupal::moduleHandler()
        ->moduleExists('geolocation_google_maps')) {
        $render_array['#maptype'] = 'google_maps';
      }
      elseif (\Drupal::moduleHandler()
        ->moduleExists('geolocation_leaflet')) {
        $render_array['#maptype'] = 'leaflet';
      }
    }
    $map_provider = $this->mapProviderManager
      ->getMapProvider($render_array['#maptype']);
    if (empty($map_provider)) {
      return $render_array;
    }
    $map_settings = [];
    if (!empty($render_array['#settings']) && is_array($render_array['#settings'])) {
      $map_settings = $render_array['#settings'];
    }
    $map_settings = $map_provider
      ->getSettings($map_settings);
    $render_array = BubbleableMetadata::mergeAttachments([
      '#attached' => [
        'library' => [
          'geolocation/geolocation.map',
        ],
      ],
    ], $render_array);
    if (!empty($render_array['#layers'])) {
      foreach (Element::children($render_array['#layers']) as $layer) {
        $render_array['#children']['layers']['layer-' . $layer] = $render_array['#layers'][$layer];
      }
    }
    foreach (Element::children($render_array) as $child) {
      $render_array['#children'][$child] = $render_array[$child];
      unset($render_array[$child]);
    }
    $render_array['#attributes'] = new Attribute($render_array['#attributes']);
    $render_array['#attributes']
      ->addClass('geolocation-map-wrapper');
    $render_array['#attributes']
      ->setAttribute('id', $render_array['#id']);
    $render_array['#attributes']
      ->setAttribute('data-map-type', $render_array['#maptype']);
    if (!empty($render_array['#centre']['lat']) && !empty($render_array['#centre']['lng'])) {
      $render_array['#attributes']
        ->setAttribute('data-centre-lat', $render_array['#centre']['lat']);
      $render_array['#attributes']
        ->setAttribute('data-centre-lng', $render_array['#centre']['lng']);
    }
    if (!empty($render_array['#centre']['lat_north_east']) && !empty($render_array['#centre']['lng_north_east']) && !empty($render_array['#centre']['lat_south_west']) && !empty($render_array['#centre']['lng_south_west'])) {
      $render_array['#attributes']
        ->setAttribute('data-centre-lat-north-east', $render_array['#centre']['lat_north_east']);
      $render_array['#attributes']
        ->setAttribute('data-centre-lng-north-east', $render_array['#centre']['lng_north_east']);
      $render_array['#attributes']
        ->setAttribute('data-centre-lat-south-west', $render_array['#centre']['lat_south_west']);
      $render_array['#attributes']
        ->setAttribute('data-centre-lng-south-west', $render_array['#centre']['lng_south_west']);
    }
    $context = [];
    if (!empty($render_array['#context'])) {
      $context = $render_array['#context'];
    }
    return $map_provider
      ->alterRenderArray($render_array, $map_settings, $context);
  }

  /**
   * Recursively return all locations in render array.
   *
   * @param array $render_array
   *   Geolocation Map render array.
   *
   * @return array
   *   Geolocation Map Locations.
   */
  public static function getLocations(array $render_array) {
    $locations = [];
    if (!empty($render_array['#type']) && $render_array['#type'] == 'geolocation_map_location') {
      $locations[] = $render_array;
    }
    elseif (!empty($render_array['#children'])) {
      foreach ($render_array['#children'] as $child) {
        if (is_array($child)) {
          $locations = array_merge($locations, static::getLocations($child));
        }
      }
    }
    else {
      foreach (Element::children($render_array) as $key) {
        if (is_array($render_array[$key])) {
          $locations = array_merge($locations, static::getLocations($render_array[$key]));
        }
      }
    }
    return $locations;
  }

}

Classes

Namesort descending Description
GeolocationMap Provides a render element to display a geolocation map.