UNPKG

c8y-openlayer

Version:

This module is designed to help integrate Openlayer with Cumulocity IoT

381 lines (316 loc) 9.88 kB
// FIXME handle date line wrap import _ol_ from '../index.js'; import _ol_array_ from '../array.js'; import _ol_control_Control_ from '../control/control.js'; import _ol_css_ from '../css.js'; import _ol_dom_ from '../dom.js'; import _ol_events_ from '../events.js'; import _ol_events_EventType_ from '../events/eventtype.js'; import _ol_layer_Layer_ from '../layer/layer.js'; import _ol_obj_ from '../obj.js'; /** * @classdesc * Control to show all the attributions associated with the layer sources * in the map. This control is one of the default controls included in maps. * By default it will show in the bottom right portion of the map, but this can * be changed by using a css selector for `.ol-attribution`. * * @constructor * @extends {ol.control.Control} * @param {olx.control.AttributionOptions=} opt_options Attribution options. * @api */ var _ol_control_Attribution_ = function(opt_options) { var options = opt_options ? opt_options : {}; /** * @private * @type {Element} */ this.ulElement_ = document.createElement('UL'); /** * @private * @type {Element} */ this.logoLi_ = document.createElement('LI'); this.ulElement_.appendChild(this.logoLi_); this.logoLi_.style.display = 'none'; /** * @private * @type {boolean} */ this.collapsed_ = options.collapsed !== undefined ? options.collapsed : true; /** * @private * @type {boolean} */ this.collapsible_ = options.collapsible !== undefined ? options.collapsible : true; if (!this.collapsible_) { this.collapsed_ = false; } var className = options.className !== undefined ? options.className : 'ol-attribution'; var tipLabel = options.tipLabel !== undefined ? options.tipLabel : 'Attributions'; var collapseLabel = options.collapseLabel !== undefined ? options.collapseLabel : '\u00BB'; if (typeof collapseLabel === 'string') { /** * @private * @type {Node} */ this.collapseLabel_ = document.createElement('span'); this.collapseLabel_.textContent = collapseLabel; } else { this.collapseLabel_ = collapseLabel; } var label = options.label !== undefined ? options.label : 'i'; if (typeof label === 'string') { /** * @private * @type {Node} */ this.label_ = document.createElement('span'); this.label_.textContent = label; } else { this.label_ = label; } var activeLabel = (this.collapsible_ && !this.collapsed_) ? this.collapseLabel_ : this.label_; var button = document.createElement('button'); button.setAttribute('type', 'button'); button.title = tipLabel; button.appendChild(activeLabel); _ol_events_.listen(button, _ol_events_EventType_.CLICK, this.handleClick_, this); var cssClasses = className + ' ' + _ol_css_.CLASS_UNSELECTABLE + ' ' + _ol_css_.CLASS_CONTROL + (this.collapsed_ && this.collapsible_ ? ' ol-collapsed' : '') + (this.collapsible_ ? '' : ' ol-uncollapsible'); var element = document.createElement('div'); element.className = cssClasses; element.appendChild(this.ulElement_); element.appendChild(button); var render = options.render ? options.render : _ol_control_Attribution_.render; _ol_control_Control_.call(this, { element: element, render: render, target: options.target }); /** * A list of currently rendered resolutions. * @type {Array.<string>} * @private */ this.renderedAttributions_ = []; /** * @private * @type {boolean} */ this.renderedVisible_ = true; /** * @private * @type {Object.<string, Element>} */ this.logoElements_ = {}; }; _ol_.inherits(_ol_control_Attribution_, _ol_control_Control_); /** * Get a list of visible attributions. * @param {olx.FrameState} frameState Frame state. * @return {Array.<string>} Attributions. * @private */ _ol_control_Attribution_.prototype.getSourceAttributions_ = function(frameState) { /** * Used to determine if an attribution already exists. * @type {Object.<string, boolean>} */ var lookup = {}; /** * A list of visible attributions. * @type {Array.<string>} */ var visibleAttributions = []; var layerStatesArray = frameState.layerStatesArray; var resolution = frameState.viewState.resolution; for (var i = 0, ii = layerStatesArray.length; i < ii; ++i) { var layerState = layerStatesArray[i]; if (!_ol_layer_Layer_.visibleAtResolution(layerState, resolution)) { continue; } var source = layerState.layer.getSource(); if (!source) { continue; } var attributionGetter = source.getAttributions2(); if (!attributionGetter) { continue; } var attributions = attributionGetter(frameState); if (!attributions) { continue; } if (Array.isArray(attributions)) { for (var j = 0, jj = attributions.length; j < jj; ++j) { if (!(attributions[j] in lookup)) { visibleAttributions.push(attributions[j]); lookup[attributions[j]] = true; } } } else { if (!(attributions in lookup)) { visibleAttributions.push(attributions); lookup[attributions] = true; } } } return visibleAttributions; }; /** * Update the attribution element. * @param {ol.MapEvent} mapEvent Map event. * @this {ol.control.Attribution} * @api */ _ol_control_Attribution_.render = function(mapEvent) { this.updateElement_(mapEvent.frameState); }; /** * @private * @param {?olx.FrameState} frameState Frame state. */ _ol_control_Attribution_.prototype.updateElement_ = function(frameState) { if (!frameState) { if (this.renderedVisible_) { this.element.style.display = 'none'; this.renderedVisible_ = false; } return; } var attributions = this.getSourceAttributions_(frameState); if (_ol_array_.equals(attributions, this.renderedAttributions_)) { return; } // remove everything but the logo while (this.ulElement_.lastChild !== this.logoLi_) { this.ulElement_.removeChild(this.ulElement_.lastChild); } // append the attributions for (var i = 0, ii = attributions.length; i < ii; ++i) { var element = document.createElement('LI'); element.innerHTML = attributions[i]; this.ulElement_.appendChild(element); } if (attributions.length === 0 && this.renderedAttributions_.length > 0) { this.element.classList.add('ol-logo-only'); } else if (this.renderedAttributions_.length === 0 && attributions.length > 0) { this.element.classList.remove('ol-logo-only'); } var visible = attributions.length > 0 || !_ol_obj_.isEmpty(frameState.logos); if (this.renderedVisible_ != visible) { this.element.style.display = visible ? '' : 'none'; this.renderedVisible_ = visible; } this.renderedAttributions_ = attributions; this.insertLogos_(frameState); }; /** * @param {?olx.FrameState} frameState Frame state. * @private */ _ol_control_Attribution_.prototype.insertLogos_ = function(frameState) { var logo; var logos = frameState.logos; var logoElements = this.logoElements_; for (logo in logoElements) { if (!(logo in logos)) { _ol_dom_.removeNode(logoElements[logo]); delete logoElements[logo]; } } var image, logoElement, logoKey; for (logoKey in logos) { var logoValue = logos[logoKey]; if (logoValue instanceof HTMLElement) { this.logoLi_.appendChild(logoValue); logoElements[logoKey] = logoValue; } if (!(logoKey in logoElements)) { image = new Image(); image.src = logoKey; if (logoValue === '') { logoElement = image; } else { logoElement = document.createElement('a'); logoElement.href = logoValue; logoElement.appendChild(image); } this.logoLi_.appendChild(logoElement); logoElements[logoKey] = logoElement; } } this.logoLi_.style.display = !_ol_obj_.isEmpty(logos) ? '' : 'none'; }; /** * @param {Event} event The event to handle * @private */ _ol_control_Attribution_.prototype.handleClick_ = function(event) { event.preventDefault(); this.handleToggle_(); }; /** * @private */ _ol_control_Attribution_.prototype.handleToggle_ = function() { this.element.classList.toggle('ol-collapsed'); if (this.collapsed_) { _ol_dom_.replaceNode(this.collapseLabel_, this.label_); } else { _ol_dom_.replaceNode(this.label_, this.collapseLabel_); } this.collapsed_ = !this.collapsed_; }; /** * Return `true` if the attribution is collapsible, `false` otherwise. * @return {boolean} True if the widget is collapsible. * @api */ _ol_control_Attribution_.prototype.getCollapsible = function() { return this.collapsible_; }; /** * Set whether the attribution should be collapsible. * @param {boolean} collapsible True if the widget is collapsible. * @api */ _ol_control_Attribution_.prototype.setCollapsible = function(collapsible) { if (this.collapsible_ === collapsible) { return; } this.collapsible_ = collapsible; this.element.classList.toggle('ol-uncollapsible'); if (!collapsible && this.collapsed_) { this.handleToggle_(); } }; /** * Collapse or expand the attribution according to the passed parameter. Will * not do anything if the attribution isn't collapsible or if the current * collapsed state is already the one requested. * @param {boolean} collapsed True if the widget is collapsed. * @api */ _ol_control_Attribution_.prototype.setCollapsed = function(collapsed) { if (!this.collapsible_ || this.collapsed_ === collapsed) { return; } this.handleToggle_(); }; /** * Return `true` when the attribution is currently collapsed or `false` * otherwise. * @return {boolean} True if the widget is collapsed. * @api */ _ol_control_Attribution_.prototype.getCollapsed = function() { return this.collapsed_; }; export default _ol_control_Attribution_;