UNPKG

terriajs

Version:

Geospatial data visualization platform.

638 lines (557 loc) 22.8 kB
'use strict'; /*global require*/ var defaultValue = require('terriajs-cesium/Source/Core/defaultValue'); var defined = require('terriajs-cesium/Source/Core/defined'); var defineProperties = require('terriajs-cesium/Source/Core/defineProperties'); var DeveloperError = require('terriajs-cesium/Source/Core/DeveloperError'); var freezeObject = require('terriajs-cesium/Source/Core/freezeObject'); var when = require('terriajs-cesium/Source/ThirdParty/when'); var knockout = require('terriajs-cesium/Source/ThirdParty/knockout'); var serializeToJson = require('../Core/serializeToJson'); var updateFromJson = require('../Core/updateFromJson'); var runLater = require('../Core/runLater'); var arraysAreEqual = require('../Core/arraysAreEqual'); /** * A member of a {@link CatalogGroup}. A member may be a {@link CatalogItem} or a * {@link CatalogGroup}. * * @alias CatalogMember * @constructor * @abstract * * @param {Terria} terria The Terria instance. */ var CatalogMember = function(terria) { if (!defined(terria)) { throw new DeveloperError('terria is required'); } this._terria = terria; /** * Gets or sets the name of the item. This property is observable. * @type {String} */ this.name = 'Unnamed Item'; /** * Gets or sets the description of the item. This property is observable. * @type {String} */ this.description = ''; /** * Gets or sets the array of section titles and contents for display in the layer info panel. * In future this may replace 'description' above - this list should not contain * sections named 'description' or 'Description' if the 'description' property * is also set as both will be displayed. * The object is of the form {name:string, content:string}. * Content will be rendered as Markdown with HTML. * This property is observable. * @type {Object[]} * @default [] */ this.info = []; /** * Gets or sets the array of section titles definining the display order of info sections. If this property * is not defined, {@link DataPreviewSections}'s DEFAULT_SECTION_ORDER is used. This property is observable. * @type {String[]} */ this.infoSectionOrder = undefined; /** * Gets or sets a value indicating whether this member was supplied by the user rather than loaded from one of the * {@link Terria#initSources}. User-supplied members must be serialized completely when, for example, * serializing enabled members for sharing. This property is observable. * @type {Boolean} * @default true */ this.isUserSupplied = true; /** * Gets or sets a value indicating whether this item is kept above other non-promoted items. * This property is observable. * @type {Boolean} * @default false */ this.isPromoted = false; /** * Gets or sets a value indicating whether this item is hidden from the catalog. This * property is observable. * @type {Boolean} * @default false */ this.isHidden = false; /** * A message object that is presented to the user when an item or group is initially clicked * The object is of the form {title:string, content:string, key: string, confirmation: boolean, confirmText: string, width: number, height: number}. * This property is observable. * @type {Object} */ this.initialMessage = undefined; /** * Gets or sets the cache duration to use for proxied URLs for this catalog member. If undefined, proxied URLs are effectively cachable * forever. The duration is expressed as a Varnish-like duration string, such as '1d' (one day) or '10000s' (ten thousand seconds). * @type {String} */ this.cacheDuration = undefined; /** * Gets or sets whether or not this member should be forced to use a proxy. * This property is not observable. * @type {Boolean} */ this.forceProxy = false; /** * Gets or sets the dictionary of custom item properties. This property is observable. * @type {Object} */ this.customProperties = {}; /** * An optional unique id for this member, that is stable across renames and moves. * Use uniqueId to get the canonical unique id for this CatalogMember, which is present even if there is no id. * @type {String} */ this.id = undefined; /** * An array of all possible keys that can be used to match to this catalog member when specified in a share link - * used for maintaining backwards compatibility when adding or changing {@link CatalogMember#id}. * * @type {String[]} */ this.shareKeys = undefined; /** * The parent {@link CatalogGroup} of this member. * * @type {CatalogGroup} */ this.parent = undefined; /** * A short report to show on the now viewing tab. This property is observable. * @type {String} */ this.shortReport = undefined; /** * The list of collapsible sections of the short report. Each element of the array is an object literal * with a `name` and `content` property. * @type {ShortReportSection[]} */ this.shortReportSections = []; /* * Gets or sets a value indicating whether this data source is currently loading. This property is observable. * @type {Boolean} */ this.isLoading = false; /** * Whether this catalog member is waiting for a disclaimer to be accepted before showing itself. * * @type {boolean} */ this.isWaitingForDisclaimer = false; /** * Indicates that the source of this data should be hidden from the UI (obviously this isn't super-secure as you * can just look at the network requests). * * @type {boolean} */ this.hideSource = false; /** * The names of items in the {@link CatalogMember#info} array that contain details of the source of this * CatalogMember's data. This should be overridden by children of this class. * * @type {Array} * @private */ this._sourceInfoItemNames = []; /** * The name of the item to show in the catalog, if different from `name`. Default undefined. * This property is observed. * @type {String} * @private */ this._nameInCatalog = undefined; this._loadingPromise = undefined; /** Lookup table for _sourceInfoItemNames, access through {@link CatalogMember#_infoItemsWithSourceInfoLookup} */ this._memoizedInfoItemsSourceLookup = undefined; knockout.track(this, ['name', 'info', 'infoSectionOrder', 'description', 'isUserSupplied', 'isPromoted', 'initialMessage', 'isHidden', 'cacheDuration', 'customProperties', 'shortReport', 'shortReportSections', 'isLoading', 'isWaitingForDisclaimer', '_nameInCatalog']); knockout.defineProperty(this, 'nameSortKey', { get: function() { var parts = this.nameInCatalog.split(/(\d+)/); return parts.map(function(part) { var parsed = parseInt(part, 10); if (parsed === parsed) { return parsed; } else { return part.trim().toLowerCase(); } }); } }); /** * Gets or sets the name of this catalog member in the catalog. By default this is just `name`, but can be overridden. * @member {String} nameInCatalog * @memberOf CatalogMember.prototype */ knockout.defineProperty(this, 'nameInCatalog', { get : function() { return defaultValue(this._nameInCatalog, this.name); }, set : function(value) { this._nameInCatalog = value; } }); }; var descriptionRegex = /description/i; defineProperties(CatalogMember.prototype, { /** * Gets the type of data item represented by this instance. * @memberOf CatalogMember.prototype * @type {String} */ type : { get : function() { throw new DeveloperError('Types derived from CatalogMember must implement a "type" property.'); } }, /** * Gets a human-readable name for this type of data source, such as 'Web Map Service (WMS)'. * @memberOf CatalogMember.prototype * @type {String} */ typeName : { get : function() { throw new DeveloperError('Types derived from CatalogMember must implement a "typeName" property.'); } }, /** * Gets a value that tells the UI whether this is a group. * Groups, when clicked, expand to show their constituent items. * @memberOf CatalogMember.prototype * @type {Boolean} */ isGroup : { get : function() { return false; } }, /** * Gets the Terria instance. * @memberOf CatalogMember.prototype * @type {Terria} */ terria : { get : function() { return this._terria; } }, /** * Gets the set of functions used to update individual properties in {@link CatalogMember#updateFromJson}. * When a property name in the returned object literal matches the name of a property on this instance, the value * will be called as a function and passed a reference to this instance, a reference to the source JSON object * literal, and the name of the property. If part of the update happens asynchronously, the updater function should * return a Promise that resolves when it is complete. * @memberOf CatalogMember.prototype * @type {Object} */ updaters : { get : function() { return CatalogMember.defaultUpdaters; } }, /** * Gets the set of functions used to serialize individual properties in {@link CatalogMember#serializeToJson}. * When a property name on the model matches the name of a property in the serializers object literal, * the value will be called as a function and passed a reference to the model, a reference to the destination * JSON object literal, and the name of the property. * @memberOf CatalogMember.prototype * @type {Object} */ serializers : { get : function() { return CatalogMember.defaultSerializers; } }, /** * Gets the set of names of the properties to be serialized for this object when {@link CatalogMember#serializeToJson} * is called for a share link. * @memberOf CatalogMember.prototype * @type {String[]} */ propertiesForSharing : { get : function() { return CatalogMember.defaultPropertiesForSharing; } }, /** * Tests whether a description is available, either in the 'description' property * or as a member of the 'info' array. * @memberOf CatalogMember.prototype * @type {Boolean} */ hasDescription : { get : function() { return this.description || (this.info && this.info.some(function(i){ return descriptionRegex.test(i.name); })); } }, /** * The canonical unique id for this CatalogMember. Will be the id property if one is present, otherwise it will fall * back to the uniqueId of this item's parent + this item's name. This means that if no id is set anywhere up the * tree, the uniqueId will be a complete path of this member's location. * @memberOf CatalogMember.prototype * @type {String} */ uniqueId : { get : function() { if (this.id) { return this.id; } var parentKey = this.parent ? this.parent.uniqueId + '/' : ''; return parentKey + this.name; } }, /** * All keys that have historically been used to resolve this member - the current uniqueId + past shareKeys. */ allShareKeys : { get: function() { var allShareKeys = [this.uniqueId]; return this.shareKeys ? allShareKeys.concat(this.shareKeys) : allShareKeys; } }, needsDisclaimerShown: { get: function() { return defined(this.initialMessage) && (!defined(this.initialMessage.key) || !this.terria.getLocalProperty(this.initialMessage.key)); } }, /** * A filtered view of {@link CatalogMember#info} that excludes info items that divulge details about the data's * source, as determined by {@link CatalogMember#__sourceInfoItemNames}. */ infoWithoutSources: { get: function() { return defaultValue(this.info, []).filter(function(infoItem) { return !defined(this._infoItemsWithSourceInfoLookup[infoItem.name]); }.bind(this)); } }, /** * Returns a lookup of _sourceInfoItemNames as a map of names to a true value. Memoizes after being called for the * first time. * * @private */ _infoItemsWithSourceInfoLookup: { get: function() { if (!defined(this._memoizedInfoItemsSourceLookup)) { this._memoizedInfoItemsSourceLookup = this._sourceInfoItemNames.reduce(function(lookupSoFar, name) { lookupSoFar[name] = true; return lookupSoFar; }, {}); } return this._memoizedInfoItemsSourceLookup; } } }); /** * Gets or sets the set of default updater functions to use in {@link CatalogMember#updateFromJson}. Types derived from this type * should expose this instance - cloned and modified if necesary - through their {@link CatalogMember#updaters} property. * @type {Object} */ CatalogMember.defaultUpdaters = { nameSortKey: function() {}, info: function(catalogItem, json, propertyName) { if (defined(json.info)) { json.info.forEach(function(infoItem) { var existingItem = catalogItem.info.filter(item => item.name === infoItem.name)[0]; if (defined(existingItem)) { var index = catalogItem.info.indexOf(existingItem); catalogItem.info.splice(index, 1, infoItem); } else { catalogItem.info.push(infoItem); } }); } } }; freezeObject(CatalogMember.defaultUpdaters); /** * Gets or sets the set of default serializer functions to use in {@link CatalogMember#serializeToJson}. Types derived from this type * should expose this instance - cloned and modified if necesary - through their {@link CatalogMember#serializers} property. * @type {Object} */ CatalogMember.defaultSerializers = { nameSortKey: function() {} }; freezeObject(CatalogMember.defaultSerializers); /** * Gets or sets the default set of properties that are serialized when serializing a {@link CatalogMember}-derived object * for a share link. * @type {String[]} */ CatalogMember.defaultPropertiesForSharing = [ 'name' ]; freezeObject(CatalogMember.defaultPropertiesForSharing); /** * Updates the catalog member from a JSON object-literal description of it. * Existing collections with the same name as a collection in the JSON description are * updated. If the description contains a collection with a name that does not yet exist, * it is created. Because parts of the update may happen asynchronously, this method * returns at Promise that will resolve when the update is completely done. * * @param {Object} json The JSON description. The JSON should be in the form of an object literal, not a string. * @param {Object} [options] Object with the following properties: * @param {Boolean} [options.onlyUpdateExistingItems] true to only update existing items and never create new ones, or false is new items * may be created by this update. * @param {Boolean} [options.isUserSupplied] If specified, sets the {@link CatalogMember#isUserSupplied} property of updated catalog members * to the given value. If not specified, the property is left unchanged. * @returns {Promise} A promise that resolves when the update is complete. */ CatalogMember.prototype.updateFromJson = function(json, options) { if (defined(options) && defined(options.isUserSupplied)) { this.isUserSupplied = options.isUserSupplied; } var updatePromise = updateFromJson(this, json, options); // Updating from JSON may trigger a load (e.g. if isEnabled is set to true). So if this catalog item // is now loading, wait on the load promise as well, which we can get by calling load. if (this.isLoading) { return when.all([updatePromise, this.load()]); } else { return updatePromise; } }; /** * Serializes the data item to JSON. * * @param {Object} [options] Object with the following properties: * @param {Function} [options.propertyFilter] Filter function that will be executed to determine whether a property * should be serialized. * @param {Function} [options.itemFilter] Filter function that will be executed for each item in a group to determine * whether that item should be serialized. * @return {Object} The serialized JSON object-literal. */ CatalogMember.prototype.serializeToJson = function(options) { options = defaultValue(options, defaultValue.EMPTY_OBJECT); var result = serializeToJson(this, options.propertyFilter, options); result.type = this.type; result.id = this.uniqueId; if (defined(this.parent)) { result.parents = getParentIds(this.parent).reverse(); } return result; }; /** * Gets the ids of all parents of a catalog member, ordered from the closest descendant to the most distant. Ignores * the root. * @private * @param catalogMember The catalog member to get parent ids for. * @param parentIds A starting list of parent ids to add to (allows the function to work recursively). * @returns {String[]} */ function getParentIds(catalogMember, parentIds) { parentIds = defaultValue(parentIds, []); if (defined(catalogMember.parent)) { return getParentIds(catalogMember.parent, parentIds.concat([catalogMember.uniqueId])); } return parentIds; } /** * Finds an {@link CatalogMember#info} section by name. * @param {String} sectionName The name of the section to find. * @return {Object} The section, or undefined if no section with that name exists. */ CatalogMember.prototype.findInfoSection = function(sectionName) { for (var i = 0; i < this.info.length; ++i) { if (this.info[i].name === sectionName) { return this.info[i]; } } return undefined; }; /** * Goes up the hierarchy and determines if this CatalogMember is connected with the root in terria.catalog, or whether it's * part of a disconnected sub-tree. */ CatalogMember.prototype.connectsWithRoot = function() { var item = this; while (item.parent) { item = item.parent; } return item === this.terria.catalog.group; }; /** * "Enables" this catalog member in a way that makes sense for its implementation (e.g. isEnabled for items, isOpen for * groups, and all its parents and ancestors in the tree. */ CatalogMember.prototype.enableWithParents = function() { throw new DeveloperError('Types derived from CatalogMember must implement a "enableWithParents" function.'); }; CatalogMember.prototype.waitForDisclaimerIfNeeded = function() { if (this.needsDisclaimerShown) { this.isWaitingForDisclaimer = true; var deferred = when.defer(); this.terria.disclaimerListener(this, function() { this.isWaitingForDisclaimer = false; deferred.resolve(); }.bind(this)); return deferred.promise; } else { return when(); } }; CatalogMember.prototype.load = function() { if (defined(this._loadingPromise)) { // Load already in progress. return this._loadingPromise; } var loadInfluencingValues = []; if (defined(this._getValuesThatInfluenceLoad)) { loadInfluencingValues = this._getValuesThatInfluenceLoad(); } if (arraysAreEqual(loadInfluencingValues, this._lastLoadInfluencingValues)) { // Already loaded, and nothing has changed to force a re-load. return undefined; } this.isLoading = true; var that = this; this._loadingPromise = runLater(function() { that._lastLoadInfluencingValues = []; if (defined(that._getValuesThatInfluenceLoad)) { that._lastLoadInfluencingValues = that._getValuesThatInfluenceLoad(); } return that._load(); }).then(function(result) { that._loadingPromise = undefined; that.isLoading = false; return result; }).otherwise(function(e) { that._lastLoadInfluencingValues = undefined; that._loadingPromise = undefined; that.isLoading = false; throw e; // keep throwing this so we can chain more otherwises. }); return this._loadingPromise; }; /** A collection of static filters functions used during serialization */ CatalogMember.itemFilters = { /** Item filter that returns true if the item is user supplied */ userSuppliedOnly: function(item) { return item.isUserSupplied; }, /** Item filter that returns true if the item is a {@link CatalogItem} that is enabled, or another kind of {@link CatalogMember}. */ enabled: function(item) { return !defined(item.isEnabled) || item.isEnabled; }, /** Item filter that returns true if an item has no local data. */ noLocalData: function(item) { return !defined(item.data); } }; CatalogMember.propertyFilters = { /** * Property filter that returns true if the property is in that item's {@link CatalogMember#propertiesForSharing} array. */ sharedOnly: function(property, item) { return item.propertiesForSharing.indexOf(property) >= 0; } }; module.exports = CatalogMember;