Source: ol/control/OverviewMap.js

/**
 * @module webgis4u/ol/control/OverviewMap
 */

import Control from 'ol/control/Control';
import LineString from 'ol/geom/LineString';
import ImageLayer from 'ol/layer/Image';
import VectorLayer from 'ol/layer/Vector';
import ImageStatic from 'ol/source/ImageStatic';
import Vector from 'ol/source/Vector';
import Stroke from 'ol/style/Stroke';
import Style from 'ol/style/Style';

import Feature from 'ol/Feature';
import Map from 'ol/Map';
import View from 'ol/View';
import * as proj from 'ol/proj';
import * as extent from 'ol/extent';


import { createElement } from '../../util/dom/createElement';
import { getMapChildElementByClassName } from '../util/getMapChildElementByClassName';
import { EPSG31287_ID } from '../proj/austria';

import { CSS_CONTROL_PREFIX } from './common';
import './OverviewMap/OverviewMap.scss';

/**
 * @typedef CreateOverviewMapOptions
 * @type {object}
 * @property {*} target The target for the map
 * @property {ol.layer.ImageLayer} imageLayer The layer with the image
 * @property {ol.source.ImageStatic} imageSource The source image used for the layer
 */

/**
 * Style used for the feature overlay
 */
const defaultStyle = new Style({
  stroke: new Stroke({
    color: '#000000',
    width: 1,
  }),
});

/**
 * The root CSS class
 */
export const CSS_CLASS = `${CSS_CONTROL_PREFIX}-overviewmap`;
/**
 * Control offering navigation via an overview map
 */
class OverviewMap extends Control {
  extent = [99E3, 279E3, 695E3, 582E3];

  projection = proj.get(EPSG31287_ID);

  /**
   * The map that shows the overview.
   * If non is present, the value is `null`.
   * @type {ol.Map|null}
   */
  overviewMap = null;

  /**
   * The previous map
   * @type {ol.Map}
   * @private
   */
  oldMap = null;

  /**
   * The target that was passed to the control
   * @type {HTMLElement|string|undefined}
   */
  controlTarget = undefined;

  /**
   * Creates a new abstract layer related control
   *
   * @param {LayerRelatedControlOptions} options
   * The options
   */
  constructor(options) {
    const imageSize = [166, 86];
    const element = createElement({
      tag: 'div',
      width: `${imageSize[0]}px`,
      height: `${imageSize[1]}px`,
    });

    const { target } = options || {};

    super({
      element,
      target,
    });

    this.controlTarget = target;
    this.imageSize = imageSize;

    // Create the image layer
    this.imageLayer = new ImageLayer({
      source: new ImageStatic({
        imageExtent: this.extent,
        size: imageSize,
        projection: this.projection,
        url: '',
        imageLoadFunction: (image) => {
          // eslint-disable-next-line no-param-reassign
          image.getImage().src = '';
        },
      }),
    });
  }

  /**
   * @returns {ol.Map|null} `null` if no map is present. Otherwise the map will be returned
   */
  getOverviewMap() {
    return this.overviewMap;
  }

  /**
   * Creates the overview map from the given options
   * @param {CreateOverviewMapOptions} options The options
   *
   * @returns {ol.Map} The overview map
   */
  createOverviewMap(options) {
    const {
      target, imageLayer,
      imageSource,
      imageSize,
    } = options;

    const imageExtent = imageSource.getImageExtent();
    const projection = imageSource.getProjection();

    const resolution = (imageExtent[2] - imageExtent[0]) / imageSize[0];
    const center = extent.getCenter(imageExtent);

    const m = new Map({
      controls: [],
      interactions: [],
      layers: [imageLayer],
      target,
      view: new View({
        projection,
        resolution,
        center,
      }),
    });

    m.on('click', this.centerMap);

    return m;
  }

  /**
   * Disposes the overview map
   */
  disposeOverviewMap() {
    if (!this.overviewMap) { return; }

    this.overviewMap.removeOverlay(this.featureOverlay);
    this.overviewMap.un('click', this.centerMap);
    this.overviewMap = null;
  }

  /**
   * @inheritdoc
   */
  setMap(map) {
    if (this.oldMap) {
      this.oldMap.getView().un('propertychange', this.handleViewPropertyChanged);
      this.element.innerText = '';
    }
    this.disposeOverviewMap();

    // Update new map
    this.oldMap = map;
    super.setMap(map);

    // Add new interactions if map is available
    if (!map) { return; }
    const view = map.getView();

    const overviewMapDiv = this.controlTarget
      || getMapChildElementByClassName({ map, className: CSS_CLASS });

    // Create the overview map
    this.overviewMap = this.createOverviewMap({
      target: this.element,
      imageLayer: this.imageLayer,
      imageSource: this.imageLayer.get('source'),
      imageSize: this.imageSize,
    });

    // Empty the 'target' and append the element
    overviewMapDiv.innerText = '';
    overviewMapDiv.appendChild(this.element);

    this.featureOverlay = new VectorLayer({
      map: this.overviewMap,
      source: new Vector({}),
      style: defaultStyle,
    });

    // Register the event handler and render initial
    view.on('propertychange', this.handleViewPropertyChanged);
    this.renderCrossHair();
  }

  /**
   * Renders the cross hair at the center from the current map
   */
  renderCrossHair() {
    // Get local variables
    const map = this.getMap();
    const view = map.getView();

    // Get extent and center
    const [left, top, right, bottom] = this.extent;
    const [centerX, centerY] = proj.transform(
      view.getCenter(),
      view.getProjection(),
      this.projection,
    );

    // Create the lines
    const lineVertical = new LineString([[centerX, top], [centerX, bottom]]);
    const lineHorizontal = new LineString([[left, centerY], [right, centerY]]);

    // Clear features and add new ones
    const foSource = this.featureOverlay.getSource();
    foSource.clear();
    foSource.addFeatures([
      new Feature({
        id: 'v',
        geometry: lineVertical,
      }),
      new Feature({
        id: 'h',
        geometry: lineHorizontal,
      }),
    ]);
  }

  /**
   * Center the original map according to the click on the overview map
   */
  centerMap = (event) => {
    const view = this.getMap().getView();
    const {
      coordinate, map,
    } = event;

    // Get the projection from the overview map view
    const overviewMapProjection = map.getView().getProjection();

    view.setCenter(proj.transform(coordinate, overviewMapProjection, view.getProjection()));
  }

  handleViewPropertyChanged = () => {
    this.renderCrossHair();
    return false;
  }
}

export default OverviewMap;