123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365 |
- /*global define*/
- define([
- '../Core/defined',
- '../Core/defineProperties',
- '../Core/DeveloperError',
- '../Core/Event',
- '../Core/RuntimeError',
- './Property'
- ], function(
- defined,
- defineProperties,
- DeveloperError,
- Event,
- RuntimeError,
- Property) {
- "use strict";
- function resolveEntity(that) {
- var entityIsResolved = true;
- if (that._resolveEntity) {
- var targetEntity = that._targetCollection.getById(that._targetId);
- if (defined(targetEntity)) {
- targetEntity.definitionChanged.addEventListener(ReferenceProperty.prototype._onTargetEntityDefinitionChanged, that);
- that._targetEntity = targetEntity;
- that._resolveEntity = false;
- } else {
- //The property has become detached. It has a valid value but is not currently resolved to an entity in the collection
- targetEntity = that._targetEntity;
- entityIsResolved = false;
- }
- if (!defined(targetEntity)) {
- throw new RuntimeError('target entity "' + that._targetId + '" could not be resolved.');
- }
- }
- return entityIsResolved;
- }
- function resolve(that) {
- var targetProperty = that._targetProperty;
- if (that._resolveProperty) {
- var entityIsResolved = resolveEntity(that);
- var names = that._targetPropertyNames;
- targetProperty = that._targetEntity;
- var length = names.length;
- for (var i = 0; i < length && defined(targetProperty); i++) {
- targetProperty = targetProperty[names[i]];
- }
- if (defined(targetProperty)) {
- that._targetProperty = targetProperty;
- that._resolveProperty = !entityIsResolved;
- } else if (!defined(that._targetProperty)) {
- throw new RuntimeError('targetProperty "' + that._targetId + '.' + names.join('.') + '" could not be resolved.');
- }
- }
- return targetProperty;
- }
- /**
- * A {@link Property} which transparently links to another property on a provided object.
- *
- * @alias ReferenceProperty
- * @constructor
- *
- * @param {EntityCollection} targetCollection The entity collection which will be used to resolve the reference.
- * @param {String} targetId The id of the entity which is being referenced.
- * @param {String} targetPropertyNames The name of the property on the target entity which we will use.
- *
- * @example
- * var collection = new Cesium.EntityCollection();
- *
- * //Create a new entity and assign a billboard scale.
- * var object1 = new Cesium.Entity('object1');
- * object1.billboard = new Cesium.BillboardGraphics();
- * object1.billboard.scale = new Cesium.ConstantProperty(2.0);
- * collection.add(object1);
- *
- * //Create a second entity and reference the scale from the first one.
- * var object2 = new Cesium.Entity('object2');
- * object2.model = new Cesium.ModelGraphics();
- * object2.model.scale = new Cesium.ReferenceProperty(collection, 'object1', ['billboard', 'scale']);
- * collection.add(object2);
- *
- * //Create a third object, but use the fromString helper function.
- * var object3 = new Cesium.Entity('object3');
- * object3.billboard = new Cesium.BillboardGraphics();
- * object3.billboard.scale = Cesium.ReferenceProperty.fromString(collection, 'object1#billboard.scale');
- * collection.add(object3);
- *
- * //You can refer to an entity with a # or . in id and property names by escaping them.
- * var object4 = new Cesium.Entity('#object.4');
- * object4.billboard = new Cesium.BillboardGraphics();
- * object4.billboard.scale = new Cesium.ConstantProperty(2.0);
- * collection.add(object4);
- *
- * var object5 = new Cesium.Entity('object5');
- * object5.billboard = new Cesium.BillboardGraphics();
- * object5.billboard.scale = Cesium.ReferenceProperty.fromString(collection, '\\#object\\.4#billboard.scale');
- * collection.add(object5);
- */
- var ReferenceProperty = function(targetCollection, targetId, targetPropertyNames) {
- //>>includeStart('debug', pragmas.debug);
- if (!defined(targetCollection)) {
- throw new DeveloperError('targetCollection is required.');
- }
- if (!defined(targetId) || targetId === '') {
- throw new DeveloperError('targetId is required.');
- }
- if (!defined(targetPropertyNames) || targetPropertyNames.length === 0) {
- throw new DeveloperError('targetPropertyNames is required.');
- }
- for (var i = 0; i < targetPropertyNames.length; i++) {
- var item = targetPropertyNames[i];
- if (!defined(item) || item === '') {
- throw new DeveloperError('reference contains invalid properties.');
- }
- }
- //>>includeEnd('debug');
- this._targetCollection = targetCollection;
- this._targetId = targetId;
- this._targetPropertyNames = targetPropertyNames;
- this._targetProperty = undefined;
- this._targetEntity = undefined;
- this._definitionChanged = new Event();
- this._resolveEntity = true;
- this._resolveProperty = true;
- targetCollection.collectionChanged.addEventListener(ReferenceProperty.prototype._onCollectionChanged, this);
- };
- defineProperties(ReferenceProperty.prototype, {
- /**
- * Gets a value indicating if this property is constant.
- * @memberof ReferenceProperty.prototype
- * @type {Boolean}
- * @readonly
- */
- isConstant : {
- get : function() {
- return Property.isConstant(resolve(this));
- }
- },
- /**
- * Gets the event that is raised whenever the definition of this property changes.
- * The definition is changed whenever the referenced property's definition is changed.
- * @memberof ReferenceProperty.prototype
- * @type {Event}
- * @readonly
- */
- definitionChanged : {
- get : function() {
- return this._definitionChanged;
- }
- },
- /**
- * Gets the reference frame that the position is defined in.
- * This property is only valid if the referenced property is a {@link PositionProperty}.
- * @memberof ReferenceProperty.prototype
- * @type {ReferenceFrame}
- * @readonly
- */
- referenceFrame : {
- get : function() {
- return resolve(this).referenceFrame;
- }
- },
- /**
- * Gets the id of the entity being referenced.
- * @memberof ReferenceProperty.prototype
- * @type {String}
- * @readonly
- */
- targetId : {
- get : function() {
- return this._targetId;
- }
- },
- /**
- * Gets the collection containing the entity being referenced.
- * @memberof ReferenceProperty.prototype
- * @type {EntityCollection}
- * @readonly
- */
- targetCollection : {
- get : function() {
- return this._targetCollection;
- }
- },
- /**
- * Gets the array of property names used to retrieve the referenced property.
- * @memberof ReferenceProperty.prototype
- * @type {String[]}
- * @readonly
- */
- targetPropertyNames : {
- get : function() {
- return this._targetPropertyNames;
- }
- },
- /**
- * Gets the resolved instance of the underlying referenced property.
- * @memberof ReferenceProperty.prototype
- * @type {Property}
- * @readonly
- */
- resolvedProperty : {
- get : function() {
- return resolve(this);
- }
- }
- });
- /**
- * Creates a new instance given the entity collection that will
- * be used to resolve it and a string indicating the target entity id and property.
- * The format of the string is "objectId#foo.bar", where # separates the id from
- * property path and . separates sub-properties. If the reference identifier or
- * or any sub-properties contains a # . or \ they must be escaped.
- *
- * @param {Entity} targetCollection
- * @param {String} referenceString
- * @returns A new instance of ReferenceProperty.
- *
- * @exception {DeveloperError} invalid referenceString.
- */
- ReferenceProperty.fromString = function(targetCollection, referenceString) {
- //>>includeStart('debug', pragmas.debug);
- if (!defined(targetCollection)) {
- throw new DeveloperError('targetCollection is required.');
- }
- if (!defined(referenceString)) {
- throw new DeveloperError('referenceString is required.');
- }
- //>>includeEnd('debug');
- var identifier;
- var values = [];
- var inIdentifier = true;
- var isEscaped = false;
- var token = '';
- for (var i = 0; i < referenceString.length; ++i) {
- var c = referenceString.charAt(i);
- if (isEscaped) {
- token += c;
- isEscaped = false;
- } else if (c === '\\') {
- isEscaped = true;
- } else if (inIdentifier && c === '#') {
- identifier = token;
- inIdentifier = false;
- token = '';
- } else if (!inIdentifier && c === '.') {
- values.push(token);
- token = '';
- } else {
- token += c;
- }
- }
- values.push(token);
- return new ReferenceProperty(targetCollection, identifier, values);
- };
- /**
- * Gets the value of the property at the provided time.
- *
- * @param {JulianDate} time The time for which to retrieve the value.
- * @param {Object} [result] The object to store the value into, if omitted, a new instance is created and returned.
- * @returns {Object} The modified result parameter or a new instance if the result parameter was not supplied.
- */
- ReferenceProperty.prototype.getValue = function(time, result) {
- return resolve(this).getValue(time, result);
- };
- /**
- * Gets the value of the property at the provided time and in the provided reference frame.
- * This method is only valid if the property being referenced is a {@link PositionProperty}.
- *
- * @param {JulianDate} time The time for which to retrieve the value.
- * @param {ReferenceFrame} referenceFrame The desired referenceFrame of the result.
- * @param {Cartesian3} [result] The object to store the value into, if omitted, a new instance is created and returned.
- * @returns {Cartesian3} The modified result parameter or a new instance if the result parameter was not supplied.
- */
- ReferenceProperty.prototype.getValueInReferenceFrame = function(time, referenceFrame, result) {
- return resolve(this).getValueInReferenceFrame(time, referenceFrame, result);
- };
- /**
- * Gets the {@link Material} type at the provided time.
- * This method is only valid if the property being referenced is a {@link MaterialProperty}.
- *
- * @param {JulianDate} time The time for which to retrieve the type.
- * @returns {String} The type of material.
- */
- ReferenceProperty.prototype.getType = function(time) {
- return resolve(this).getType(time);
- };
- /**
- * Compares this property to the provided property and returns
- * <code>true</code> if they are equal, <code>false</code> otherwise.
- *
- * @param {Property} [other] The other property.
- * @returns {Boolean} <code>true</code> if left and right are equal, <code>false</code> otherwise.
- */
- ReferenceProperty.prototype.equals = function(other) {
- if (this === other) {
- return true;
- }
- var names = this._targetPropertyNames;
- var otherNames = other._targetPropertyNames;
- if (this._targetCollection !== other._targetCollection || //
- this._targetId !== other._targetId || //
- names.length !== otherNames.length) {
- return false;
- }
- var length = this._targetPropertyNames.length;
- for (var i = 0; i < length; i++) {
- if (names[i] !== otherNames[i]) {
- return false;
- }
- }
- return true;
- };
- ReferenceProperty.prototype._onTargetEntityDefinitionChanged = function(targetEntity, name, value, oldValue) {
- if (this._targetPropertyNames[0] === name) {
- this._resolveProperty = true;
- this._definitionChanged.raiseEvent(this);
- }
- };
- ReferenceProperty.prototype._onCollectionChanged = function(collection, added, removed) {
- var targetEntity = this._targetEntity;
- if (defined(targetEntity)) {
- if (removed.indexOf(targetEntity) !== -1) {
- targetEntity.definitionChanged.removeEventListener(ReferenceProperty.prototype._onTargetEntityDefinitionChanged, this);
- this._resolveEntity = true;
- this._resolveProperty = true;
- } else if (this._resolveEntity) {
- //If targetEntity is defined but resolveEntity is true, then the entity is detached
- //and any change to the collection needs to incur an attempt to resolve in order to re-attach.
- //without this if block, a reference that becomes re-attached will not signal definitionChanged
- resolve(this);
- if (!this._resolveEntity) {
- this._definitionChanged.raiseEvent(this);
- }
- }
- }
- };
- return ReferenceProperty;
- });
|