Instead of having `ol.events.Key` be a listener object or an array of listener objects, it should be less error prone to have it just be a single listener object. To avoid using too many functions with multiple return types, the `ol.events.*` functions for registering and unregistering listeners no longer accept an array of event types (and only a single key is returned when registering). To make it convenient for users to register multiple listeners at once, the `observable.on()` method accepts an array of event types. Internally in the library, we should use the less risky `ol.events.listen()`.
236 lines
6.1 KiB
JavaScript
236 lines
6.1 KiB
JavaScript
// FIXME should handle all geo-referenced data, not just vector data
|
|
|
|
goog.provide('ol.interaction.DragAndDrop');
|
|
goog.provide('ol.interaction.DragAndDropEvent');
|
|
|
|
goog.require('goog.asserts');
|
|
goog.require('goog.functions');
|
|
goog.require('ol.events');
|
|
goog.require('ol.events.Event');
|
|
goog.require('ol.events.EventType');
|
|
goog.require('ol.interaction.Interaction');
|
|
goog.require('ol.proj');
|
|
|
|
|
|
/**
|
|
* @classdesc
|
|
* Handles input of vector data by drag and drop.
|
|
*
|
|
* @constructor
|
|
* @extends {ol.interaction.Interaction}
|
|
* @fires ol.interaction.DragAndDropEvent
|
|
* @param {olx.interaction.DragAndDropOptions=} opt_options Options.
|
|
* @api stable
|
|
*/
|
|
ol.interaction.DragAndDrop = function(opt_options) {
|
|
|
|
var options = opt_options ? opt_options : {};
|
|
|
|
goog.base(this, {
|
|
handleEvent: ol.interaction.DragAndDrop.handleEvent
|
|
});
|
|
|
|
/**
|
|
* @private
|
|
* @type {Array.<function(new: ol.format.Feature)>}
|
|
*/
|
|
this.formatConstructors_ = options.formatConstructors ?
|
|
options.formatConstructors : [];
|
|
|
|
/**
|
|
* @private
|
|
* @type {ol.proj.Projection}
|
|
*/
|
|
this.projection_ = options.projection ?
|
|
ol.proj.get(options.projection) : null;
|
|
|
|
/**
|
|
* @private
|
|
* @type {Array.<ol.events.Key>}
|
|
*/
|
|
this.dropListenKeys_ = null;
|
|
|
|
};
|
|
goog.inherits(ol.interaction.DragAndDrop, ol.interaction.Interaction);
|
|
|
|
|
|
/**
|
|
* @param {Event} event Event.
|
|
* @this {ol.interaction.DragAndDrop}
|
|
* @private
|
|
*/
|
|
ol.interaction.DragAndDrop.handleDrop_ = function(event) {
|
|
var files = event.dataTransfer.files;
|
|
var i, ii, file;
|
|
for (i = 0, ii = files.length; i < ii; ++i) {
|
|
file = files.item(i);
|
|
var reader = new FileReader();
|
|
reader.addEventListener(ol.events.EventType.LOAD,
|
|
goog.partial(this.handleResult_, file).bind(this));
|
|
reader.readAsText(file);
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* @param {Event} event Event.
|
|
* @private
|
|
*/
|
|
ol.interaction.DragAndDrop.handleStop_ = function(event) {
|
|
event.stopPropagation();
|
|
event.preventDefault();
|
|
event.dataTransfer.dropEffect = 'copy';
|
|
};
|
|
|
|
|
|
/**
|
|
* @param {File} file File.
|
|
* @param {Event} event Load event.
|
|
* @private
|
|
*/
|
|
ol.interaction.DragAndDrop.prototype.handleResult_ = function(file, event) {
|
|
var result = event.target.result;
|
|
var map = this.getMap();
|
|
goog.asserts.assert(map, 'map must be set');
|
|
var projection = this.projection_;
|
|
if (!projection) {
|
|
var view = map.getView();
|
|
goog.asserts.assert(view, 'map must have view');
|
|
projection = view.getProjection();
|
|
goog.asserts.assert(projection !== undefined,
|
|
'projection should be defined');
|
|
}
|
|
var formatConstructors = this.formatConstructors_;
|
|
var features = [];
|
|
var i, ii;
|
|
for (i = 0, ii = formatConstructors.length; i < ii; ++i) {
|
|
var formatConstructor = formatConstructors[i];
|
|
var format = new formatConstructor();
|
|
var readFeatures = this.tryReadFeatures_(format, result);
|
|
if (readFeatures) {
|
|
var featureProjection = format.readProjection(result);
|
|
var transform = ol.proj.getTransform(featureProjection, projection);
|
|
var j, jj;
|
|
for (j = 0, jj = readFeatures.length; j < jj; ++j) {
|
|
var feature = readFeatures[j];
|
|
var geometry = feature.getGeometry();
|
|
if (geometry) {
|
|
geometry.applyTransform(transform);
|
|
}
|
|
features.push(feature);
|
|
}
|
|
}
|
|
}
|
|
this.dispatchEvent(
|
|
new ol.interaction.DragAndDropEvent(
|
|
ol.interaction.DragAndDropEventType.ADD_FEATURES, this, file,
|
|
features, projection));
|
|
};
|
|
|
|
|
|
/**
|
|
* Handles the {@link ol.MapBrowserEvent map browser event} unconditionally and
|
|
* neither prevents the browser default nor stops event propagation.
|
|
* @param {ol.MapBrowserEvent} mapBrowserEvent Map browser event.
|
|
* @return {boolean} `false` to stop event propagation.
|
|
* @this {ol.interaction.DragAndDrop}
|
|
* @api
|
|
*/
|
|
ol.interaction.DragAndDrop.handleEvent = goog.functions.TRUE;
|
|
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
ol.interaction.DragAndDrop.prototype.setMap = function(map) {
|
|
if (this.dropListenKeys_) {
|
|
this.dropListenKeys_.forEach(ol.events.unlistenByKey);
|
|
this.dropListenKeys_ = null;
|
|
}
|
|
goog.base(this, 'setMap', map);
|
|
if (map) {
|
|
var dropArea = map.getViewport();
|
|
this.dropListenKeys_ = [
|
|
ol.events.listen(dropArea, ol.events.EventType.DROP,
|
|
ol.interaction.DragAndDrop.handleDrop_, this),
|
|
ol.events.listen(dropArea, ol.events.EventType.DRAGENTER,
|
|
ol.interaction.DragAndDrop.handleStop_, this),
|
|
ol.events.listen(dropArea, ol.events.EventType.DRAGOVER,
|
|
ol.interaction.DragAndDrop.handleStop_, this),
|
|
ol.events.listen(dropArea, ol.events.EventType.DROP,
|
|
ol.interaction.DragAndDrop.handleStop_, this)
|
|
]
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* @param {ol.format.Feature} format Format.
|
|
* @param {string} text Text.
|
|
* @private
|
|
* @return {Array.<ol.Feature>} Features.
|
|
*/
|
|
ol.interaction.DragAndDrop.prototype.tryReadFeatures_ = function(format, text) {
|
|
try {
|
|
return format.readFeatures(text);
|
|
} catch (e) {
|
|
return null;
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* @enum {string}
|
|
*/
|
|
ol.interaction.DragAndDropEventType = {
|
|
/**
|
|
* Triggered when features are added
|
|
* @event ol.interaction.DragAndDropEvent#addfeatures
|
|
* @api stable
|
|
*/
|
|
ADD_FEATURES: 'addfeatures'
|
|
};
|
|
|
|
|
|
/**
|
|
* @classdesc
|
|
* Events emitted by {@link ol.interaction.DragAndDrop} instances are instances
|
|
* of this type.
|
|
*
|
|
* @constructor
|
|
* @extends {ol.events.Event}
|
|
* @implements {oli.interaction.DragAndDropEvent}
|
|
* @param {ol.interaction.DragAndDropEventType} type Type.
|
|
* @param {Object} target Target.
|
|
* @param {File} file File.
|
|
* @param {Array.<ol.Feature>=} opt_features Features.
|
|
* @param {ol.proj.Projection=} opt_projection Projection.
|
|
*/
|
|
ol.interaction.DragAndDropEvent = function(type, target, file, opt_features, opt_projection) {
|
|
|
|
goog.base(this, type, target);
|
|
|
|
/**
|
|
* The features parsed from dropped data.
|
|
* @type {Array.<ol.Feature>|undefined}
|
|
* @api stable
|
|
*/
|
|
this.features = opt_features;
|
|
|
|
/**
|
|
* The dropped file.
|
|
* @type {File}
|
|
* @api stable
|
|
*/
|
|
this.file = file;
|
|
|
|
/**
|
|
* The feature projection.
|
|
* @type {ol.proj.Projection|undefined}
|
|
* @api
|
|
*/
|
|
this.projection = opt_projection;
|
|
|
|
};
|
|
goog.inherits(ol.interaction.DragAndDropEvent, ol.events.Event);
|