Source: control/modify.js

import throttle from "lodash.throttle";
import { singleClick } from "ol/events/condition";
import { Interaction, Modify } from "ol/interaction";
import { unByKey } from "ol/Observable";

import image from "../../img/modify_geometry2.svg";
import Delete from "../interaction/delete";
import Move from "../interaction/move";
import SelectModify from "../interaction/selectmodify";
import SelectMove from "../interaction/selectmove";
import Control from "./control";

/**
 * Control for modifying geometries.
 * @extends {Control}
 * @alias ole.ModifyControl
 */
class ModifyControl extends Control {
  /**
   * @param {Object} [options] Tool options.
   * @param {number} [options.hitTolerance=5] Select tolerance in pixels.
   * @param {ol.Collection<ol.Feature>} [options.features] Destination for drawing.
   * @param {ol.source.Vector} [options.source] Destination for drawing.
   * @param {Object} [options.selectMoveOptions] Options for the select interaction used to move features.
   * @param {Object} [options.selectModifyOptions] Options for the select interaction used to modify features.
   * @param {Object} [options.moveInteractionOptions] Options for the move interaction.
   * @param {Object} [options.modifyInteractionOptions] Options for the modify interaction.
   * @param {Object} [options.deleteInteractionOptions] Options for the delete interaction.
   * @param {Object} [options.deselectInteractionOptions] Options for the deselect interaction. Default: features are deselected on click on map.
   * @param {Function} [options.cursorStyleHandler] Options to override default cursor styling behavior.
   */
  constructor(options = {}) {
    super({
      className: "ole-control-modify",
      image,
      title: "Modify geometry",
      ...options,
    });

    /**
     * Buffer around the coordintate clicked in pixels.
     * @type {number}
     * @private
     */
    this.hitTolerance =
      options.hitTolerance === undefined ? 5 : options.hitTolerance;

    /**
     * Filter function to determine which features are elligible for selection.
     * By default we exclude features on unmanaged layers(for ex: nodes to delete).
     * @type {function(ol.Feature, ol.layer.Layer)}
     * @private
     */
    this.selectFilter =
      options.selectFilter ||
      ((feature, layer) => {
        if (layer && this.layerFilter) {
          return this.layerFilter(layer);
        }
        return !!layer;
      });

    /**
     *
     * Return features elligible for selection on specific pixel.
     * @type {function(ol.events.MapBrowserEvent)}
     * @private
     */
    this.getFeatureAtPixel = this.getFeatureAtPixel.bind(this);

    /* Cursor management */
    this.previousCursor = null;
    this.cursorTimeout = null;
    this.cursorHandlerThrottled = throttle(this.cursorHandler.bind(this), 150, {
      leading: true,
    });
    this.cursorStyleHandler =
      options?.cursorStyleHandler ||
      ((cursorStyle) => {
        return cursorStyle;
      });

    /* Interactions */
    this.createSelectMoveInteraction(options.selectMoveOptions);
    this.createSelectModifyInteraction(options.selectModifyOptions);
    this.createModifyInteraction(options.modifyInteractionOptions);
    this.createMoveInteraction(options.moveInteractionOptions);
    this.createDeleteInteraction(options.deleteInteractionOptions);
    this.createDeselectInteraction(options.deselectInteractionOptions);
  }

  /**
   * @inheritdoc
   */
  activate() {
    super.activate();
    this.deselectInteraction.setActive(true);
    this.deleteInteraction.setActive(true);
    this.selectModify.setActive(true);
    // For the default behavior it's very important to add selectMove after selectModify.
    // It will avoid single/dbleclick mess.
    this.selectMove.setActive(true);
    this.addListeners();
  }

  /**
   * Add others listeners on the map than interactions.
   * @param {*} evt
   * @private
   */
  addListeners() {
    this.removeListeners();
    this.cursorListenerKeys = [
      this.map?.on("pointerdown", (evt) => {
        const element = evt.map.getViewport();
        if (element?.style?.cursor === "grab") {
          this.changeCursor("grabbing");
        }
      }),
      this.map?.on("pointermove", this.cursorHandlerThrottled),
      this.map?.on("pointerup", (evt) => {
        const element = evt.map.getViewport();
        if (element?.style?.cursor === "grabbing") {
          this.changeCursor("grab");
        }
      }),
    ];
  }

  /**
   * Change cursor style.
   * @param {string} cursor New cursor name.
   * @private
   */
  changeCursor(cursor) {
    if (!this.getActive()) {
      return;
    }
    const newCursor = this.cursorStyleHandler(cursor);
    const element = this.map.getViewport();
    if (
      (element.style.cursor || newCursor) &&
      element.style.cursor !== newCursor
    ) {
      if (this.previousCursor === null) {
        this.previousCursor = element.style.cursor;
      }
      element.style.cursor = newCursor;
    }
  }

  /**
   * Create the interaction used to delete selected features.
   * @param {*} options
   * @private
   */
  createDeleteInteraction(options = {}) {
    /**
     * @type {ol.interaction.Delete}
     * @private
     */
    this.deleteInteraction = new Delete({ source: this.source, ...options });

    this.deleteInteraction.on("delete", () => {
      this.changeCursor(null);
    });
    this.deleteInteraction.setActive(false);
  }

  /**
   * Create the interaction used to deselected features when we click on the map.
   * @param {*} options
   * @private
   */
  createDeselectInteraction(options = {}) {
    // it's important that this condition was the same as the selectModify's
    // deleteCondition to avoid the selection of the feature under the node to delete.
    const condition = options.condition || singleClick;

    /**
     * @type {ol.interaction.Interaction}
     * @private
     */
    this.deselectInteraction = new Interaction({
      handleEvent: (mapBrowserEvent) => {
        if (!condition(mapBrowserEvent)) {
          return true;
        }
        const onFeature = this.getFeatureAtPixel(mapBrowserEvent.pixel);
        const onVertex = this.isHoverVertexFeatureAtPixel(
          mapBrowserEvent.pixel,
        );

        if (!onVertex && !onFeature) {
          // Default: Clear selection on click outside features.
          this.selectMove.getFeatures().clear();
          this.selectModify.getFeatures().clear();
          return false;
        }
        return true;
      },
    });
    this.deselectInteraction.setActive(false);
  }

  /**
   * Create the interaction used to modify vertexes of features.
   * @param {*} options
   * @private
   */
  createModifyInteraction(options = {}) {
    /**
     * @type {ol.interaction.Modify}
     * @private
     */
    this.modifyInteraction = new Modify({
      deleteCondition: singleClick,
      features: this.selectModify.getFeatures(),
      ...options,
    });

    this.modifyInteraction.on("modifystart", (evt) => {
      this.editor.setEditFeature(evt.features.item(0));
      this.isModifying = true;
    });

    this.modifyInteraction.on("modifyend", () => {
      this.editor.setEditFeature();
      this.isModifying = false;
    });
    this.modifyInteraction.setActive(false);
  }

  /**
   * Create the interaction used to move feature.
   * @param {*} options
   * @private
   */
  createMoveInteraction(options = {}) {
    /**
     * @type {ole.interaction.Move}
     * @private
     */
    this.moveInteraction = new Move({
      features: this.selectMove.getFeatures(),
      ...options,
    });

    this.moveInteraction.on("movestart", (evt) => {
      this.editor.setEditFeature(evt.feature);
      this.isMoving = true;
    });

    this.moveInteraction.on("moveend", () => {
      this.editor.setEditFeature();
      this.isMoving = false;
    });
    this.moveInteraction.setActive(false);
  }

  /**
   * Create the interaction used to select feature to modify.
   * @param {*} options
   * @private
   */
  createSelectModifyInteraction(options = {}) {
    /**
     * Select interaction to modify features.
     * @type {ol.interaction.Select}
     */
    this.selectModify = new SelectModify({
      filter: this.selectFilter,
      hitTolerance: this.hitTolerance,
      ...options,
    });

    this.selectModify.getFeatures().on("add", () => {
      this.selectMove.getFeatures().clear();
      this.modifyInteraction.setActive(true);
      this.deleteInteraction.setFeatures(this.selectModify.getFeatures());
    });

    this.selectModify.getFeatures().on("remove", () => {
      // Deactive interaction when the select array is empty
      if (this.selectModify.getFeatures().getLength() === 0) {
        this.modifyInteraction.setActive(false);
        this.deleteInteraction.setFeatures();
      }
    });
    this.selectModify.setActive(false);
  }

  /**
   * Create the interaction used to select feature to move.
   * @param {*} options
   * @private
   */
  createSelectMoveInteraction(options = {}) {
    /**
     * Select interaction to move features.
     * @type {ol.interaction.Select}
     * @private
     */
    this.selectMove = new SelectMove({
      filter: (feature, layer) => {
        // If the feature is already selected by modify interaction ignore the selection.
        if (this.isSelectedByModify(feature)) {
          return false;
        }
        return this.selectFilter(feature, layer);
      },
      hitTolerance: this.hitTolerance,
      ...options,
    });

    this.selectMove.getFeatures().on("add", () => {
      this.selectModify.getFeatures().clear();
      this.moveInteraction.setActive(true);
      this.deleteInteraction.setFeatures(this.selectMove.getFeatures());
    });

    this.selectMove.getFeatures().on("remove", () => {
      // Deactive interaction when the select array is empty
      if (this.selectMove.getFeatures().getLength() === 0) {
        this.moveInteraction.setActive(false);
        this.deleteInteraction.setFeatures();
      }
    });
    this.selectMove.setActive(false);
  }

  /**
   * Handle the move event of the move interaction.
   * @param {ol.MapBrowserEvent} evt Event.
   * @private
   */
  cursorHandler(evt) {
    if (evt.dragging || this.isMoving || this.isModifying) {
      this.changeCursor("grabbing");
      return;
    }

    const feature = this.getFeatureAtPixel(evt.pixel);
    if (!feature) {
      this.changeCursor(this.previousCursor);
      this.previousCursor = null;
      return;
    }

    if (this.isSelectedByMove(feature)) {
      this.changeCursor("grab");
    } else if (this.isSelectedByModify(feature)) {
      if (this.isHoverVertexFeatureAtPixel(evt.pixel)) {
        this.changeCursor("grab");
      } else {
        this.changeCursor(this.previousCursor);
      }
    } else {
      // Feature available for selection.
      this.changeCursor("pointer");
    }
  }

  /**
   * @inheritdoc
   */
  deactivate(silent) {
    this.removeListeners();
    this.selectMove.getFeatures().clear();
    this.selectModify.getFeatures().clear();
    this.deselectInteraction.setActive(false);
    this.deleteInteraction.setActive(false);
    this.selectModify.setActive(false);
    this.selectMove.setActive(false);
    super.deactivate(silent);
  }

  /**
   * Get a selectable feature at a pixel.
   * @param {*} pixel
   */
  getFeatureAtPixel(pixel) {
    const feature = this.map.forEachFeatureAtPixel(
      pixel,
      (feat, layer) => {
        if (this.selectFilter(feat, layer)) {
          return feat;
        }
        return null;
      },
      {
        hitTolerance: this.hitTolerance,
        layerFilter: this.layerFilter,
      },
    );
    return feature;
  }

  /**
   * Detect if a vertex is hovered.
   * @param {*} pixel
   */
  isHoverVertexFeatureAtPixel(pixel) {
    let isHoverVertex = false;
    this.map.forEachFeatureAtPixel(
      pixel,
      (feat, layer) => {
        if (!layer) {
          isHoverVertex = true;
          return true;
        }
        return false;
      },
      {
        hitTolerance: this.hitTolerance,
      },
    );
    return isHoverVertex;
  }

  isSelectedByModify(feature) {
    return this.selectModify.getFeatures().getArray().indexOf(feature) !== -1;
  }

  isSelectedByMove(feature) {
    return this.selectMove.getFeatures().getArray().indexOf(feature) !== -1;
  }

  /**
   * Remove others listeners on the map than interactions.
   * @param {*} evt
   * @private
   */
  removeListeners() {
    unByKey(this.cursorListenerKeys);
  }

  setMap(map) {
    if (this.map) {
      this.map.removeInteraction(this.modifyInteraction);
      this.map.removeInteraction(this.moveInteraction);
      this.map.removeInteraction(this.selectMove);
      this.map.removeInteraction(this.selectModify);
      this.map.removeInteraction(this.deleteInteraction);
      this.map.removeInteraction(this.deselectInteraction);
      this.removeListeners();
    }
    super.setMap(map);
    if (this.getActive()) {
      this.addListeners();
    }
    this.map?.addInteraction(this.deselectInteraction);
    this.map?.addInteraction(this.deleteInteraction);
    this.map?.addInteraction(this.selectModify);
    // For the default behvior it's very important to add selectMove after selectModify.
    // It will avoid single/dbleclick mess.
    this.map?.addInteraction(this.selectMove);
    this.map?.addInteraction(this.moveInteraction);
    this.map?.addInteraction(this.modifyInteraction);
  }
}

export default ModifyControl;