Source: ol/control/Search.js

/**
 * @module webgis4u/ol/control/Search
 */

import Control from 'ol/control/Control';
import GeoJSON from 'ol/format/GeoJSON';
import VectorLayer from 'ol/layer/Vector';
import * as proj from 'ol/proj';
import VectorSource from 'ol/source/Vector';

import AutoComplete from '../../components/AutoComplete';
import { asyncDebounce } from '../../util/promise/asyncDebounce';
import { sendRequest } from '../../util/web/sendRequest';

import { getIconStyle } from '../style/style';
import { zoomToLayerExtent } from '../util/zoomToLayerExtent';

import './Search/Search.scss';


/**
 * Is called when an error occurs. Can be overriden by your own business logic.
 *
 * @callback OnErrorCallback
 * @param {type} request The request.
 * @param {type} status The status.
 * @param {type} error The error.
 * @param {ol.Layer} layer The layer.
 * @param {ol.Map} map The map.
 */

/**
 * Is called when the user hovers over a search result in the ajax serach field.
 *
 * @callback OnHoverCallback
 * @param {string} suggestion The suggestion.
 * @param {ol.layer.Vector} layer The layer on which the search results will be drawn.
 * @param {ol.Map} map The map.
 * @param {ol.format.GeoJSON} geoJsonFormat The GeoJsonFormat parser.
 */

/**
 * Is called when a search result is selected.
 *
 * @callback OnSelectCallback
 * @param {string} suggestion The sugestion.
 * @param {ol.layer.Vector} layer The layer on which the search results will be drawn.
 * @param {ol.Map} map The map.
 * @param {ol.format.GeoJSON} geoJsonFormat The GeoJsonFormat parser.
 */

/**
 * Is called when the search results are returned. Can be overridden by your specific code.
 *
 * @callback OnShowCallback
 * @param {Object.<string, ol.vector.Feature>} suggestions The suggestions as key value pair (html, feature).
 * @param {ol.layer.Vector} layer The layer on which the search results will be drawn.
 * @param {ol.Map} map The map.
 * @param {ol.format.GeoJSON} geoJsonFormat The GeoJsonFormat parser.
 */

/**
 * Return value or default value
 */
function valueOrDefault(value, defaultValue) {
  return (value === undefined || value === null)
    ? defaultValue
    : value;
}

/**
 * Convert the acquired feature data
 * @param {any} featureData The feature to read
 * @param {ol.ProjectionLike} projection Projection of the feature geometries created by the format reader.
 * @param {ol.format.GeoJSON} geoJsonFormat
 *
 * @returns {ol.Feature}
 */
function toFeature(featureData, projection, geoJsonFormat) {
  return geoJsonFormat.readFeature(
    featureData, {
      featureProjection: projection,
    },
  );
}

/**
 * @type {module:webgis4u/ol/control/Search~OnShowCallback}
 */
function defaultOnShow(suggestions, layer, map, geoJsonFormat) {
  layer.getSource().clear();
  const mapProj = map.getView().getProjection();

  // eslint-disable-next-line no-plusplus
  for (let i = 0; i < suggestions.length; i++) {
    layer.getSource().addFeature(
      toFeature(suggestions[i].data, mapProj, geoJsonFormat),
    );
  }
  zoomToLayerExtent(layer, map);
}

/**
 * @type {module:webgis4u/ol/control/Search~OnHoverCallback}
 */
function defaultOnHover(suggestion, layer, map, geoJsonFormat) {
  layer.getSource().clear();
  layer.getSource().addFeature(
    toFeature(suggestion.data, map.getView().getProjection(), geoJsonFormat),
  );
}

/**
 * @type {module:webgis4u/ol/control/Search~OnSelectCallback}
 */
function defaultOnSelect(suggestion, layer, map, geoJsonFormat) {
  const layeyrSource = layer.getSource();
  layeyrSource.clear();
  layeyrSource.addFeature(
    toFeature(suggestion.data, map.getView().getProjection(), geoJsonFormat),
  );
  zoomToLayerExtent(layer, map);
}

/**
 * @type {module:webgis4u/ol/control/Search~OnErrorCallback}
 */
function defaultOnError(request, status, error, layer) {
  layer.getSource().clear();
  console.log('error: ', request, status, error);
}

/**
 * The default name used for the search field
 */
const HTML_NAME_SEARCH_FIELD = 'webgis4uSearchField';

/**
 * Provides google like search capability for the map.
 * The service is always application specific and must therefore be provided
 * by the application that uses the webgis client.
 *
 * @example //for  search response format (= just an empty array)
 * [
 *   {
 *     "value": "display value 1",
 *     "data": {"type":"Feature","geometry":{"type":"Point","coordinates":[1187555.69,6019550.63]},"properties":{"lid":"layer0","fid":"fid0"}}
 *   },
 *   {
 *     "value": "display value 2",
 *     "data": {"type":"Feature","geometry":{"type":"Point","coordinates":[13555.69,6015650.63]},"properties":{"lid":"layer0","fid":"fid1"}}
 *   },
 *   {
 *     "value": "display value n",
 *     "data": {"type":"Feature","geometry":{"type":"Point","coordinates":[1087555.69,5419550.63]},"properties":{"lid":"layer1","fid":"fidn"}}
 *   }
 * ]
 * */
class Search extends Control {
  /**
   * The dfeault nothing found message for the search control. Can be overriden with an individual message.
   */
  static MESSAGE_NOTHING_FOUND = 'Leider nichts gefunden';

  /**
   * The dfeault nothing found message for the search control. Can be overriden with an individual message.
   */
  static MESSAGE_PENDING = 'Bitte um Geduld...';

  /**
   * The style to show all search results.
   */
  static showStyle = getIconStyle({
    src: '',
    anchorX: 0.5,
    anchorY: 0,
    opacity: 0.8,
  });

  /**
   * The style to show a single search result on hover.
   */
  static hoverStyle = getIconStyle({
    src: '',
    anchorX: 0.5,
    anchorY: 0,
    opacity: 0.5,
  });

  /**
   * The style to show a selected search result.
   */
  static selectStyle = Search.showStyle;

  /**
   * The default search URL. Can be overridden with an individual value (defaults value is "/ugisSearch").
   * @example //for setting a different search URL
   * Search.url = "/myApplicationSpecificURL";
   */
  static URL = '/ugisSearch';

  /**
   * The number of potential hits (defaults value is 19).
   * @example //for setting a different search limit
   * Search.limit = 10;
   */
  static limit = 19;

  /**
   * The number af chars after which the search starts (Avoids unecessary traffic in the server). The default value is 3.
   * @example //for setting a different minLength
   * Search.minLength = 2;
   */
  static minLength = 3;

  /**
   * The timout in ms for a search request (default = 2000).
   */
  static timeout = 2000;

  /**
   * Preprocesses the query. It gets as input parameters the query string and the map and has to output a json array
   * @example // for setting a custom search response
   * Search.preprocessQuery = (query, map) => ({query, 'extent': map.getView().calculateExtent(map.getSize())});
   */
  static preprocessQuery = query => ({ query });


  /**
   * @type {module:webgis4u/ol/control/Search~OnErrorCallback}
   * Is called when an error occurs. Can be overriden by your own business logic.
   *
   * @example
   * Search.onError = (request, status, error, layer, map) => {
   *   layer.getSource().clear();
   *   if (status == "timeout") {
   *       // timeout -> reload the page and try again
   *       console.log("timeout");
   *   } else {
   *       // another error occured
   *       console.log("error: " + request + status + err);
   *   }
   * };
   */
  static onError = defaultOnError;

  /**
   * @type {module:webgis4u/ol/control/Search~OnHoverCallback}
   */
  static onHover = defaultOnHover;

  /**
   * @type {module:webgis4u/ol/control/Search~OnSelectCallback}
   *
   * @example
   * //Can be overridden with your code
   * Search.onSelect = (suggestion, layer, map, geoJsonFormat) => {
   *    layer.getSource().clear();
   *    layer.getSource().addFeature(feature);
   *    zoomToExtent(feature.getGeometry().getExtent(), map);
   * };
   */
  static onSelect = defaultOnSelect;

  /**
 * @type {module:webgis4u/ol/control/Search~OnShowCallback}
 */
  static onShow = defaultOnShow;

  /**
   * @type {ol.Map|null}
   */
  map_ = null;

  /**
   * @type {HTMLElement}
   */
  _mapEl = null;

  _searchField = null;

  _suggestions = null;

  /**
   * Parser with the default geojson projection EPSG:4326
   * @type {ol.format.GeoJSON}
   */
  _geoJsonFormat = new GeoJSON({ defaultProjection: proj.get('EPSG:4326') });

  /**
   * @type {ol.layer.Vector}
   */
  _searchOverlay;

  /**
   * @type {ol.layer.Vector}
   */
  layerSearchResults;

  /**
   * The selector for the search field
   * @type {string}
   */
  searchFieldSelector;

  /**
   * The number of maximum shown result items
   * @type {number}
   */
  limit;

  /**
   * Constructor
   *
   * @param {object} [options]  The following properties are supported.
   * @param {int} [options.minLength] The length when a request to the datasource is triggered.
   * @param {int} [options.limit] The number of displayed hits.
   * @param {string} [options.url] The search URL.
   * @param {string} [options.preprocessQuery] The search preprocessing function.
   * @param {string} [options.searchField] The search Field as jQuery selector.
   * @param {string} [options.timeout] The timeout in ms for the search Ajax Request.
   */
  constructor(passedOptions) {
    const options = passedOptions || {};

    const element = document.createElement('div');

    super({
      element,
      target: options.target,
    });

    // Default values
    this.minLength = valueOrDefault(options.minLength, Search.minLength);
    this.limit = valueOrDefault(options.limit, Search.limit);
    this.searchURL = valueOrDefault(options.searchURL, Search.URL);
    this.timeout = valueOrDefault(options.timeout, Search.timeout);
    this.searchFieldSelector = valueOrDefault(
      options.searchField,
      `input[name="${HTML_NAME_SEARCH_FIELD}"]`,
    );

    // Default functions
    this.preprocessQuery = valueOrDefault(options.preprocessQuery, Search.preprocessQuery);
    this.onError = valueOrDefault(options.onError, Search.onError);
    this.onHover = valueOrDefault(options.onHover, Search.onHover);
    this.onShow = valueOrDefault(options.onShow, Search.onShow);
    this.onSelect = valueOrDefault(options.onSelect, Search.onSelect);

    this.initLayers();
  }

  /**
   * Gets the search overlay layer. Which is a vector layer on which the search result is visualized.
   * @returns {ol.layer.Vector}
   * @example
   * mySerachOverlayLayer = mySearchControl.getSearchOverlay();

    */
  getSearchOverlay() { return this._searchOverlay; }

  /**
   * Layer containing all search results
   * @returns {ol.layer.Vector}
   * @example
   * mySerachOverlayLayer = mySearchControl.getSearchResult();
   */
  getSearchResult() { return this.layerSearchResults; }

  /**
   * @returns {any} The suggestions
   */
  getSuggestions() { return this._suggestions; }

  /**
   * Initialize the layers
   * @private
   */
  initLayers() {
    this.layerSearchResults = new VectorLayer({
      map: this.map_,
      source: new VectorSource(),
      style: Search.selectStyle,
    });

    // Create the search result overlay
    this._searchOverlay = new VectorLayer({
      map: this.map_,
      source: new VectorSource(),
      style: Search.hoverStyle,
    });
  }

  /**
   * @inheritdoc
   * @param {ol.Map} map
   */
  setMap(map) {
    this.clearSearchResults();

    /* unbind everything an cleanup */
    if (this.map_) {
      this.map_.removeLayer(this.layerSearchResults);
      this.map_.removeLayer(this._searchOverlay);
    }

    super.setMap(map);

    this.map_ = map;
    // Nothing more to do if there is no map
    if (!map) { return; }

    // Find the new map target
    this._mapEl = map.getTargetElement();
    this._searchField = this._mapEl.parentElement.querySelector(this.searchFieldSelector);
    // Only proceed if the search field exists
    if (!this._searchField) { return; }

    this.initLayers();

    // Create a debounced version of getFiltered list
    const getSource = asyncDebounce(
      query => this.getFilterdList(this.searchURL, map, query),
      500,
    );

    this.autoComplete = new AutoComplete({
      element: this._searchField,
      minLength: this.minLength,
      messages: {
        notFound: Search.MESSAGE_NOTHING_FOUND,
        pending: Search.MESSAGE_PENDING,
      },
      source: getSource,
      getChoiceText: result => result.value,
      onListUpdated: choices => this.showSearchResults(choices),
      onItemSelected: (ev, suggestion) => {
        this._searchOverlay.getSource().clear();
        this.onSelect(suggestion, this.layerSearchResults, map, this._geoJsonFormat);
      },
      onItemHover: (ev, suggestion) => {
        this.onHover(suggestion, this._searchOverlay, map, this._geoJsonFormat);
      },
    });
  }

  /**
   * Retrieve a list of filtered items
   * @param {string} url The url
   * @param {ol.Map} map The map
   * @param {string} query The query
   * @returns {Promise} The result
   * @private
   */
  async getFilterdList(url, map, query) {
    return new Promise((resolve, reject) => {
      sendRequest({
        url,
        type: 'GET',
        data: this.preprocessQuery(query, map),
        timeout: this.timeout,
        success: (json) => {
          resolve(json);
        },
        error: (xhr, status, error) => {
          resolve([]);
          // reject();
          this.onError(xhr, status, error, this.layerSearchResults, map);
        },
      });
    });
  }

  /**
   * Clears the search results
   * @private
   */
  clearSearchResults() {
    // clear all existing search results.
    this._searchOverlay.getSource().clear();
    this.layerSearchResults.getSource().clear();
  }

  /**
   * shows the search resuls.
   * @param {Array} suggestions The found suggestions
   * @private
   */
  showSearchResults(suggestions) {
    this.clearSearchResults();

    // Get the elements from this
    const {
      layerSearchResults,
      map_,
      _geoJsonFormat,
    } = this;

    this._suggestions = suggestions;
    this.onShow(suggestions, layerSearchResults, map_, _geoJsonFormat);
  }
}

export default Search;