mirror of https://github.com/CesiumGS/cesium.git
571 lines
17 KiB
JavaScript
571 lines
17 KiB
JavaScript
import {
|
|
Cartesian3,
|
|
JulianDate,
|
|
Matrix3,
|
|
Matrix4,
|
|
Quaternion,
|
|
TrackingReferenceFrame,
|
|
TimeInterval,
|
|
TimeIntervalCollection,
|
|
Transforms,
|
|
BillboardGraphics,
|
|
BoxGraphics,
|
|
ConstantPositionProperty,
|
|
ConstantProperty,
|
|
CorridorGraphics,
|
|
CylinderGraphics,
|
|
EllipseGraphics,
|
|
EllipsoidGraphics,
|
|
Entity,
|
|
LabelGraphics,
|
|
ModelGraphics,
|
|
PathGraphics,
|
|
PlaneGraphics,
|
|
PointGraphics,
|
|
PolygonGraphics,
|
|
PolylineGraphics,
|
|
PolylineVolumeGraphics,
|
|
RectangleGraphics,
|
|
WallGraphics,
|
|
} from "../../index.js";
|
|
|
|
describe("DataSources/Entity", function () {
|
|
it("constructor sets expected properties.", function () {
|
|
let entity = new Entity();
|
|
expect(entity.id).toBeDefined();
|
|
expect(entity.name).toBeUndefined();
|
|
expect(entity.billboard).toBeUndefined();
|
|
expect(entity.box).toBeUndefined();
|
|
expect(entity.corridor).toBeUndefined();
|
|
expect(entity.cylinder).toBeUndefined();
|
|
expect(entity.description).toBeUndefined();
|
|
expect(entity.ellipse).toBeUndefined();
|
|
expect(entity.ellipsoid).toBeUndefined();
|
|
expect(entity.label).toBeUndefined();
|
|
expect(entity.model).toBeUndefined();
|
|
expect(entity.orientation).toBeUndefined();
|
|
expect(entity.path).toBeUndefined();
|
|
expect(entity.plane).toBeUndefined();
|
|
expect(entity.point).toBeUndefined();
|
|
expect(entity.polygon).toBeUndefined();
|
|
expect(entity.polyline).toBeUndefined();
|
|
expect(entity.polylineVolume).toBeUndefined();
|
|
expect(entity.position).toBeUndefined();
|
|
expect(entity.rectangle).toBeUndefined();
|
|
expect(entity.viewFrom).toBeUndefined();
|
|
expect(entity.wall).toBeUndefined();
|
|
expect(entity.entityCollection).toBeUndefined();
|
|
expect(entity.trackingReferenceFrame).toBe(
|
|
TrackingReferenceFrame.AUTODETECT,
|
|
);
|
|
|
|
const options = {
|
|
id: "someId",
|
|
name: "bob",
|
|
show: false,
|
|
trackingReferenceFrame: TrackingReferenceFrame.INERTIAL,
|
|
availability: new TimeIntervalCollection(),
|
|
parent: new Entity(),
|
|
customProperty: {},
|
|
billboard: {},
|
|
box: {},
|
|
corridor: {},
|
|
cylinder: {},
|
|
description: "description",
|
|
ellipse: {},
|
|
ellipsoid: {},
|
|
label: {},
|
|
model: {},
|
|
orientation: new Quaternion(1, 2, 3, 4),
|
|
path: {},
|
|
plane: {},
|
|
point: {},
|
|
polygon: {},
|
|
polyline: {},
|
|
polylineVolume: {},
|
|
position: new Cartesian3(5, 6, 7),
|
|
rectangle: {},
|
|
viewFrom: new Cartesian3(8, 9, 10),
|
|
wall: {},
|
|
};
|
|
|
|
entity = new Entity(options);
|
|
expect(entity.id).toEqual(options.id);
|
|
expect(entity.name).toEqual(options.name);
|
|
expect(entity.show).toBe(options.show);
|
|
expect(entity.trackingReferenceFrame).toBe(options.trackingReferenceFrame);
|
|
expect(entity.availability).toBe(options.availability);
|
|
expect(entity.parent).toBe(options.parent);
|
|
expect(entity.customProperty).toBe(options.customProperty);
|
|
|
|
expect(entity.billboard).toBeInstanceOf(BillboardGraphics);
|
|
expect(entity.box).toBeInstanceOf(BoxGraphics);
|
|
expect(entity.corridor).toBeInstanceOf(CorridorGraphics);
|
|
expect(entity.cylinder).toBeInstanceOf(CylinderGraphics);
|
|
expect(entity.description).toBeInstanceOf(ConstantProperty);
|
|
expect(entity.ellipse).toBeInstanceOf(EllipseGraphics);
|
|
expect(entity.ellipsoid).toBeInstanceOf(EllipsoidGraphics);
|
|
expect(entity.label).toBeInstanceOf(LabelGraphics);
|
|
expect(entity.model).toBeInstanceOf(ModelGraphics);
|
|
expect(entity.orientation).toBeInstanceOf(ConstantProperty);
|
|
expect(entity.path).toBeInstanceOf(PathGraphics);
|
|
expect(entity.plane).toBeInstanceOf(PlaneGraphics);
|
|
expect(entity.point).toBeInstanceOf(PointGraphics);
|
|
expect(entity.polygon).toBeInstanceOf(PolygonGraphics);
|
|
expect(entity.polyline).toBeInstanceOf(PolylineGraphics);
|
|
expect(entity.polylineVolume).toBeInstanceOf(PolylineVolumeGraphics);
|
|
expect(entity.position).toBeInstanceOf(ConstantPositionProperty);
|
|
expect(entity.rectangle).toBeInstanceOf(RectangleGraphics);
|
|
expect(entity.viewFrom).toBeInstanceOf(ConstantProperty);
|
|
expect(entity.wall).toBeInstanceOf(WallGraphics);
|
|
|
|
expect(entity.entityCollection).toBeUndefined();
|
|
expect(entity.trackingReferenceFrame).toBe(TrackingReferenceFrame.INERTIAL);
|
|
});
|
|
|
|
it("isAvailable is always true if no availability defined.", function () {
|
|
const entity = new Entity();
|
|
expect(entity.isAvailable(JulianDate.now())).toEqual(true);
|
|
});
|
|
|
|
it("isAvailable throw if no time specified.", function () {
|
|
const entity = new Entity();
|
|
expect(function () {
|
|
entity.isAvailable();
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("constructor creates a unique id if one is not provided.", function () {
|
|
const object = new Entity();
|
|
const object2 = new Entity();
|
|
expect(object.id).toBeDefined();
|
|
expect(object.id).not.toEqual(object2.id);
|
|
});
|
|
|
|
it("isAvailable works.", function () {
|
|
const entity = new Entity();
|
|
const interval = TimeInterval.fromIso8601({
|
|
iso8601: "2000-01-01/2001-01-01",
|
|
});
|
|
const intervals = new TimeIntervalCollection();
|
|
intervals.addInterval(interval);
|
|
entity.availability = intervals;
|
|
expect(
|
|
entity.isAvailable(
|
|
JulianDate.addSeconds(interval.start, -1, new JulianDate()),
|
|
),
|
|
).toEqual(false);
|
|
expect(entity.isAvailable(interval.start)).toEqual(true);
|
|
expect(entity.isAvailable(interval.stop)).toEqual(true);
|
|
expect(
|
|
entity.isAvailable(
|
|
JulianDate.addSeconds(interval.stop, 1, new JulianDate()),
|
|
),
|
|
).toEqual(false);
|
|
});
|
|
|
|
it("definitionChanged works for all properties", function () {
|
|
const entity = new Entity();
|
|
const propertyNames = entity.propertyNames;
|
|
const propertyNamesLength = propertyNames.length;
|
|
|
|
const listener = jasmine.createSpy("listener");
|
|
entity.definitionChanged.addEventListener(listener);
|
|
|
|
let i;
|
|
let name;
|
|
let newValue;
|
|
let oldValue;
|
|
//We loop through twice to ensure that oldValue is properly passed in.
|
|
for (let x = 0; x < 2; x++) {
|
|
for (i = 0; i < propertyNamesLength; i++) {
|
|
name = propertyNames[i];
|
|
newValue = new ConstantProperty(1);
|
|
oldValue = entity[propertyNames[i]];
|
|
entity[name] = newValue;
|
|
expect(listener).toHaveBeenCalledWith(entity, name, newValue, oldValue);
|
|
}
|
|
}
|
|
});
|
|
|
|
it("definitionChanged works for properties added via addProperty", function () {
|
|
const entity = new Entity();
|
|
|
|
const propertyName = "justForTest";
|
|
entity.addProperty(propertyName);
|
|
const oldValue = new ConstantProperty(1);
|
|
entity[propertyName] = oldValue;
|
|
|
|
const listener = jasmine.createSpy("listener");
|
|
entity.definitionChanged.addEventListener(listener);
|
|
|
|
const newValue = new ConstantProperty(1);
|
|
entity[propertyName] = newValue;
|
|
expect(listener).toHaveBeenCalledWith(
|
|
entity,
|
|
propertyName,
|
|
newValue,
|
|
oldValue,
|
|
);
|
|
});
|
|
|
|
it("merge ignores reserved property names when called with a plain object.", function () {
|
|
const entity = new Entity();
|
|
|
|
//Technically merge requires passing an Entity instance, but we call it internally
|
|
//with a plain object during construction to set up custom properties.
|
|
entity.merge({
|
|
name: undefined,
|
|
availability: undefined,
|
|
parent: undefined,
|
|
});
|
|
expect(entity.name).toBeUndefined();
|
|
expect(entity.availability).toBeUndefined();
|
|
expect(entity.parent).toBeUndefined();
|
|
});
|
|
|
|
it("merge does not overwrite availability", function () {
|
|
const entity = new Entity();
|
|
const interval = TimeInterval.fromIso8601({
|
|
iso8601: "2000-01-01/2001-01-01",
|
|
});
|
|
entity.availability = interval;
|
|
|
|
const entity2 = new Entity();
|
|
const interval2 = TimeInterval.fromIso8601({
|
|
iso8601: "2000-01-01/2001-01-01",
|
|
});
|
|
entity2.availability = interval2;
|
|
|
|
entity.merge(entity2);
|
|
expect(entity.availability).toBe(interval);
|
|
});
|
|
|
|
it("merge does not overwrite _children", function () {
|
|
const entity = new Entity();
|
|
|
|
entity.merge({
|
|
children: true,
|
|
});
|
|
expect(entity._children).toEqual([]);
|
|
});
|
|
|
|
it("merge works with custom properties.", function () {
|
|
const propertyName = "customProperty";
|
|
const value = "fizzbuzz";
|
|
|
|
const source = new Entity({
|
|
id: "source",
|
|
});
|
|
source.addProperty(propertyName);
|
|
source[propertyName] = value;
|
|
|
|
const target = new Entity({
|
|
id: "target",
|
|
});
|
|
|
|
//Merging should actually call addProperty for the customProperty.
|
|
spyOn(target, "addProperty").and.callThrough();
|
|
target.merge(source);
|
|
|
|
expect(target.addProperty).toHaveBeenCalledWith(propertyName);
|
|
expect(target[propertyName]).toEqual(source[propertyName]);
|
|
});
|
|
|
|
it("merge throws with undefined source", function () {
|
|
const entity = new Entity();
|
|
expect(function () {
|
|
entity.merge(undefined);
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("computeModelMatrix throws if no time specified.", function () {
|
|
const entity = new Entity();
|
|
expect(function () {
|
|
entity.computeModelMatrix();
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("computeModelMatrix returns undefined when position is undefined.", function () {
|
|
const entity = new Entity();
|
|
entity.orientation = new ConstantProperty(Quaternion.IDENTITY);
|
|
expect(entity.computeModelMatrix(new JulianDate())).toBeUndefined();
|
|
});
|
|
|
|
it("computeModelMatrix returns correct value.", function () {
|
|
const entity = new Entity();
|
|
|
|
const position = new Cartesian3(123456, 654321, 123456);
|
|
const orientation = new Quaternion(1, 2, 3, 4);
|
|
Quaternion.normalize(orientation, orientation);
|
|
|
|
entity.position = new ConstantProperty(position);
|
|
entity.orientation = new ConstantProperty(orientation);
|
|
|
|
const modelMatrix = entity.computeModelMatrix(new JulianDate());
|
|
const expected = Matrix4.fromRotationTranslation(
|
|
Matrix3.fromQuaternion(orientation),
|
|
position,
|
|
);
|
|
expect(modelMatrix).toEqual(expected);
|
|
});
|
|
|
|
it("computeModelMatrix returns ENU when quaternion is undefined.", function () {
|
|
const entity = new Entity();
|
|
const position = new Cartesian3(123456, 654321, 123456);
|
|
entity.position = new ConstantProperty(position);
|
|
|
|
const modelMatrix = entity.computeModelMatrix(new JulianDate());
|
|
const expected = Transforms.eastNorthUpToFixedFrame(position);
|
|
expect(modelMatrix).toEqual(expected);
|
|
});
|
|
|
|
it("computeModelMatrix works with result parameter.", function () {
|
|
const entity = new Entity();
|
|
const position = new Cartesian3(123456, 654321, 123456);
|
|
entity.position = new ConstantProperty(position);
|
|
|
|
const result = new Matrix4();
|
|
const modelMatrix = entity.computeModelMatrix(new JulianDate(), result);
|
|
const expected = Transforms.eastNorthUpToFixedFrame(position);
|
|
expect(modelMatrix).toBe(result);
|
|
expect(modelMatrix).toEqual(expected);
|
|
});
|
|
|
|
it("can add and remove custom properties.", function () {
|
|
const entity = new Entity();
|
|
expect(entity.hasOwnProperty("bob")).toBe(false);
|
|
expect(entity.propertyNames).not.toContain("bob");
|
|
|
|
entity.addProperty("bob");
|
|
expect(entity.hasOwnProperty("bob")).toBe(true);
|
|
expect(entity.propertyNames).toContain("bob");
|
|
|
|
entity.removeProperty("bob");
|
|
expect(entity.hasOwnProperty("bob")).toBe(false);
|
|
expect(entity.propertyNames).not.toContain("bob");
|
|
});
|
|
|
|
it("can re-add removed properties", function () {
|
|
const entity = new Entity();
|
|
entity.addProperty("bob");
|
|
entity.removeProperty("bob");
|
|
entity.addProperty("bob");
|
|
expect(entity.hasOwnProperty("bob")).toBe(true);
|
|
expect(entity.propertyNames).toContain("bob");
|
|
});
|
|
|
|
it("addProperty throws with no property specified.", function () {
|
|
const entity = new Entity();
|
|
expect(function () {
|
|
entity.addProperty(undefined);
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("addProperty throws with no property specified.", function () {
|
|
const entity = new Entity();
|
|
expect(function () {
|
|
entity.addProperty(undefined);
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("removeProperty throws with no property specified.", function () {
|
|
const entity = new Entity();
|
|
expect(function () {
|
|
entity.removeProperty(undefined);
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("addProperty throws when adding an existing property.", function () {
|
|
const entity = new Entity();
|
|
entity.addProperty("bob");
|
|
expect(function () {
|
|
entity.addProperty("bob");
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("removeProperty throws when non-existent property.", function () {
|
|
const entity = new Entity();
|
|
expect(function () {
|
|
entity.removeProperty("bob");
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("addProperty throws with defined reserved property name.", function () {
|
|
const entity = new Entity();
|
|
expect(function () {
|
|
entity.addProperty("merge");
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("removeProperty throws with defined reserved property name.", function () {
|
|
const entity = new Entity();
|
|
expect(function () {
|
|
entity.removeProperty("merge");
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("addProperty throws with undefined reserved property name.", function () {
|
|
const entity = new Entity();
|
|
expect(entity.name).toBeUndefined();
|
|
expect(function () {
|
|
entity.addProperty("name");
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("removeProperty throws with undefined reserved property name.", function () {
|
|
const entity = new Entity();
|
|
expect(entity.name).toBeUndefined();
|
|
expect(function () {
|
|
entity.removeProperty("name");
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("isShowing works without parent.", function () {
|
|
const entity = new Entity({
|
|
show: false,
|
|
});
|
|
expect(entity.isShowing).toBe(false);
|
|
|
|
const listener = jasmine.createSpy("listener");
|
|
entity.definitionChanged.addEventListener(listener);
|
|
|
|
entity.show = true;
|
|
expect(listener.calls.count()).toBe(2);
|
|
expect(listener.calls.argsFor(0)).toEqual([
|
|
entity,
|
|
"isShowing",
|
|
true,
|
|
false,
|
|
]);
|
|
expect(listener.calls.argsFor(1)).toEqual([entity, "show", true, false]);
|
|
expect(entity.isShowing).toBe(true);
|
|
|
|
listener.calls.reset();
|
|
|
|
entity.show = false;
|
|
expect(listener.calls.count()).toBe(2);
|
|
expect(listener.calls.argsFor(0)).toEqual([
|
|
entity,
|
|
"isShowing",
|
|
false,
|
|
true,
|
|
]);
|
|
expect(listener.calls.argsFor(1)).toEqual([entity, "show", false, true]);
|
|
expect(entity.isShowing).toBe(false);
|
|
});
|
|
|
|
function ancestorShowTest(entity, ancestor) {
|
|
const listener = jasmine.createSpy("listener");
|
|
entity.definitionChanged.addEventListener(listener);
|
|
|
|
ancestor.show = false;
|
|
|
|
//Setting ancestor show to false causes entity to raise
|
|
//its own isShowing event, but not the show event.
|
|
expect(listener.calls.count()).toBe(1);
|
|
expect(listener.calls.argsFor(0)).toEqual([
|
|
entity,
|
|
"isShowing",
|
|
false,
|
|
true,
|
|
]);
|
|
expect(entity.show).toBe(true);
|
|
expect(entity.isShowing).toBe(false);
|
|
|
|
listener.calls.reset();
|
|
|
|
//Since isShowing is already false, setting show to false causes the show event
|
|
//but not the isShowing event to be raised
|
|
entity.show = false;
|
|
expect(entity.show).toBe(false);
|
|
expect(listener.calls.count()).toBe(1);
|
|
expect(listener.calls.argsFor(0)).toEqual([entity, "show", false, true]);
|
|
|
|
listener.calls.reset();
|
|
|
|
//Setting ancestor show to true does not trigger the entity.isShowing event
|
|
//because entity.show is false;
|
|
ancestor.show = true;
|
|
expect(entity.show).toBe(false);
|
|
expect(entity.isShowing).toBe(false);
|
|
expect(listener.calls.count()).toBe(0);
|
|
|
|
listener.calls.reset();
|
|
|
|
//Setting entity.show to try now causes both events to be raised
|
|
//because the ancestor is also showing.
|
|
entity.show = true;
|
|
expect(listener.calls.count()).toBe(2);
|
|
expect(listener.calls.argsFor(0)).toEqual([
|
|
entity,
|
|
"isShowing",
|
|
true,
|
|
false,
|
|
]);
|
|
expect(listener.calls.argsFor(1)).toEqual([entity, "show", true, false]);
|
|
expect(entity.show).toBe(true);
|
|
expect(entity.isShowing).toBe(true);
|
|
}
|
|
|
|
it("isShowing works with parent.", function () {
|
|
const parent = new Entity();
|
|
const entity = new Entity();
|
|
entity.parent = parent;
|
|
ancestorShowTest(entity, parent);
|
|
});
|
|
|
|
it("isShowing works with grandparent.", function () {
|
|
const grandparent = new Entity();
|
|
const parent = new Entity();
|
|
parent.parent = grandparent;
|
|
const entity = new Entity();
|
|
entity.parent = parent;
|
|
ancestorShowTest(entity, grandparent);
|
|
});
|
|
|
|
it("isShowing works when replacing parent.", function () {
|
|
const entity = new Entity();
|
|
entity.parent = new Entity();
|
|
|
|
const listener = jasmine.createSpy("listener");
|
|
entity.definitionChanged.addEventListener(listener);
|
|
|
|
entity.parent = new Entity({
|
|
show: false,
|
|
});
|
|
|
|
expect(listener.calls.count()).toBe(2);
|
|
expect(listener.calls.argsFor(0)).toEqual([
|
|
entity,
|
|
"isShowing",
|
|
false,
|
|
true,
|
|
]);
|
|
expect(entity.show).toBe(true);
|
|
expect(entity.isShowing).toBe(false);
|
|
});
|
|
|
|
it("isShowing works when removing parent.", function () {
|
|
const entity = new Entity();
|
|
entity.parent = new Entity({
|
|
show: false,
|
|
});
|
|
expect(entity.isShowing).toBe(false);
|
|
|
|
const listener = jasmine.createSpy("listener");
|
|
entity.definitionChanged.addEventListener(listener);
|
|
|
|
entity.parent = undefined;
|
|
|
|
expect(listener.calls.count()).toBe(2);
|
|
expect(listener.calls.argsFor(0)).toEqual([
|
|
entity,
|
|
"isShowing",
|
|
true,
|
|
false,
|
|
]);
|
|
expect(entity.isShowing).toBe(true);
|
|
});
|
|
});
|