cesium/packages/engine/Specs/DataSources/KmlDataSourceSpec.js

5865 lines
196 KiB
JavaScript

import {
ArcType,
BoundingRectangle,
Cartesian2,
Cartesian3,
ClockRange,
ClockStep,
Color,
combine,
Credit,
defer,
Ellipsoid,
Event,
HeadingPitchRange,
HeadingPitchRoll,
isDataUri,
Iso8601,
JulianDate,
Math as CesiumMath,
NearFarScalar,
PerspectiveFrustum,
Rectangle,
RequestErrorEvent,
Resource,
RuntimeError,
ColorMaterialProperty,
EntityCollection,
ImageMaterialProperty,
KmlCamera,
KmlDataSource,
KmlLookAt,
KmlTour,
KmlTourFlyTo,
KmlTourWait,
Camera,
HeightReference,
HorizontalOrigin,
LabelStyle,
SceneMode,
} from "../../index.js";
import createCamera from "../../../../Specs/createCamera.js";
import pollToPromise from "../../../../Specs/pollToPromise.js";
describe("DataSources/KmlDataSource", function () {
const parser = new DOMParser();
//simple.png in the DATA/KML directory
const image =
"";
const uberStyle =
'\
<Style>\
<LineStyle>\
<color>aaaaaaaa</color>\
<width>2</width>\
</LineStyle>\
<PolyStyle>\
<color>cccccccc</color>\
<fill>0</fill>\
<outline>0</outline>\
</PolyStyle>\
<IconStyle>\
<color>dddddddd</color>\
<scale>3</scale>\
<heading>45</heading>\
<Icon>\
<href>test.png</href>\
</Icon>\
<hotSpot x="1" y="2" xunits="pixels" yunits="pixels"/>\
</IconStyle>\
<LabelStyle>\
<color>eeeeeeee</color>\
<scale>4</scale>\
</LabelStyle>\
</Style>';
const uberLineColor = Color.fromBytes(0xaa, 0xaa, 0xaa, 0xaa);
const uberLineWidth = 2;
const uberPolyColor = Color.fromBytes(0xcc, 0xcc, 0xcc, 0xcc);
const uberPolyFill = false;
const uberPolyOutline = false;
const uberIconColor = Color.fromBytes(0xdd, 0xdd, 0xdd, 0xdd);
const uberIconScale = 3;
const uberIconHeading = CesiumMath.toRadians(-45);
const uberIconHotspot = new Cartesian2(45, -42);
const uberIconHref = Resource.DEFAULT.getDerivedResource({
url: "test.png",
}).url;
const uberLabelColor = Color.fromBytes(0xee, 0xee, 0xee, 0xee);
const uberLabelScale = 4;
const expectedRefreshLinkHref = Resource.DEFAULT.getDerivedResource({
url: "./Data/KML/refresh.kml",
}).url;
const screenOverlayContainer = document.createElement("div");
const uberCamera = {
positionWC: new Cartesian3(0.0, 0.0, 0.0),
directionWC: new Cartesian3(0.0, 0.0, 1.0),
upWC: new Cartesian3(0.0, 1.0, 0.0),
pitch: 0.0,
heading: 0.0,
frustum: new PerspectiveFrustum(),
computeViewRectangle: function () {
return Rectangle.MAX_VALUE;
},
pickEllipsoid: function () {
return undefined;
},
};
const uberCanvas = {
clientWidth: 512,
clientHeight: 512,
};
const options = {
credit: "This is my credit",
screenOverlayContainer: screenOverlayContainer,
};
uberCamera.frustum.fov = CesiumMath.PI_OVER_FOUR;
uberCamera.frustum.aspectRatio = 1.0;
beforeEach(function () {
// Reset camera - x value will change during onStop tests
uberCamera.positionWC.x = 0.0;
});
it("default constructor has expected values", function () {
const dataSource = new KmlDataSource(options);
expect(dataSource.name).toBeUndefined();
expect(dataSource.clock).toBeUndefined();
expect(dataSource.entities).toBeInstanceOf(EntityCollection);
expect(dataSource.isLoading).toBe(false);
expect(dataSource.changedEvent).toBeInstanceOf(Event);
expect(dataSource.errorEvent).toBeInstanceOf(Event);
expect(dataSource.loadingEvent).toBeInstanceOf(Event);
expect(dataSource.unsupportedNodeEvent).toBeInstanceOf(Event);
expect(dataSource.show).toBe(true);
expect(dataSource.credit).toBeInstanceOf(Credit);
});
it("setting name raises changed event", function () {
const dataSource = new KmlDataSource(options);
const spy = jasmine.createSpy("changedEvent");
dataSource.changedEvent.addEventListener(spy);
const newName = "garfield";
dataSource.name = newName;
expect(dataSource.name).toEqual(newName);
expect(spy.calls.count()).toEqual(1);
expect(spy).toHaveBeenCalledWith(dataSource);
});
it("show sets underlying entity collection show.", function () {
const dataSource = new KmlDataSource(options);
dataSource.show = false;
expect(dataSource.show).toBe(false);
expect(dataSource.show).toEqual(dataSource.entities.show);
dataSource.show = true;
expect(dataSource.show).toBe(true);
expect(dataSource.show).toEqual(dataSource.entities.show);
});
it("load throws with undefined KML", function () {
const dataSource = new KmlDataSource(options);
expect(function () {
dataSource.load(undefined);
}).toThrowDeveloperError();
});
it("load works with a KMZ file", function () {
const dataSource = new KmlDataSource(options);
return Resource.fetchBlob("Data/KML/simple.kmz")
.then(function (blob) {
return dataSource.load(blob);
})
.then(function (source) {
expect(source).toBe(dataSource);
expect(source.entities.values.length).toEqual(1);
});
});
it("load rejects loading non-KMZ file", function () {
const dataSource = new KmlDataSource(options);
const spy = jasmine.createSpy("errorEvent");
dataSource.errorEvent.addEventListener(spy);
return Resource.fetchBlob("Data/Images/Blue.png")
.then(function (blob) {
return dataSource.load(blob);
})
.catch(function (e) {
expect(e).toBeInstanceOf(RuntimeError);
expect(spy).toHaveBeenCalled();
});
});
it("load rejects KMZ file with no KML contained", function () {
return Resource.fetchBlob("Data/KML/empty.kmz")
.then(function (blob) {
return KmlDataSource.load(blob, options);
})
.catch(function (e) {
expect(e).toBeInstanceOf(RuntimeError);
expect(e.message).toEqual("KMZ file does not contain a KML document.");
});
});
it("load works with a KML URL", function () {
const dataSource = new KmlDataSource(options);
return dataSource
.load("Data/KML/simple.kml", options)
.then(function (source) {
expect(source).toBe(dataSource);
expect(source.entities.values.length).toEqual(1);
});
});
it("load works with a KML Resource", function () {
const dataSource = new KmlDataSource(options);
const resource = new Resource({
url: "Data/KML/simple.kml",
});
return dataSource.load(resource, options).then(function (source) {
expect(source).toBe(dataSource);
expect(source.entities.values.length).toEqual(1);
});
});
it("load works with a KMZ URL", function () {
const dataSource = new KmlDataSource(options);
return dataSource.load("Data/KML/simple.kmz").then(function (source) {
expect(source).toBe(dataSource);
expect(source.entities.values.length).toEqual(1);
});
});
it("load works with a KMZ URL with Windows-style paths", function () {
const dataSource = new KmlDataSource(options);
return dataSource.load("Data/KML/backslash.kmz").then(function (source) {
expect(source).toBe(dataSource);
expect(
isDataUri(source.entities.values[0]._billboard._image._value.url),
).toBe(true);
});
});
it("load works with a KMZ Resource", function () {
const dataSource = new KmlDataSource(options);
const resource = new Resource({
url: "Data/KML/simple.kmz",
});
return dataSource.load(resource).then(function (source) {
expect(source).toBe(dataSource);
expect(source.entities.values.length).toEqual(1);
});
});
it("load does deferred loading", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark id="Bob">\
</Placemark>\
<Placemark id="Bob2">\
</Placemark>\
</Document>';
jasmine.clock().install();
jasmine.clock().mockDate(new Date());
// Jasmine doesn't mock performance.now(), so force Date.now()
spyOn(KmlDataSource, "_getTimestamp").and.callFake(function () {
return Date.now();
});
const OrigDeferredLoading = KmlDataSource._DeferredLoading;
let deferredLoading;
spyOn(KmlDataSource, "_DeferredLoading").and.callFake(
function (datasource) {
deferredLoading = new OrigDeferredLoading(datasource);
const process = deferredLoading._process.bind(deferredLoading);
spyOn(deferredLoading, "_process").and.callFake(function (isFirst) {
jasmine.clock().tick(1001); // Step over a second everytime, so we only process 1 feature
return process(isFirst);
});
const giveUpTime = deferredLoading._giveUpTime.bind(deferredLoading);
spyOn(deferredLoading, "_giveUpTime").and.callFake(function () {
giveUpTime();
jasmine.clock().tick(1); // Fire the setTimeout callback
});
return deferredLoading;
},
);
const dataSource = new KmlDataSource(options);
return dataSource
.load(parser.parseFromString(kml, "text/xml"), options)
.then(function (source) {
expect(deferredLoading._process.calls.count()).toEqual(3); // Document and 2 placemarks
jasmine.clock().uninstall();
});
});
it("load inserts missing namespace declaration into kml", function () {
const dataSource = new KmlDataSource(options);
return dataSource
.load("Data/KML/undeclaredNamespaces.kml")
.then(function (source) {
expect(source).toBe(dataSource);
expect(source.entities.values.length).toEqual(1);
});
});
it("load inserts missing namespace declaration into kmz", function () {
const dataSource = new KmlDataSource(options);
return dataSource
.load("Data/KML/undeclaredNamespaces.kmz")
.then(function (source) {
expect(source).toBe(dataSource);
expect(source.entities.values.length).toEqual(1);
});
});
it("load deletes duplicate namespace declaration in kml", function () {
const datasource = new KmlDataSource(options);
return datasource
.load("Data/KML/duplicateNamespace.kml")
.then(function (source) {
expect(source).toBe(datasource);
expect(source.entities.values.length).toEqual(1);
});
});
it("load deletes duplicate namespace declaration in kmz", function () {
const dataSource = new KmlDataSource(options);
return dataSource
.load("Data/KML/duplicateNamespace.kmz")
.then(function (source) {
expect(source).toBe(dataSource);
expect(source.entities.values.length).toEqual(1);
});
});
it("load rejects nonexistent URL", function () {
return KmlDataSource.load("test.invalid", options).catch(function (e) {
expect(e).toBeInstanceOf(RequestErrorEvent);
});
});
it("load rejects loading non-KML URL", function () {
return KmlDataSource.load("Data/Images/Blue.png", options).catch(
function (e) {
expect(e).toBeInstanceOf(RuntimeError);
},
);
});
it("load rejects valid KMZ zip URL with no KML contained", function () {
return KmlDataSource.load("Data/KML/empty.kmz", options).catch(
function (e) {
expect(e).toBeInstanceOf(RuntimeError);
expect(e.message).toEqual("KMZ file does not contain a KML document.");
},
);
});
it("if load contains <icon> tag with no image included, no image is added", function () {
const dataSource = new KmlDataSource(options);
return Resource.fetchBlob("Data/KML/simpleNoIcon.kml")
.then(function (blob) {
return dataSource.load(blob);
})
.then(function (source) {
expect(source.entities);
expect(source.entities.values.length).toEqual(1);
expect(source.entities._entities._array.length).toEqual(1);
expect(
source.entities._entities._array[0]._billboard._image,
).toBeUndefined();
});
});
it("if load does not contain icon <style> tag for placemark, default yellow pin does show", function () {
const dataSource = new KmlDataSource(options);
return Resource.fetchBlob("Data/KML/simpleNoStyle.kml")
.then(function (blob) {
return dataSource.load(blob);
})
.then(function (source) {
expect(source.entities);
expect(source.entities.values.length).toEqual(1);
expect(source.entities._entities._array.length).toEqual(1);
expect(
source.entities._entities._array[0]._billboard._image._value,
).toEqual(dataSource._pinBuilder.fromColor(Color.YELLOW, 64));
});
});
it("if load contains empty <IconStyle> tag for placemark, default yellow pin does show", function () {
const dataSource = new KmlDataSource(options);
return Resource.fetchBlob("Data/KML/simpleEmptyIconStyle.kml")
.then(function (blob) {
return dataSource.load(blob);
})
.then(function (source) {
expect(source.entities);
expect(source.entities.values.length).toEqual(1);
expect(source.entities._entities._array.length).toEqual(1);
expect(
source.entities._entities._array[0]._billboard._image._value,
).toEqual(dataSource._pinBuilder.fromColor(Color.YELLOW, 64));
});
});
it("can handle embedded data tags with missing source attributes", function () {
const dataSource = new KmlDataSource(options);
return Resource.fetchBlob("Data/KML/simpleScriptMissingSrcAttribute.kmz")
.then((blob) => dataSource.load(blob))
.catch(fail);
});
it("sets DataSource name from Document", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document>\
<name>NameInKml</name>\
</Document>';
return KmlDataSource.load(parser.parseFromString(kml, "text/xml"), {
sourceUri: "NameFromUri.kml",
}).then(function (dataSource) {
expect(dataSource.name).toEqual("NameInKml", options);
});
});
it("sets DataSource name from Document with KML element", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<kml>\
<Document>\
<name>NameInKml</name>\
</Document>\
</kml>';
return KmlDataSource.load(parser.parseFromString(kml, "text/xml"), {
sourceUri: "NameFromUri.kml",
}).then(function (dataSource) {
expect(dataSource.name).toEqual("NameInKml", options);
});
});
it("sets DataSource name from sourceUri when not in file", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document>\
</Document>';
return KmlDataSource.load(parser.parseFromString(kml, "text/xml"), {
sourceUri: "NameFromUri.kml",
}).then(function (dataSource) {
expect(dataSource.name).toEqual("NameFromUri.kml");
});
});
it("raises changed event when the name changes", function () {
let kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document>\
<name>NameInKml</name>\
</Document>';
const dataSource = new KmlDataSource(options);
const spy = jasmine.createSpy("changedEvent");
dataSource.changedEvent.addEventListener(spy);
return dataSource
.load(parser.parseFromString(kml, "text/xml"))
.then(function () {
//Initial load
expect(spy).toHaveBeenCalledWith(dataSource);
spy.calls.reset();
return dataSource
.load(parser.parseFromString(kml, "text/xml"))
.then(function () {
//Loading KML with same name
expect(spy).not.toHaveBeenCalled();
kml = kml.replace("NameInKml", "newName");
spy.calls.reset();
return dataSource
.load(parser.parseFromString(kml, "text/xml"))
.then(function () {
//Loading KML with different name.
expect(spy).toHaveBeenCalledWith(dataSource);
});
});
});
});
it("raises loadingEvent event at start and end of load", function () {
const dataSource = new KmlDataSource(options);
const spy = jasmine.createSpy("loadingEvent");
dataSource.loadingEvent.addEventListener(spy);
const promise = dataSource.load("Data/KML/simple.kml");
expect(spy).toHaveBeenCalledWith(dataSource, true);
spy.calls.reset();
return promise.then(function () {
expect(spy).toHaveBeenCalledWith(dataSource, false);
});
});
it("raises unsupportedNodeEvent event when parsing an unsupported kml node type", function () {
const dataSource = new KmlDataSource(options);
const spy = jasmine.createSpy("unsupportedNodeEvent");
dataSource.unsupportedNodeEvent.addEventListener(spy);
return dataSource.load("Data/KML/unsupported.kml").then(function () {
const nodeNames = ["PhotoOverlay"];
expect(spy.calls.count()).toEqual(1);
for (let i = 0; i < nodeNames.length; i++) {
const args = spy.calls.argsFor(i);
expect(args.length).toEqual(7);
expect(args[0]).toBe(dataSource);
expect(args[2].localName).toEqual(nodeNames[i]);
expect(args[3]).toBeInstanceOf(EntityCollection);
expect(args[4]).toBeInstanceOf(EntityCollection);
}
});
});
it("sets DatasourceClock based on feature availability", function () {
const beginDate = JulianDate.fromIso8601("2000-01-01");
const endDate = JulianDate.fromIso8601("2000-01-04");
let kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<GroundOverlay>\
<TimeSpan>\
<begin>2000-01-01</begin>\
<end>2000-01-03</end>\
</TimeSpan>\
</GroundOverlay>\
<Placemark>\
<gx:Track>\
<when>2000-01-02</when>\
<gx:coord>1 2 3</gx:coord>\
<when>2000-01-04</when>\
<gx:coord>4 5 6</gx:coord>\
</gx:Track>\
</Placemark>\
</Document>';
return KmlDataSource.load(parser.parseFromString(kml, "text/xml"), options)
.then(function (dataSource) {
const clock = dataSource.clock;
expect(dataSource.clock).toBeDefined();
expect(clock.startTime).toEqual(beginDate);
expect(clock.stopTime).toEqual(endDate);
expect(clock.currentTime).toEqual(beginDate);
expect(clock.clockRange).toEqual(ClockRange.LOOP_STOP);
expect(clock.clockStep).toEqual(ClockStep.SYSTEM_CLOCK_MULTIPLIER);
expect(clock.multiplier).toEqual(
JulianDate.secondsDifference(endDate, beginDate) / 60,
);
return dataSource;
})
.then(function (dataSource) {
//Loading a static data set should clear the clock.
kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<GroundOverlay>\
</GroundOverlay>\
</Document>';
return dataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
);
})
.then(function (dataSource) {
expect(dataSource.clock).toBeUndefined();
});
});
it("Feature: id", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark id="Bob">\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.id).toBe("Bob");
});
});
it("Feature: duplicate id", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark id="Bob">\
</Placemark>\
<Placemark id="Bob">\
</Placemark>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].id).toBe("Bob");
expect(entities[1].id).not.toBe("Bob");
});
});
it("Feature: name", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<name>bob</name>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.name).toBe("bob");
expect(entity.label).toBeDefined();
expect(entity.label.text.getValue()).toBe("bob");
});
});
it("Feature: address", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<address>1826 South 16th Street</address>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.kml.address).toBe("1826 South 16th Street");
});
});
it("Feature: phoneNumber", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<phoneNumber>555-555-5555</phoneNumber>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.kml.phoneNumber).toBe("555-555-5555");
});
});
it("Feature: Snippet", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Snippet>Hey!</Snippet>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.kml.snippet).toBe("Hey!");
});
});
it("Feature: atom:author", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:atom="http://www.w3.org/2005/Atom">\
<atom:author>\
<atom:name>J.R.R. Tolkien</atom:name>\
<atom:email>gandalf@greyhavens.invalid</atom:email>\
<atom:uri>http://greyhavens.invalid</atom:uri>\
</atom:author>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.kml).toBeDefined();
expect(entity.kml.author).toBeDefined();
expect(entity.kml.author.name).toBe("J.R.R. Tolkien");
expect(entity.kml.author.email).toBe("gandalf@greyhavens.invalid");
expect(entity.kml.author.uri).toBe("http://greyhavens.invalid");
});
});
it("Feature: atom:link", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:atom="http://www.w3.org/2005/Atom">\
<atom:link\
href="http://test.invalid"\
hreflang="en-US"\
rel="alternate"\
type="text/plain"\
title="Invalid!"\
length="123"/>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.kml).toBeDefined();
expect(entity.kml.link).toBeDefined();
expect(entity.kml.link.href).toEqual("http://test.invalid");
expect(entity.kml.link.hreflang).toEqual("en-US");
expect(entity.kml.link.rel).toEqual("alternate");
expect(entity.kml.link.type).toEqual("text/plain");
expect(entity.kml.link.title).toEqual("Invalid!");
expect(entity.kml.link.length).toEqual("123");
});
});
it("Feature: TimeSpan with begin and end", function () {
const endDate = JulianDate.fromIso8601("1945-08-06");
const beginDate = JulianDate.fromIso8601("1941-12-07");
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<TimeSpan>\
<begin>1945-08-06</begin>\
<end>1941-12-07</end>\
</TimeSpan>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.availability).toBeDefined();
expect(entity.availability.start).toEqual(beginDate);
expect(entity.availability.stop).toEqual(endDate);
});
});
it("Feature: TimeSpan flips dates when end is earlier", function () {
const endDate = JulianDate.fromIso8601("1945-08-06");
const beginDate = JulianDate.fromIso8601("1941-12-07");
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<TimeSpan>\
<begin>1941-12-07</begin>\
<end>1945-08-06</end>\
</TimeSpan>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.availability).toBeDefined();
expect(entity.availability.start).toEqual(beginDate);
expect(entity.availability.stop).toEqual(endDate);
});
});
it("Feature: TimeSpan gracefully handles empty fields", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<TimeSpan>\
</TimeSpan>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.availability).toBeUndefined();
});
});
it("Feature: TimeSpan works with end only interval", function () {
const date = JulianDate.fromIso8601("1941-12-07");
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<TimeSpan>\
<end>1941-12-07</end>\
</TimeSpan>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.availability).toBeDefined();
expect(entity.availability.start).toEqual(Iso8601.MINIMUM_VALUE);
expect(entity.availability.stop).toEqual(date);
});
});
it("Feature: TimeSpan works with begin only interval", function () {
const date = JulianDate.fromIso8601("1941-12-07");
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<TimeSpan>\
<begin>1941-12-07</begin>\
</TimeSpan>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.availability).toBeDefined();
expect(entity.availability.start).toEqual(date);
expect(entity.availability.stop).toEqual(Iso8601.MAXIMUM_VALUE);
});
});
it("Feature: TimeStamp works", function () {
const date = JulianDate.fromIso8601("1941-12-07");
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<TimeStamp>\
<when>1941-12-07</when>\
</TimeStamp>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.availability).toBeDefined();
expect(entity.availability.start).toEqual(date);
expect(entity.availability.stop).toEqual(Iso8601.MAXIMUM_VALUE);
});
});
it("Feature: visibility works", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<visibility>0</visibility>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.show).toBe(false);
});
});
it("Feature: TimeStamp gracefully handles empty fields", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<TimeStamp>\
</TimeStamp>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.availability).toBeUndefined();
});
});
it("Feature: TimeStamp gracefully handles empty when field", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<TimeStamp>\
<when></when>\
</TimeStamp>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.availability).toBeUndefined();
});
});
it("Feature: ExtendedData <Data> schema", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<ExtendedData>\
<Data name="prop1">\
<displayName>Property 1</displayName>\
<value>1</value>\
</Data>\
<Data name="prop2">\
<value>2</value>\
</Data>\
<Data name="prop3">\
<displayName>Property 3</displayName>\
</Data>\
</ExtendedData>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.kml.extendedData).toBeDefined();
expect(entity.kml.extendedData.prop1).toBeDefined();
expect(entity.kml.extendedData.prop1.displayName).toEqual("Property 1");
expect(entity.kml.extendedData.prop1.value).toEqual("1");
expect(entity.kml.extendedData.prop2).toBeDefined();
expect(entity.kml.extendedData.prop2.displayName).toBeUndefined();
expect(entity.kml.extendedData.prop2.value).toEqual("2");
expect(entity.kml.extendedData.prop3).toBeDefined();
expect(entity.kml.extendedData.prop3.displayName).toEqual("Property 3");
expect(entity.kml.extendedData.prop3.value).toBeUndefined();
});
});
it("GroundOverlay: Sets defaults", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.name).toBeUndefined();
expect(entity.availability).toBeUndefined();
expect(entity.description).toBeUndefined();
expect(entity.rectangle).toBeDefined();
expect(entity.rectangle.height).toBeUndefined();
expect(entity.rectangle.rotation).toBeUndefined();
expect(entity.rectangle.coordinates).toBeUndefined();
expect(entity.rectangle.material).toBeUndefined();
});
});
it("GroundOverlay: Sets rectangle image material", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay>\
<Icon>\
<href>http://test.invalid/image.png</href>\
</Icon>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.rectangle.material).toBeInstanceOf(ImageMaterialProperty);
expect(entity.rectangle.material.image.getValue().url).toEqual(
"http://test.invalid/image.png",
);
});
});
it("GroundOverlay: Sets rectangle image material with color", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay>\
<color>7F0000FF</color>\
<Icon>\
<href>http://test.invalid/image.png</href>\
</Icon>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.rectangle.material).toBeInstanceOf(ImageMaterialProperty);
expect(entity.rectangle.material.image.getValue().url).toEqual(
"http://test.invalid/image.png",
);
expect(entity.rectangle.material.color.getValue()).toEqual(
new Color(1.0, 0.0, 0.0, 127 / 255),
);
});
});
it("GroundOverlay: Sets rectangle color material", function () {
const color = Color.fromBytes(0xcc, 0xdd, 0xee, 0xff);
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay>\
<color>ffeeddcc</color>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.rectangle.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.rectangle.material.color.getValue()).toEqual(color);
});
});
it("GroundOverlay: Sets rectangle coordinates, rotation and zIndex", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay>\
<LatLonBox>\
<west>3</west>\
<south>1</south>\
<east>4</east>\
<north>2</north>\
<rotation>45</rotation>\
</LatLonBox>\
<drawOrder>3</drawOrder>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon).toBeUndefined();
expect(entity.rectangle.coordinates.getValue()).toEqualEpsilon(
Rectangle.fromDegrees(3, 1, 4, 2),
CesiumMath.EPSILON14,
);
expect(entity.rectangle.rotation.getValue()).toEqual(Math.PI / 4);
expect(entity.rectangle.stRotation.getValue()).toEqual(Math.PI / 4);
expect(entity.rectangle.zIndex.getValue()).toEqual(3);
});
});
it("GroundOverlay: Handles wrapping longitude.", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay>\
<LatLonBox>\
<west>-180</west>\
<south>-90</south>\
<east>180</east>\
<north>90</north>\
</LatLonBox>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon).toBeUndefined();
expect(entity.rectangle.coordinates.getValue()).toEqual(
Rectangle.fromDegrees(-180, -90, 180, 90),
);
});
});
it("GroundOverlay: Handles out-of-range latitudes.", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay>\
<LatLonBox>\
<west>-180</west>\
<south>-100</south>\
<east>180</east>\
<north>100</north>\
</LatLonBox>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon).toBeUndefined();
expect(entity.rectangle.coordinates.getValue()).toEqual(
Rectangle.fromDegrees(-180, -90, 180, 90),
);
});
});
it("GroundOverlay: Sets polygon coordinates for gx:LatLonQuad", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:LatLonQuad>\
<coordinates>\
1,2 3,4 5,6 7,8\
</coordinates>\
</gx:LatLonQuad>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.rectangle).toBeUndefined();
expect(entity.polygon.hierarchy.getValue().positions).toEqualEpsilon(
Cartesian3.fromDegreesArray([1, 2, 3, 4, 5, 6, 7, 8]),
CesiumMath.EPSILON14,
);
});
});
it("GroundOverlay: Sets polygon image for gx:LatLonQuad", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Icon>\
<href>http://test.invalid/image.png</href>\
</Icon>\
<gx:LatLonQuad>\
<coordinates>\
1,2 3,4 5,6 7,8\
</coordinates>\
</gx:LatLonQuad>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.material).toBeInstanceOf(ImageMaterialProperty);
expect(entity.polygon.material.image.getValue().url).toEqual(
"http://test.invalid/image.png",
);
});
});
it("GroundOverlay: Sets polygon zIndex for gx:LatLonQuad", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:LatLonQuad>\
<coordinates>\
1,2 3,4 5,6 7,8\
</coordinates>\
</gx:LatLonQuad>\
<drawOrder>3</drawOrder>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.zIndex.getValue()).toEqual(3);
});
});
it("GroundOverlay: Sets rectangle absolute height", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay>\
<altitudeMode>absolute</altitudeMode>\
<altitude>23</altitude>\
</GroundOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.rectangle.height.getValue()).toEqual(23);
});
});
it("ScreenOverlay: Single overlay image created", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<ScreenOverlay>\
<Icon>\
<href>http://invalid.url/</href>\
</Icon>\
<screenXY x="0" y="1" xunits="fraction" yunits="fraction" />\
<overlayXY x="0" y="1" xunits="fraction" yunits="fraction" />\
<size x="-1" y="-1" xunits="pixels" yunits="pixels" />\
</ScreenOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(screenOverlayContainer.children.length).toEqual(1);
const child = screenOverlayContainer.children[0];
expect(child.tagName).toEqual("IMG");
expect(child.getAttribute("src")).toEqual("http://invalid.url/");
child.onload();
expect(child.style.position).toEqual("absolute");
expect(child.style.width).toEqual("");
expect(child.style.height).toEqual("");
expect(child.style.top).toEqual("");
expect(["calc(100% + 0px)", "calc(100% - 0px)"]).toContain(
child.style.bottom,
);
expect(child.style.right).toEqual("");
expect(["calc(0% + 0px)", "calc(0% - 0px)"]).toContain(child.style.left);
dataSource.destroy();
});
});
it("ScreenOverlay: Multiple overlay images created", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document>\
<ScreenOverlay>\
<Icon>\
<href>http://invalid.url/first</href>\
</Icon>\
<screenXY x="0" y="1" xunits="fraction" yunits="fraction" />\
<overlayXY x="0" y="1" xunits="fraction" yunits="fraction" />\
<size x="-1" y="-1" xunits="pixels" yunits="pixels" />\
</ScreenOverlay>\
<ScreenOverlay>\
<Icon>\
<href>http://invalid.url/second</href>\
</Icon>\
<screenXY x="0" y="1" xunits="fraction" yunits="fraction" />\
<overlayXY x="0" y="1" xunits="fraction" yunits="fraction" />\
<size x="-1" y="-1" xunits="pixels" yunits="pixels" />\
</ScreenOverlay>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(screenOverlayContainer.children.length).toEqual(2);
expect(screenOverlayContainer.children[0].tagName).toEqual("IMG");
expect(screenOverlayContainer.children[0].getAttribute("src")).toEqual(
"http://invalid.url/first",
);
expect(screenOverlayContainer.children[1].tagName).toEqual("IMG");
expect(screenOverlayContainer.children[1].getAttribute("src")).toEqual(
"http://invalid.url/second",
);
dataSource.destroy();
});
});
it("ScreenOverlay: Overlay pixel offset", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<ScreenOverlay>\
<Icon>\
<href>http://invalid.url/</href>\
</Icon>\
<screenXY x=".3" y=".5" xunits="fraction" yunits="fraction" />\
<overlayXY x="10" y="25" xunits="pixels" yunits="pixels" />\
<size x="-1" y="-1" xunits="pixels" yunits="pixels" />\
</ScreenOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(screenOverlayContainer.children.length).toEqual(1);
const child = screenOverlayContainer.children[0];
expect(child.tagName).toEqual("IMG");
expect(child.getAttribute("src")).toEqual("http://invalid.url/");
child.onload();
expect(child.style.position).toEqual("absolute");
expect(child.style.width).toEqual("");
expect(child.style.height).toEqual("");
expect(child.style.top).toEqual("");
expect(child.style.bottom).toEqual("calc(50% - 25px)");
expect(child.style.right).toEqual("");
expect(child.style.left).toEqual("calc(30% - 10px)");
dataSource.destroy();
});
});
it("ScreenOverlay: Screen pixel offset", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<ScreenOverlay>\
<Icon>\
<href>http://invalid.url/</href>\
</Icon>\
<screenXY x="100" y="250" xunits="pixels" yunits="pixels" />\
<overlayXY x="47" y="31" xunits="pixels" yunits="pixels" />\
<size x="-1" y="-1" xunits="pixels" yunits="pixels" />\
</ScreenOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(screenOverlayContainer.children.length).toEqual(1);
const child = screenOverlayContainer.children[0];
expect(child.tagName).toEqual("IMG");
expect(child.getAttribute("src")).toEqual("http://invalid.url/");
child.onload();
expect(child.style.position).toEqual("absolute");
expect(child.style.width).toEqual("");
expect(child.style.height).toEqual("");
expect(child.style.top).toEqual("");
expect(child.style.bottom).toEqual("219px");
expect(child.style.right).toEqual("");
expect(child.style.left).toEqual("53px");
dataSource.destroy();
});
});
it("ScreenOverlay: Screen insetPixel offset", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<ScreenOverlay>\
<Icon>\
<href>http://invalid.url/</href>\
</Icon>\
<screenXY x="100" y="250" xunits="pixels" yunits="insetPixels" />\
<overlayXY x="47" y="31" xunits="pixels" yunits="pixels" />\
<size x="-1" y="-1" xunits="pixels" yunits="pixels" />\
</ScreenOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(screenOverlayContainer.children.length).toEqual(1);
const child = screenOverlayContainer.children[0];
expect(child.tagName).toEqual("IMG");
expect(child.getAttribute("src")).toEqual("http://invalid.url/");
child.onload();
expect(child.style.position).toEqual("absolute");
expect(child.style.width).toEqual("");
expect(child.style.height).toEqual("");
expect(child.style.top).toEqual("219px");
expect(child.style.bottom).toEqual("");
expect(child.style.right).toEqual("");
expect(child.style.left).toEqual("53px");
dataSource.destroy();
});
});
it("ScreenOverlay: Clean up", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<ScreenOverlay>\
<Icon>\
<href>http://invalid.url/</href>\
</Icon>\
<screenXY x="100" y="250" xunits="pixels" yunits="insetPixels" />\
<overlayXY x="47" y="31" xunits="pixels" yunits="pixels" />\
<size x="-1" y="-1" xunits="pixels" yunits="pixels" />\
</ScreenOverlay>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(screenOverlayContainer.children.length).toEqual(1);
dataSource.destroy();
expect(screenOverlayContainer.children.length).toEqual(0);
});
});
it("Styles: supports local styles with styleUrl", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document>\
<Style id="testStyle">\
<IconStyle>\
<scale>3</scale>\
</IconStyle>\
</Style>\
<Placemark>\
<styleUrl>#testStyle</styleUrl>\
</Placemark>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].billboard.scale.getValue()).toEqual(3.0);
});
});
it("Styles: supports local styles with styleUrl missing #", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document>\
<Style id="testStyle">\
<IconStyle>\
<scale>3</scale>\
</IconStyle>\
</Style>\
<Placemark>\
<styleUrl>testStyle</styleUrl>\
</Placemark>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].billboard.scale.getValue()).toEqual(3.0);
});
});
it("Styles: supports external styles with styleUrl", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<styleUrl>Data/KML/externalStyle.kml#testStyle</styleUrl>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].billboard.scale.getValue()).toEqual(3.0);
});
});
it("Styles: supports external style maps with styleUrl", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<styleUrl>Data/KML/externalStyle.kml#testStyleMap</styleUrl>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].billboard.scale.getValue()).toEqual(3.0);
});
});
it("Styles: inline styles take precedance over shared styles", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document>\
<Style id="testStyle">\
<IconStyle>\
<scale>3</scale>\
<Icon>\
<href>http://test.invalid</href>\
</Icon>\
</IconStyle>\
</Style>\
<Placemark>\
<styleUrl>#testStyle</styleUrl>\
<Style>\
<IconStyle>\
<scale>2</scale>\
<heading>4</heading>\
</IconStyle>\
</Style>\
</Placemark>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
const billboard = entities[0].billboard;
expect(billboard.scale.getValue()).toEqual(2.0);
expect(billboard.rotation.getValue()).toEqual(CesiumMath.toRadians(-4.0));
expect(billboard.image.getValue().url).toEqual("http://test.invalid/");
});
});
it("Styles: colorMode random", function () {
CesiumMath.setRandomNumberSeed(0);
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<color>ccffffff</color>\
<colorMode>random</colorMode>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const generatedColor =
dataSource.entities.values[0].billboard.color.getValue();
expect(generatedColor.red).toBeLessThan(1.0);
expect(generatedColor.green).toBeLessThan(1.0);
expect(generatedColor.blue).toBeLessThan(1.0);
expect(generatedColor.alpha).toEqual(0.8);
});
});
it("Styles: colorMode random black", function () {
CesiumMath.setRandomNumberSeed(0);
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<color>cc000000</color>\
<colorMode>random</colorMode>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const generatedColor =
dataSource.entities.values[0].billboard.color.getValue();
expect(generatedColor.red).toEqual(0);
expect(generatedColor.green).toEqual(0);
expect(generatedColor.blue).toEqual(0);
expect(generatedColor.alpha).toEqual(0.8);
});
});
it("Styles: empty color", function () {
CesiumMath.setRandomNumberSeed(0);
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<color></color>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values[0].billboard.color).toBeUndefined();
});
});
it("Styles: Applies expected styles to Point geometry", function () {
const kml = `<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>${uberStyle}\
<name>TheName</name>\
<Point>\
<altitudeMode>absolute</altitudeMode>\
<coordinates>1,2,3</coordinates>\
</Point>\
</Placemark>\
</Document>`;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.label).toBeDefined();
expect(entity.label.text.getValue()).toEqual("TheName");
expect(entity.label.fillColor.getValue()).toEqual(uberLabelColor);
expect(entity.label.scale.getValue()).toEqual(uberLabelScale);
expect(entity.billboard.color.getValue()).toEqual(uberIconColor);
expect(entity.billboard.scale.getValue()).toEqual(uberIconScale);
expect(entity.billboard.rotation.getValue()).toEqual(uberIconHeading);
expect(entity.billboard.image.getValue().url).toEqual(uberIconHref);
expect(entity.billboard.pixelOffset.getValue()).toEqual(uberIconHotspot);
expect(entity.polyline).toBeUndefined();
});
});
it("Styles: Applies expected styles to extruded Point geometry", function () {
const kml = `<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>${uberStyle}\
<name>TheName</name>\
<Point>\
<extrude>1</extrude>\
<altitudeMode>absolute</altitudeMode>\
<coordinates>1,2,3</coordinates>\
</Point>\
</Placemark>\
</Document>`;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.label.text.getValue()).toEqual("TheName");
expect(entity.label.fillColor.getValue()).toEqual(uberLabelColor);
expect(entity.label.scale.getValue()).toEqual(uberLabelScale);
expect(entity.billboard.color.getValue()).toEqual(uberIconColor);
expect(entity.billboard.scale.getValue()).toEqual(uberIconScale);
expect(entity.billboard.rotation.getValue()).toEqual(uberIconHeading);
expect(entity.billboard.image.getValue().url).toEqual(uberIconHref);
expect(entity.billboard.pixelOffset.getValue()).toEqual(uberIconHotspot);
expect(entity.polyline.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.polyline.material.color.getValue()).toEqual(uberLineColor);
expect(entity.polyline.width.getValue()).toEqual(uberLineWidth);
});
});
it("Styles: Applies expected styles to LineString geometry", function () {
const kml = `<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>${uberStyle}\
<name>TheName</name>\
<LineString>\
<coordinates>1,2,3 \
4,5,6 \
</coordinates>\
</LineString>\
</Placemark>\
</Document>`;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polyline.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.polyline.material.color.getValue()).toEqual(uberLineColor);
expect(entity.polyline.width.getValue()).toEqual(uberLineWidth);
expect(entity.label).toBeUndefined();
expect(entity.wall).toBeUndefined();
});
});
it("Styles: Applies expected styles to extruded LineString geometry", function () {
const kml = `<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>${uberStyle}\
<name>TheName</name>\
<LineString>\
<extrude>1</extrude>\
<altitudeMode>absolute</altitudeMode>\
<coordinates>1,2,3 \
4,5,6 \
</coordinates>\
</LineString>\
</Placemark>\
</Document>`;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.wall.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.wall.material.color.getValue()).toEqual(uberPolyColor);
expect(entity.wall.fill.getValue()).toEqual(uberPolyFill);
expect(entity.wall.outline.getValue()).toEqual(true);
expect(entity.wall.outlineColor.getValue()).toEqual(uberLineColor);
expect(entity.wall.outlineWidth.getValue()).toEqual(uberLineWidth);
expect(entity.polyline).toBeUndefined();
expect(entity.label).toBeUndefined();
});
});
it("Styles: Applies expected styles to Polygon geometry", function () {
const kml = `<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>${uberStyle}\
<Polygon>\
<extrude>1</extrude>\
<altitudeMode>absolute</altitudeMode>\
<outerBoundaryIs>\
<LinearRing>\
<coordinates>\
1,2,3\
4,5,6\
7,8,9\
</coordinates>\
</LinearRing>\
</outerBoundaryIs>\
</Polygon>\
</Placemark>\
</Document>`;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.polygon.material.color.getValue()).toEqual(uberPolyColor);
expect(entity.polygon.fill.getValue()).toEqual(uberPolyFill);
expect(entity.polygon.outline.getValue()).toEqual(uberPolyOutline);
expect(entity.polygon.outlineColor.getValue()).toEqual(uberLineColor);
expect(entity.polygon.outlineWidth.getValue()).toEqual(uberLineWidth);
expect(entity.label).toBeUndefined();
});
});
it("Styles: Applies expected styles to gx:Track geometry", function () {
const kml = `<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>${uberStyle}\
<name>TheName</name>\
<gx:Track>\
<altitudeMode>absolute</altitudeMode>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>7 8 9</gx:coord>\
</gx:Track>\
</Placemark>\
</Document>`;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.label).toBeDefined();
expect(entity.label.text.getValue()).toEqual("TheName");
expect(entity.label.fillColor.getValue()).toEqual(uberLabelColor);
expect(entity.label.scale.getValue()).toEqual(uberLabelScale);
expect(entity.billboard.color.getValue()).toEqual(uberIconColor);
expect(entity.billboard.scale.getValue()).toEqual(uberIconScale);
expect(entity.billboard.rotation.getValue()).toEqual(uberIconHeading);
expect(entity.billboard.image.getValue().url).toEqual(uberIconHref);
expect(entity.billboard.pixelOffset.getValue()).toEqual(uberIconHotspot);
expect(entity.path).toBeDefined();
expect(entity.path.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.path.material.color.getValue()).toEqual(uberLineColor);
expect(entity.path.width.getValue()).toEqual(uberLineWidth);
expect(entity.path.leadTime.getValue()).toEqual(0);
expect(entity.path.trailTime).toBeUndefined();
expect(entity.polyline).toBeUndefined();
});
});
it("Styles: Applies expected styles to extruded gx:Track geometry", function () {
const kml = `<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>${uberStyle}\
<name>TheName</name>\
<gx:Track>\
<extrude>1</extrude>\
<altitudeMode>absolute</altitudeMode>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>7 8 9</gx:coord>\
</gx:Track>\
</Placemark>\
</Document>`;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.label.text.getValue()).toEqual("TheName");
expect(entity.label.fillColor.getValue()).toEqual(uberLabelColor);
expect(entity.label.scale.getValue()).toEqual(uberLabelScale);
expect(entity.billboard.color.getValue()).toEqual(uberIconColor);
expect(entity.billboard.scale.getValue()).toEqual(uberIconScale);
expect(entity.billboard.rotation.getValue()).toEqual(uberIconHeading);
expect(entity.billboard.image.getValue().url).toEqual(uberIconHref);
expect(entity.billboard.pixelOffset.getValue()).toEqual(uberIconHotspot);
expect(entity.path).toBeDefined();
expect(entity.path.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.path.material.color.getValue()).toEqual(uberLineColor);
expect(entity.path.width.getValue()).toEqual(uberLineWidth);
expect(entity.path.leadTime.getValue()).toEqual(0);
expect(entity.path.trailTime).toBeUndefined();
expect(entity.polyline.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.polyline.material.color.getValue()).toEqual(uberLineColor);
expect(entity.polyline.width.getValue()).toEqual(uberLineWidth);
});
});
it("Styles: Applies expected styles to gx:MultiTrack geometry", function () {
const kml = `<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>${uberStyle}\
<name>TheName</name>\
<gx:MultiTrack>\
<gx:Track>\
<altitudeMode>absolute</altitudeMode>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>7 8 9</gx:coord>\
</gx:Track>\
</gx:MultiTrack>\
</Placemark>\
</Document>`;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.label).toBeDefined();
expect(entity.label.text.getValue()).toEqual("TheName");
expect(entity.label.fillColor.getValue()).toEqual(uberLabelColor);
expect(entity.label.scale.getValue()).toEqual(uberLabelScale);
expect(entity.billboard.color.getValue()).toEqual(uberIconColor);
expect(entity.billboard.scale.getValue()).toEqual(uberIconScale);
expect(entity.billboard.rotation.getValue()).toEqual(uberIconHeading);
expect(entity.billboard.image.getValue().url).toEqual(uberIconHref);
expect(entity.billboard.pixelOffset.getValue()).toEqual(uberIconHotspot);
expect(entity.path).toBeDefined();
expect(entity.path.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.path.material.color.getValue()).toEqual(uberLineColor);
expect(entity.path.width.getValue()).toEqual(uberLineWidth);
expect(entity.path.leadTime.getValue()).toEqual(0);
expect(entity.path.trailTime).toBeUndefined();
expect(entity.polyline).toBeUndefined();
});
});
it("Styles: Applies expected styles to extruded gx:MultiTrack geometry", function () {
const kml = `<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>${uberStyle}\
<name>TheName</name>\
<gx:MultiTrack>\
<gx:Track>\
<extrude>1</extrude>\
<altitudeMode>absolute</altitudeMode>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>7 8 9</gx:coord>\
</gx:Track>\
</gx:MultiTrack>\
</Placemark>\
</Document>`;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.label.text.getValue()).toEqual("TheName");
expect(entity.label.fillColor.getValue()).toEqual(uberLabelColor);
expect(entity.label.scale.getValue()).toEqual(uberLabelScale);
expect(entity.billboard.color.getValue()).toEqual(uberIconColor);
expect(entity.billboard.scale.getValue()).toEqual(uberIconScale);
expect(entity.billboard.rotation.getValue()).toEqual(uberIconHeading);
expect(entity.billboard.image.getValue().url).toEqual(uberIconHref);
expect(entity.billboard.pixelOffset.getValue()).toEqual(uberIconHotspot);
expect(entity.path).toBeDefined();
expect(entity.path.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.path.material.color.getValue()).toEqual(uberLineColor);
expect(entity.path.width.getValue()).toEqual(uberLineWidth);
expect(entity.path.leadTime.getValue()).toEqual(0);
expect(entity.path.trailTime).toBeUndefined();
expect(entity.polyline.material).toBeInstanceOf(ColorMaterialProperty);
expect(entity.polyline.material.color.getValue()).toEqual(uberLineColor);
expect(entity.polyline.width.getValue()).toEqual(uberLineWidth);
});
});
it("Styles: Applies local StyleMap", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2">\
<Placemark>\
<StyleMap>\
<Pair>\
<key>normal</key>\
<Style>\
<IconStyle>\
<scale>2</scale>\
</IconStyle>\
</Style>\
</Pair>\
</StyleMap>\
</Placemark>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.billboard.scale.getValue()).toBe(2.0);
});
});
it("Styles: Applies normal styleUrl StyleMap", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2">\
<StyleMap id="styleMapExample">\
<Pair>\
<key>normal</key>\
<Style>\
<IconStyle>\
<scale>2</scale>\
</IconStyle>\
</Style>\
</Pair>\
</StyleMap>\
<Placemark>\
<styleUrl>#styleMapExample</styleUrl>\
</Placemark>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.billboard.scale.getValue()).toBe(2.0);
});
});
it("Styles: Applies normal StyleMap containing styleUrl", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2">\
<Style id="normalStyle">\
<IconStyle>\
<scale>2</scale>\
</IconStyle>\
</Style>\
<StyleMap id="styleMapExample">\
<Pair>\
<key>normal</key>\
<styleUrl>#normalStyle</styleUrl>\
</Pair>\
</StyleMap>\
<Placemark>\
<styleUrl>#styleMapExample</styleUrl>\
</Placemark>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.billboard.scale.getValue()).toBe(2.0);
});
});
it("Styles: Applies normal StyleMap containing styleUrl without #", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2">\
<Style id="normalStyle">\
<IconStyle>\
<scale>2</scale>\
</IconStyle>\
</Style>\
<StyleMap id="styleMapExample">\
<Pair>\
<key>normal</key>\
<styleUrl>normalStyle</styleUrl>\
</Pair>\
</StyleMap>\
<Placemark>\
<styleUrl>#styleMapExample</styleUrl>\
</Placemark>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.billboard.scale.getValue()).toBe(2.0);
});
});
it("IconStyle: handles empty element", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].billboard).toBeDefined();
});
});
it("IconStyle: Sets billboard image absolute path", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<Icon>\
<href>http://test.invalid/image.png</href>\
</Icon>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
const billboard = entities[0].billboard;
expect(billboard.image.getValue().url).toEqual(
"http://test.invalid/image.png",
);
});
});
it("IconStyle: Sets billboard with root:// Url", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<Icon>\
<href>root://icons/palette-3</href>\
</Icon>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
const billboard = entities[0].billboard;
expect(billboard.image.getValue().url).toEqual(
"https://maps.google.com/mapfiles/kml/pal3/icon56.png",
);
});
});
it("IconStyle: Sets billboard image relative path", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<Icon>\
<href>image.png</href>\
</Icon>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(parser.parseFromString(kml, "text/xml"), {
sourceUri: "http://test.invalid",
}).then(function (dataSource) {
const entities = dataSource.entities.values;
const billboard = entities[0].billboard;
expect(billboard.image.getValue().url).toEqual(
"http://test.invalid/image.png",
);
});
});
it("IconStyle: Sets billboard image inside KMZ", function () {
return KmlDataSource.load("Data/KML/simple.kmz", options).then(
function (dataSource) {
const entities = dataSource.entities.values;
const billboard = entities[0].billboard;
expect(billboard.image.getValue().url).toEqual(image);
},
);
});
it("IconStyle: Sets billboard image with subregion", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Style>\
<IconStyle>\
<Icon>\
<href>whiteShapes.png</href>\
<gx:x>49</gx:x>\
<gx:y>43</gx:y>\
<gx:w>18</gx:w>\
<gx:h>18</gx:h>\
</Icon>\
</IconStyle>\
</Style>\
</Placemark>';
const expectedIconHref = Resource.DEFAULT.getDerivedResource({
url: "whiteShapes.png",
}).url;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const billboard = dataSource.entities.values[0].billboard;
expect(billboard.image.getValue().url).toEqual(expectedIconHref);
expect(billboard.imageSubRegion.getValue()).toEqual(
new BoundingRectangle(49, 43, 18, 18),
);
});
});
it("IconStyle: Sets billboard image with hotSpot fractions", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<hotSpot x="0.25" y="0.75" xunits="fraction" yunits="fraction"/>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const billboard = dataSource.entities.values[0].billboard;
expect(billboard.pixelOffset.getValue()).toEqual(new Cartesian2(8, 8));
});
});
it("IconStyle: Sets billboard image with hotSpot pixels", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<hotSpot x="1" y="2" xunits="pixels" yunits="pixels"/>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const billboard = dataSource.entities.values[0].billboard;
expect(billboard.pixelOffset.getValue()).toEqual(new Cartesian2(15, -14));
});
});
it("IconStyle: Sets billboard image with hotSpot insetPixels", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<hotSpot x="1" y="2" xunits="insetPixels" yunits="insetPixels"/>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const billboard = dataSource.entities.values[0].billboard;
expect(billboard.pixelOffset.getValue()).toEqual(new Cartesian2(-15, 14));
});
});
it("IconStyle: Sets color", function () {
const color = Color.fromBytes(0xcc, 0xdd, 0xee, 0xff);
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<color>ffeeddcc</color>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].billboard.color.getValue()).toEqual(color);
});
});
it("IconStyle: Sets scale", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<scale>2.2</scale>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].billboard.scale.getValue()).toEqual(2.2);
});
});
it("IconStyle: Sets heading", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<heading>4</heading>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].billboard.rotation.getValue()).toEqual(
CesiumMath.toRadians(-4),
);
expect(entities[0].billboard.alignedAxis.getValue()).toEqual(
Cartesian3.UNIT_Z,
);
});
});
it("BalloonStyle: specify all properties", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark id="The ID">\
<Style>\
<BalloonStyle>\
<bgColor>00224466</bgColor>\
<textColor>66442200</textColor>\
<text>$[name] $[description] $[address] $[Snippet] $[id] $[prop1/displayName] $[prop1] $[prop2/displayName] $[prop2]</text>\
</BalloonStyle>\
</Style>\
<name>The Name</name>\
<description>The Description</description>\
<address>The Address</address>\
<Snippet>The Snippet</Snippet>\
<ExtendedData>\
<Data name="prop1">\
<displayName>The Property</displayName>\
<value>The Value</value>\
</Data>\
</ExtendedData>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const element = document.createElement("div");
element.innerHTML = entity.description.getValue();
const div = element.firstChild;
expect(div.style["word-wrap"]).toEqual("break-word");
expect(div.style["background-color"]).toEqual("rgba(102, 68, 34, 0)");
expect(div.style.color).toEqual("rgba(0, 34, 68, 0.4)");
expect(div.textContent).toEqual(
"The Name The Description The Address The Snippet The ID The Property The Value $[prop2/displayName] $[prop2]",
);
});
});
it("BalloonStyle: entity replacement removes missing values", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<BalloonStyle>\
<text>$[name] $[description] $[address] $[Snippet]</text>\
</BalloonStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const element = document.createElement("div");
element.innerHTML = entity.description.getValue();
const div = element.firstChild;
expect(div.textContent).toEqual(" ");
});
});
it("BalloonStyle: description without BalloonStyle", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<description>The Description</description>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const element = document.createElement("div");
element.innerHTML = entity.description.getValue();
const div = element.firstChild;
expect(div.style["word-wrap"]).toEqual("break-word");
expect(div.style["background-color"]).toEqual("rgb(255, 255, 255)");
expect(div.style.color).toEqual("rgb(0, 0, 0)");
expect(div.textContent).toEqual("The Description");
});
});
it("BalloonStyle: creates table from ExtendedData", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<ExtendedData>\
<Data name="prop1">\
<displayName>Property 1</displayName>\
<value>1</value>\
</Data>\
<Data name="prop2">\
<value>2</value>\
</Data>\
<Data name="prop3">\
<displayName>Property 3</displayName>\
</Data>\
</ExtendedData>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const element = document.createElement("div");
element.innerHTML = entity.description.getValue();
const div = element.firstChild;
expect(div.style["word-wrap"]).toEqual("break-word");
expect(div.style["background-color"]).toEqual("rgb(255, 255, 255)");
expect(div.style.color).toEqual("rgb(0, 0, 0)");
const table = div.firstChild;
expect(table.localName).toEqual("table");
expect(table.rows.length).toBe(3);
expect(table.rows[0].cells.length).toEqual(2);
expect(table.rows[1].cells.length).toEqual(2);
expect(table.rows[2].cells.length).toEqual(2);
expect(table.rows[0].cells[0].textContent).toEqual("Property 1");
expect(table.rows[1].cells[0].textContent).toEqual("prop2");
expect(table.rows[2].cells[0].textContent).toEqual("Property 3");
expect(table.rows[0].cells[1].textContent).toEqual("1");
expect(table.rows[1].cells[1].textContent).toEqual("2");
expect(table.rows[2].cells[1].textContent).toEqual("");
});
});
it("BalloonStyle: does not create a description for empty ExtendedData", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<ExtendedData>\
</ExtendedData>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.description).toBeUndefined();
});
});
it("BalloonStyle: description creates links from text", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<description>http://cesiumjs.org</description>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const element = document.createElement("div");
element.innerHTML = entity.description.getValue();
const a = element.firstChild.firstChild;
expect(a.localName).toEqual("a");
expect(a.getAttribute("href")).toEqual("http://cesiumjs.org/");
expect(a.getAttribute("target")).toEqual("_blank");
});
});
it("BalloonStyle: relative description paths absolute to sourceUri", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<description><![CDATA[<img src="foo/bar.png"/>]]></description>\
</Placemark>';
return KmlDataSource.load(parser.parseFromString(kml, "text/xml"), {
sourceUri: "http://test.invalid",
}).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const element = document.createElement("div");
element.innerHTML = entity.description.getValue();
const a = element.firstChild.firstChild;
expect(a.localName).toEqual("img");
expect(a.getAttribute("src")).toEqual("http://test.invalid/foo/bar.png");
});
});
it("BalloonStyle: description retargets existing links to _blank", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<description><![CDATA[<a href="http://cesiumjs.org" target="_self">Homepage</a>]]></description>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const element = document.createElement("div");
element.innerHTML = entity.description.getValue();
const a = element.firstChild.firstChild;
expect(a.localName).toEqual("a");
expect(a.textContent).toEqual("Homepage");
expect(a.getAttribute("href")).toEqual("http://cesiumjs.org/");
expect(a.getAttribute("target")).toEqual("_blank");
});
});
it("BalloonStyle: description does not create links from non-explicit urls", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<description>states.id google.com</description>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const element = document.createElement("div");
element.innerHTML = entity.description.getValue();
const div = element.firstChild;
expect(div.innerHTML).toEqual("states.id google.com");
});
});
it("BalloonStyle: description is rewritten for embedded kmz links and images", function () {
return KmlDataSource.load("Data/KML/simple.kmz", options).then(
function (dataSource) {
const entity = dataSource.entities.values[0];
const description = entity.description.getValue();
const div = document.createElement("div");
div.innerHTML = description;
expect(div.textContent).toEqual("image.png image.png");
const children = div.firstChild.querySelectorAll("*");
expect(children.length).toEqual(2);
const link = children[0];
expect(link.localName).toEqual("a");
expect(link.getAttribute("href")).toEqual(image);
expect(link.getAttribute("download")).toEqual("image.png");
const img = children[1];
expect(img.localName).toEqual("img");
expect(img.src).toEqual(image);
},
);
});
it("LabelStyle: Sets defaults", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<LabelStyle>\
</LabelStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
const label = entities[0].label;
expect(label).toBeDefined();
expect(label.text).toBeUndefined();
expect(label.fillColor).toBeUndefined();
expect(label.outlineColor).toBeUndefined();
expect(label.outlineWidth).toBeUndefined();
expect(label.show).toBeUndefined();
expect(label.scale).toBeUndefined();
expect(label.verticalOrigin).toBeUndefined();
expect(label.eyeOffset).toBeUndefined();
expect(label.pixelOffsetScaleByDistance).toBeUndefined();
expect(label.text).toBeUndefined();
expect(label.fillColor).toBeUndefined();
expect(label.outlineColor).toBeUndefined();
expect(label.outlineWidth).toBeUndefined();
expect(label.show).toBeUndefined();
expect(label.scale).toBeUndefined();
expect(label.verticalOrigin).toBeUndefined();
expect(label.eyeOffset).toBeUndefined();
expect(label.pixelOffsetScaleByDistance).toBeUndefined();
expect(label.font.getValue()).toEqual("16px sans-serif");
expect(label.style.getValue()).toEqual(LabelStyle.FILL_AND_OUTLINE);
expect(label.horizontalOrigin.getValue()).toEqual(HorizontalOrigin.LEFT);
expect(label.pixelOffset.getValue()).toEqual(new Cartesian2(17, 0));
expect(label.translucencyByDistance.getValue()).toEqual(
new NearFarScalar(3000000, 1.0, 5000000, 0.0),
);
});
});
it("LabelStyle: Sets color", function () {
const color = Color.fromBytes(0xcc, 0xdd, 0xee, 0xff);
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<LabelStyle>\
<color>ffeeddcc</color>\
</LabelStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].label.fillColor.getValue()).toEqual(color);
});
});
it("LabelStyle: Sets scale", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<scale>2</scale>\
</IconStyle>\
<LabelStyle>\
</LabelStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].label.pixelOffset.getValue()).toEqual(
new Cartesian2(33, 0),
);
});
});
it("LabelStyle: Sets pixelOffset when billboard scaled", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<scale>3</scale>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].label.pixelOffset.getValue()).toEqual(
new Cartesian3(3 * 16 + 1, 0),
);
});
});
it("LabelStyle: Sets pixelOffset when billboard scaled", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<IconStyle>\
<scale>0</scale>\
</IconStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities[0].label.pixelOffset).toBeUndefined();
expect(entities[0].label.horizontalOrigin).toBeUndefined();
});
});
it("LineStyle: Sets defaults", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<LineStyle>\
</LineStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
const polyline = entities[0].polyline;
expect(polyline).toBeDefined();
expect(polyline.positions).toBeUndefined();
expect(polyline.arcType).toBeUndefined();
expect(polyline.width).toBeUndefined();
expect(polyline.show).toBeUndefined();
expect(polyline.material).toBeUndefined();
expect(polyline.granularity).toBeUndefined();
});
});
it("LineStyle: Sets color", function () {
const color = Color.fromBytes(0xcc, 0xdd, 0xee, 0xff);
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<LineStyle>\
<color>ffeeddcc</color>\
</LineStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
const polyline = entities[0].polyline;
expect(polyline.material).toBeInstanceOf(ColorMaterialProperty);
expect(polyline.material.color.getValue()).toEqual(color);
});
});
it("LineStyle: Sets width", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<LineStyle>\
<width>2.75</width>\
</LineStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
const polyline = entities[0].polyline;
expect(polyline.width.getValue()).toEqual(2.75);
});
});
it("PolyStyle: Sets defaults", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<PolyStyle>\
</PolyStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
const polygon = entities[0].polygon;
expect(polygon).toBeDefined();
expect(polygon.hierarchy).toBeUndefined();
expect(polygon.height).toBeUndefined();
expect(polygon.extrudedHeight).toBeUndefined();
expect(polygon.show).toBeUndefined();
expect(polygon.fill).toBeUndefined();
expect(polygon.material).toBeUndefined();
expect(polygon.outlineWidth).toBeUndefined();
expect(polygon.stRotation).toBeUndefined();
expect(polygon.granularity).toBeUndefined();
expect(polygon.perPositionHeight).toBeUndefined();
expect(polygon.outline.getValue()).toBe(true);
expect(polygon.outlineColor.getValue()).toEqual(Color.WHITE);
});
});
it("PolyStyle: Sets color", function () {
const color = Color.fromBytes(0xcc, 0xdd, 0xee, 0xff);
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<PolyStyle>\
<color>ffeeddcc</color>\
</PolyStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
const polygon = entities[0].polygon;
expect(polygon.material).toBeInstanceOf(ColorMaterialProperty);
expect(polygon.material.color.getValue()).toEqual(color);
});
});
it("PolyStyle: Sets fill", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<PolyStyle>\
<fill>0</fill>\
</PolyStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
const polygon = entities[0].polygon;
expect(polygon.fill.getValue()).toEqual(false);
});
});
it("PolyStyle: Sets outline", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<PolyStyle>\
<outline>0</outline>\
</PolyStyle>\
</Style>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
const polygon = entities[0].polygon;
expect(polygon.outline.getValue()).toEqual(false);
});
});
it("Folder: sets parent property", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Folder id="parent">\
<Placemark id="child">\
</Placemark>\
</Folder>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities;
const folder = entities.getById("parent");
const placemark = entities.getById("child");
expect(entities.values.length).toBe(2);
expect(folder).toBeDefined();
expect(placemark.parent).toBe(folder);
});
});
it("Folder: timespan for folder", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Folder>\
<Placemark id="child">\
</Placemark>\
<TimeSpan>\
<begin>2000-01-01</begin>\
<end>2000-01-03</end>\
</TimeSpan>\
</Folder>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const start = JulianDate.fromIso8601("2000-01-01");
const stop = JulianDate.fromIso8601("2000-01-03");
const entities = dataSource.entities.values;
expect(entities.length).toBe(2);
const interval = entities[0].availability;
expect(interval.start).toEqual(start);
expect(interval.stop).toEqual(stop);
expect(entities[0].availability).toEqual(entities[1].availability);
});
});
it("Folder: timespan for folder and feature", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Folder>\
<Placemark id="child">\
<TimeSpan>\
<begin>2000-01-02</begin>\
<end>2000-01-03</end>\
</TimeSpan>\
</Placemark>\
<TimeSpan>\
<begin>2000-01-01</begin>\
<end>2000-01-04</end>\
</TimeSpan>\
</Folder>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const startFolder = JulianDate.fromIso8601("2000-01-01");
const stopFolder = JulianDate.fromIso8601("2000-01-04");
const startFeature = JulianDate.fromIso8601("2000-01-02");
const stopFeature = JulianDate.fromIso8601("2000-01-03");
const entities = dataSource.entities.values;
expect(entities.length).toBe(2);
let interval = entities[0].availability;
expect(interval.start).toEqual(startFolder);
expect(interval.stop).toEqual(stopFolder);
interval = entities[1].availability;
expect(interval.start).toEqual(startFeature);
expect(interval.stop).toEqual(stopFeature);
});
});
it("Folder: timestamp for folder", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Folder>\
<Placemark id="child">\
</Placemark>\
<TimeStamp>\
<when>2000-01-03</when>\
</TimeStamp>\
</Folder>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const start = JulianDate.fromIso8601("2000-01-03");
const entities = dataSource.entities.values;
expect(entities.length).toBe(2);
const interval = entities[0].availability;
expect(interval.start).toEqual(start);
expect(interval.stop).toEqual(Iso8601.MAXIMUM_VALUE);
expect(entities[0].availability).toEqual(entities[1].availability);
});
});
it("Folder: timestamp for folder and feature", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Folder>\
<Placemark id="child">\
<TimeSpan>\
<begin>2000-01-04</begin>\
<end>2000-01-05</end>\
</TimeSpan>\
</Placemark>\
<TimeStamp>\
<when>2000-01-03</when>\
</TimeStamp>\
</Folder>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const startFolder = JulianDate.fromIso8601("2000-01-03");
const startFeature = JulianDate.fromIso8601("2000-01-04");
const stopFeature = JulianDate.fromIso8601("2000-01-05");
const entities = dataSource.entities.values;
expect(entities.length).toBe(2);
let interval = entities[0].availability;
expect(interval.start).toEqual(startFolder);
expect(interval.stop).toEqual(Iso8601.MAXIMUM_VALUE);
interval = entities[1].availability;
expect(interval.start).toEqual(startFeature);
expect(interval.stop).toEqual(stopFeature);
});
});
it("Geometry Point: handles empty Point", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(0, 0, 0),
);
expect(entities[0].polyline).toBeUndefined();
});
});
it("Geometry Point: handles invalid coordinates", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
<altitudeMode>absolute</altitudeMode>\
<coordinates>1,2,3,4</coordinates>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(1, 2, 3),
);
expect(entities[0].polyline).toBeUndefined();
});
});
it("Geometry Point: handles empty coordinates", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
<coordinates></coordinates>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(0, 0, 0),
);
expect(entities[0].polyline).toBeUndefined();
});
});
it("Geometry Point: sets heightReference to clampToGround", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
<coordinates>1,2,3</coordinates>\
</Point>\
</Placemark>';
return KmlDataSource.load(parser.parseFromString(kml, "text/xml"), {
clampToGround: true,
}).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(
entities[0].billboard.heightReference.getValue(Iso8601.MINIMUM_VALUE),
).toEqual(HeightReference.CLAMP_TO_GROUND);
expect(entities[0].polyline).toBeUndefined();
});
});
it("Geometry Point: sets position altitudeMode absolute", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
<altitudeMode>absolute</altitudeMode>\
<coordinates>1,2,3</coordinates>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(1, 2, 3),
);
expect(entities[0].billboard.pixelOffset).toBeUndefined();
expect(entities[0].polyline).toBeUndefined();
});
});
it("Geometry Point: sets position altitudeMode relativeToGround", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
<altitudeMode>relativeToGround</altitudeMode>\
<coordinates>1,2,3</coordinates>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(1, 2, 3),
);
expect(entities[0].billboard.pixelOffset).toBeUndefined();
expect(entities[0].polyline).toBeUndefined();
});
});
it("Geometry Point: does not extrude when altitudeMode is clampToGround", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
<altitudeMode>clampToGround</altitudeMode>\
<coordinates>1,2</coordinates>\
<extrude>1</extrude>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(1, 2, 0),
);
expect(entities[0].billboard.pixelOffset).toBeUndefined();
expect(entities[0].polyline).toBeUndefined();
});
});
it("Geometry Point: does not extrude when gx:altitudeMode is clampToSeaFloor", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Point>\
<gx:altitudeMode>clampToSeaFloor</gx:altitudeMode>\
<coordinates>1,2</coordinates>\
<extrude>1</extrude>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(1, 2),
);
expect(entities[0].polyline).toBeUndefined();
});
});
it("Geometry Point: extrudes when altitudeMode is relativeToGround", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
<altitudeMode>relativeToGround</altitudeMode>\
<coordinates>1,2,3</coordinates>\
<extrude>1</extrude>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(1, 2, 3),
);
expect(entities[0].polyline).toBeDefined();
const positions = entities[0].polyline.positions.getValue(
Iso8601.MINIMUM_VALUE,
);
expect(positions).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2, 3), Cartesian3.fromDegrees(1, 2, 0)],
CesiumMath.EPSILON13,
);
});
});
it("Geometry Point: extrudes when gx:altitudeMode is relativeToSeaFloor", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Point>\
<gx:altitudeMode>relativeToSeaFloor</gx:altitudeMode>\
<coordinates>1,2,3</coordinates>\
<extrude>1</extrude>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(1, 2, 3),
);
expect(entities[0].polyline).toBeDefined();
const positions = entities[0].polyline.positions.getValue(
Iso8601.MINIMUM_VALUE,
);
expect(positions).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2, 3), Cartesian3.fromDegrees(1, 2, 0)],
CesiumMath.EPSILON13,
);
});
});
it("Geometry Point: extrudes when altitudeMode is absolute", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
<altitudeMode>absolute</altitudeMode>\
<coordinates>1,2,3</coordinates>\
<extrude>1</extrude>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(entities[0].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(1, 2, 3),
);
expect(entities[0].polyline).toBeDefined();
const positions = entities[0].polyline.positions.getValue(
Iso8601.MINIMUM_VALUE,
);
expect(positions).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2, 3), Cartesian3.fromDegrees(1, 2, 0)],
CesiumMath.EPSILON13,
);
});
});
it("Geometry Point: correctly converts coordinates using earth ellipsoid", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
<coordinates>24.070617695061806,87.90173269295278,0</coordinates>\
</Point>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
expect(
entities[0].position.getValue(Iso8601.MINIMUM_VALUE),
).toEqualEpsilon(
new Cartesian3(213935.5635247161, 95566.36983235707, 6352461.425213023),
CesiumMath.EPSILON13,
);
});
});
it("Geometry Point: correctly converts coordinates using moon ellipsoid", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Point>\
<coordinates>24.070617695061806,87.90173269295278,0</coordinates>\
</Point>\
</Placemark>';
const moonOptions = combine(options, { ellipsoid: Ellipsoid.MOON });
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
moonOptions,
).then(function (moonDataSource) {
const entities = moonDataSource.entities.values;
expect(entities.length).toEqual(1);
expect(
entities[0].position.getValue(Iso8601.MINIMUM_VALUE),
).toEqualEpsilon(
new Cartesian3(58080.7702560248, 25945.04756005268, 1736235.0758562544),
CesiumMath.EPSILON13,
);
});
});
it("Geometry Polygon: handles empty coordinates", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<outerBoundaryIs>\
<LinearRing>\
<coordinates>\
</coordinates>\
</LinearRing>\
</outerBoundaryIs>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.hierarchy).toBeUndefined();
});
});
it("Geometry Polygon: without holes", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<outerBoundaryIs>\
<LinearRing>\
<coordinates>\
1,2,3\
4,5,6\
7,8,9\
</coordinates>\
</LinearRing>\
</outerBoundaryIs>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const coordinates = [
Cartesian3.fromDegrees(1, 2, 3),
Cartesian3.fromDegrees(4, 5, 6),
Cartesian3.fromDegrees(7, 8, 9),
];
expect(entity.polygon.hierarchy.getValue().positions).toEqual(
coordinates,
);
});
});
it("Geometry Polygon: with holes", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<outerBoundaryIs>\
<LinearRing>\
<coordinates>\
1,2,3\
4,5,6\
7,8,9\
</coordinates>\
</LinearRing>\
</outerBoundaryIs>\
<innerBoundaryIs>\
<LinearRing>\
<coordinates>\
1.1,2.1,3.1\
4.1,5.1,6.1\
7.1,8.1,9.1\
</coordinates>\
</LinearRing>\
</innerBoundaryIs>\
<innerBoundaryIs>\
<LinearRing>\
<coordinates>\
1.2,2.2,3.2\
4.2,5.2,6.2\
7.2,8.2,9.2\
</coordinates>\
</LinearRing>\
</innerBoundaryIs>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const coordinates = [
Cartesian3.fromDegrees(1, 2, 3),
Cartesian3.fromDegrees(4, 5, 6),
Cartesian3.fromDegrees(7, 8, 9),
];
const holeCoordinates = [
Cartesian3.fromDegrees(1.1, 2.1, 3.1),
Cartesian3.fromDegrees(4.1, 5.1, 6.1),
Cartesian3.fromDegrees(7.1, 8.1, 9.1),
];
const holeCoordinates2 = [
Cartesian3.fromDegrees(1.2, 2.2, 3.2),
Cartesian3.fromDegrees(4.2, 5.2, 6.2),
Cartesian3.fromDegrees(7.2, 8.2, 9.2),
];
const hierarchy = entity.polygon.hierarchy.getValue();
expect(hierarchy.positions).toEqual(coordinates);
expect(hierarchy.holes.length).toEqual(2);
expect(hierarchy.holes[0].positions).toEqual(holeCoordinates);
expect(hierarchy.holes[0].holes).toEqual([]);
expect(hierarchy.holes[1].positions).toEqual(holeCoordinates2);
expect(hierarchy.holes[1].holes).toEqual([]);
});
});
it("Geometry Polygon: altitudeMode relativeToGround and can extrude", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<altitudeMode>relativeToGround</altitudeMode>\
<extrude>1</extrude>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.perPositionHeight.getValue()).toEqual(true);
expect(entity.polygon.extrudedHeight.getValue()).toEqual(0);
});
});
it("Geometry Polygon: altitudeMode absolute and can extrude", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<altitudeMode>absolute</altitudeMode>\
<extrude>1</extrude>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.perPositionHeight.getValue()).toEqual(true);
expect(entity.polygon.extrudedHeight.getValue()).toEqual(0);
});
});
it("Geometry Polygon: altitudeMode clampToGround and cannot extrude", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Polygon>\
<altitudeMode>clampToGround</altitudeMode>\
<extrude>1</extrude>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.perPositionHeight).toBeUndefined();
expect(entity.polygon.extrudedHeight).toBeUndefined();
});
});
it("Geometry Polygon: gx:altitudeMode relativeToSeaFloor and can extrude", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Polygon>\
<gx:altitudeMode>relativeToSeaFloor</gx:altitudeMode>\
<extrude>1</extrude>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.perPositionHeight.getValue()).toEqual(true);
expect(entity.polygon.extrudedHeight.getValue()).toEqual(0);
});
});
it("Geometry Polygon: gx:altitudeMode clampToSeaFloor and can extrude", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Polygon>\
<gx:altitudeMode>clampToSeaFloor</gx:altitudeMode>\
<extrude>1</extrude>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.perPositionHeight).toBeUndefined();
expect(entity.polygon.extrudedHeight).toBeUndefined();
});
});
it("Geometry Polygon: When loaded with the Ellipsoid.MOON, the coordinates should be on the lunar ellipsoid, otherwise on Earth.", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<outerBoundaryIs>\
<LinearRing>\
<coordinates>24.070617695061806,87.90173269295278,0\
49.598705282838125,87.04553528415659,0\
34.373553362965566,86.015550572534,0\
14.570663006937881,86.60174704052636,0\
24.070617695061806,87.90173269295278,0\
</coordinates>\
</LinearRing>\
</outerBoundaryIs>\
</Polygon>\
</Placemark>';
const moonOptions = combine(options, { ellipsoid: Ellipsoid.MOON });
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
moonOptions,
).then(function (moonDataSource) {
const entity = moonDataSource.entities.values[0];
const moonPoint = entity.polygon.hierarchy.getValue().positions[0];
expect(moonPoint).toEqualEpsilon(
new Cartesian3(58080.7702560248, 25945.04756005268, 1736235.0758562544),
CesiumMath.EPSILON13,
);
});
});
it("Geometry Polygon: When loaded with the default ellipsoid, the coordinates should be on Earth.", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<outerBoundaryIs>\
<LinearRing>\
<coordinates>24.070617695061806,87.90173269295278,0\
49.598705282838125,87.04553528415659,0\
34.373553362965566,86.015550572534,0\
14.570663006937881,86.60174704052636,0\
24.070617695061806,87.90173269295278,0\
</coordinates>\
</LinearRing>\
</outerBoundaryIs>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
const earthPoint = entity.polygon.hierarchy.getValue().positions[0];
expect(earthPoint).toEqualEpsilon(
new Cartesian3(213935.5635247161, 95566.36983235707, 6352461.425213023),
CesiumMath.EPSILON13,
);
});
});
it("Geometry LineString: handles empty element", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<LineString>\
</LineString>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
const entity = entities[0];
expect(entity.wall).toBeUndefined();
expect(entity.polyline).toBeDefined();
expect(entity.polyline.arcType.getValue()).toEqual(ArcType.NONE);
});
});
it("Geometry LineString: sets positions (clampToGround default)", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<LineString>\
<coordinates>1,2,3 \
4,5,6 \
</coordinates>\
</LineString>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
const entity = entities[0];
expect(entity.wall).toBeUndefined();
expect(entity.polyline).toBeDefined();
const positions = entity.polyline.positions.getValue(
Iso8601.MINIMUM_VALUE,
);
expect(positions).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2), Cartesian3.fromDegrees(4, 5)],
CesiumMath.EPSILON10,
);
expect(entity.polyline.arcType.getValue()).toEqual(ArcType.NONE);
});
});
it("Geometry LineString: sets wall positions when extruded", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<LineString>\
<altitudeMode>absolute</altitudeMode>\
<extrude>1</extrude>\
<coordinates>1,2,3 \
4,5,6 \
</coordinates>\
</LineString>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
const entity = entities[0];
expect(entity.polyline).toBeUndefined();
expect(entity.wall).toBeDefined();
const positions = entity.wall.positions.getValue(Iso8601.MINIMUM_VALUE);
expect(positions).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2, 3), Cartesian3.fromDegrees(4, 5, 6)],
CesiumMath.EPSILON10,
);
});
});
it("Geometry LineString: sets positions altitudeMode clampToGround, cannot extrude, can tessellate", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<LineString>\
<altitudeMode>clampToGround</altitudeMode>\
<extrude>1</extrude>\
<tessellate>1</tessellate>\
<coordinates>1,2,3 4,5,6</coordinates>\
</LineString>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
const entity = entities[0];
expect(entity.polyline.arcType).toBeUndefined();
const positions = entity.polyline.positions.getValue(
Iso8601.MINIMUM_VALUE,
);
expect(positions).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2), Cartesian3.fromDegrees(4, 5)],
CesiumMath.EPSILON10,
);
});
});
it("Geometry LineString: sets positions altitudeMode gx:clampToSeaFloor, cannot extrude, can tessellate", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<LineString>\
<gx:altitudeMode>clampToSeaFloor</gx:altitudeMode>\
<extrude>1</extrude>\
<tessellate>1</tessellate>\
<coordinates>1,2,3 4,5,6</coordinates>\
</LineString>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
const entity = entities[0];
expect(entity.polyline.arcType).toBeUndefined();
const positions = entity.polyline.positions.getValue(
Iso8601.MINIMUM_VALUE,
);
expect(positions).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2), Cartesian3.fromDegrees(4, 5)],
CesiumMath.EPSILON10,
);
});
});
it("Geometry LineString: sets positions altitudeMode gx:relativeToSeaFloor, can extrude, cannot tessellate", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<LineString>\
<gx:altitudeMode>relativeToSeaFloor</gx:altitudeMode>\
<extrude>1</extrude>\
<tessellate>1</tessellate>\
<coordinates>1,2,3 4,5,6</coordinates>\
</LineString>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
const entity = entities[0];
expect(entity.polyline).toBeUndefined(true);
expect(entity.wall).toBeDefined();
const positions = entity.wall.positions.getValue(Iso8601.MINIMUM_VALUE);
expect(positions).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2, 3), Cartesian3.fromDegrees(4, 5, 6)],
CesiumMath.EPSILON10,
);
});
});
it("Geometry LineString: sets positions altitudeMode relativeToGround, can extrude, cannot tessellate", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<LineString>\
<altitudeMode>relativeToGround</altitudeMode>\
<extrude>1</extrude>\
<tessellate>1</tessellate>\
<coordinates>1,2,3 4,5,6</coordinates>\
</LineString>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
const entity = entities[0];
expect(entity.polyline).toBeUndefined();
expect(entity.wall).toBeDefined();
const positions = entity.wall.positions.getValue(Iso8601.MINIMUM_VALUE);
expect(positions).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2, 3), Cartesian3.fromDegrees(4, 5, 6)],
CesiumMath.EPSILON10,
);
});
});
it("Geometry LineString: sets positions altitudeMode absolute, can extrude, cannot tessellate", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<LineString>\
<altitudeMode>absolute</altitudeMode>\
<extrude>1</extrude>\
<tessellate>1</tessellate>\
<coordinates>1,2,3 4,5,6</coordinates>\
</LineString>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(1);
const entity = entities[0];
expect(entity.polyline).toBeUndefined();
expect(entity.wall).toBeDefined();
const positions = entity.wall.positions.getValue(Iso8601.MINIMUM_VALUE);
expect(positions).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2, 3), Cartesian3.fromDegrees(4, 5, 6)],
CesiumMath.EPSILON10,
);
});
});
it("Geometry gx:Track: sets position and availability", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:Track>\
<when>2000-01-01T00:00:00Z</when>\
<gx:coord>1 2 3</gx:coord>\
<when>2000-01-01T00:00:01Z</when>\
<gx:coord>4 5 6</gx:coord>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>7 8 9</gx:coord>\
</gx:Track>\
</Placemark>';
return KmlDataSource.load(parser.parseFromString(kml, "text/xml"), {
clampToGround: true,
}).then(function (dataSource) {
const time1 = JulianDate.fromIso8601("2000-01-01T00:00:00Z");
const time2 = JulianDate.fromIso8601("2000-01-01T00:00:01Z");
const time3 = JulianDate.fromIso8601("2000-01-01T00:00:02Z");
const entity = dataSource.entities.values[0];
expect(entity.position.getValue(time1)).toEqualEpsilon(
Cartesian3.fromDegrees(1, 2, 3),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time2)).toEqualEpsilon(
Cartesian3.fromDegrees(4, 5, 6),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time3)).toEqualEpsilon(
Cartesian3.fromDegrees(7, 8, 9),
CesiumMath.EPSILON12,
);
expect(entity.billboard.heightReference.getValue(time1)).toEqual(
HeightReference.CLAMP_TO_GROUND,
);
expect(entity.billboard.heightReference.getValue(time2)).toEqual(
HeightReference.CLAMP_TO_GROUND,
);
expect(entity.billboard.heightReference.getValue(time3)).toEqual(
HeightReference.CLAMP_TO_GROUND,
);
expect(entity.polyline).toBeUndefined();
expect(entity.availability.start).toEqual(time1);
expect(entity.availability.stop).toEqual(time3);
});
});
it("Geometry gx:Track: sets position clampToGround, cannot extrude", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:Track>\
<altitudeMode>clampToGround</altitudeMode>\
<extrude>1</extrude>\
<when>2000-01-01T00:00:00Z</when>\
<gx:coord>1 2 3</gx:coord>\
<when>2000-01-01T00:00:01Z</when>\
<gx:coord>4 5 6</gx:coord>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>7 8 9</gx:coord>\
</gx:Track>\
</Placemark>';
return KmlDataSource.load(parser.parseFromString(kml, "text/xml"), {
clampToGround: true,
}).then(function (dataSource) {
const time1 = JulianDate.fromIso8601("2000-01-01T00:00:00Z");
const time2 = JulianDate.fromIso8601("2000-01-01T00:00:01Z");
const time3 = JulianDate.fromIso8601("2000-01-01T00:00:02Z");
const entity = dataSource.entities.values[0];
expect(entity.position.getValue(time1)).toEqualEpsilon(
Cartesian3.fromDegrees(1, 2, 3),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time2)).toEqualEpsilon(
Cartesian3.fromDegrees(4, 5, 6),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time3)).toEqualEpsilon(
Cartesian3.fromDegrees(7, 8, 9),
CesiumMath.EPSILON12,
);
expect(entity.billboard.heightReference.getValue(time1)).toEqual(
HeightReference.CLAMP_TO_GROUND,
);
expect(entity.billboard.heightReference.getValue(time2)).toEqual(
HeightReference.CLAMP_TO_GROUND,
);
expect(entity.billboard.heightReference.getValue(time3)).toEqual(
HeightReference.CLAMP_TO_GROUND,
);
expect(entity.polyline).toBeUndefined();
});
});
it("Geometry gx:Track: sets position altitudeMode absolute, can extrude", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:Track>\
<altitudeMode>absolute</altitudeMode>\
<extrude>1</extrude>\
<when>2000-01-01T00:00:00Z</when>\
<gx:coord>1 2 3</gx:coord>\
<when>2000-01-01T00:00:01Z</when>\
<gx:coord>4 5 6</gx:coord>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>7 8 9</gx:coord>\
</gx:Track>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const time1 = JulianDate.fromIso8601("2000-01-01T00:00:00Z");
const time2 = JulianDate.fromIso8601("2000-01-01T00:00:01Z");
const time3 = JulianDate.fromIso8601("2000-01-01T00:00:02Z");
const entity = dataSource.entities.values[0];
expect(entity.position.getValue(time1)).toEqualEpsilon(
Cartesian3.fromDegrees(1, 2, 3),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time2)).toEqualEpsilon(
Cartesian3.fromDegrees(4, 5, 6),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time3)).toEqualEpsilon(
Cartesian3.fromDegrees(7, 8, 9),
CesiumMath.EPSILON12,
);
expect(entity.polyline.positions.getValue(time1)).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2, 3), Cartesian3.fromDegrees(1, 2)],
CesiumMath.EPSILON12,
);
expect(entity.polyline.positions.getValue(time2)).toEqualEpsilon(
[Cartesian3.fromDegrees(4, 5, 6), Cartesian3.fromDegrees(4, 5)],
CesiumMath.EPSILON12,
);
expect(entity.polyline.positions.getValue(time3)).toEqualEpsilon(
[Cartesian3.fromDegrees(7, 8, 9), Cartesian3.fromDegrees(7, 8)],
CesiumMath.EPSILON12,
);
});
});
it("Geometry gx:Track: sets position altitudeMode relativeToGround, can extrude", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:Track>\
<altitudeMode>relativeToGround</altitudeMode>\
<extrude>1</extrude>\
<when>2000-01-01T00:00:00Z</when>\
<gx:coord>1 2 3</gx:coord>\
<when>2000-01-01T00:00:01Z</when>\
<gx:coord>4 5 6</gx:coord>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>7 8 9</gx:coord>\
</gx:Track>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const time1 = JulianDate.fromIso8601("2000-01-01T00:00:00Z");
const time2 = JulianDate.fromIso8601("2000-01-01T00:00:01Z");
const time3 = JulianDate.fromIso8601("2000-01-01T00:00:02Z");
const entity = dataSource.entities.values[0];
expect(entity.position.getValue(time1)).toEqualEpsilon(
Cartesian3.fromDegrees(1, 2, 3),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time2)).toEqualEpsilon(
Cartesian3.fromDegrees(4, 5, 6),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time3)).toEqualEpsilon(
Cartesian3.fromDegrees(7, 8, 9),
CesiumMath.EPSILON12,
);
expect(entity.polyline.positions.getValue(time1)).toEqualEpsilon(
[Cartesian3.fromDegrees(1, 2, 3), Cartesian3.fromDegrees(1, 2)],
CesiumMath.EPSILON12,
);
expect(entity.polyline.positions.getValue(time2)).toEqualEpsilon(
[Cartesian3.fromDegrees(4, 5, 6), Cartesian3.fromDegrees(4, 5)],
CesiumMath.EPSILON12,
);
expect(entity.polyline.positions.getValue(time3)).toEqualEpsilon(
[Cartesian3.fromDegrees(7, 8, 9), Cartesian3.fromDegrees(7, 8)],
CesiumMath.EPSILON12,
);
});
});
it("Geometry gx:Track: sets position and availability when missing values", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:Track>\
<when>2000-01-01T00:00:00Z</when>\
<gx:coord>1 2 3</gx:coord>\
<when>2000-01-01T00:00:01Z</when>\
<gx:coord>4 5 6</gx:coord>\
<when>2000-01-01T00:00:02Z</when>\
</gx:Track>\
</Placemark>';
return KmlDataSource.load(parser.parseFromString(kml, "text/xml"), {
clampToGround: true,
}).then(function (dataSource) {
const time1 = JulianDate.fromIso8601("2000-01-01T00:00:00Z");
const time2 = JulianDate.fromIso8601("2000-01-01T00:00:01Z");
const time3 = JulianDate.fromIso8601("2000-01-01T00:00:02Z");
const entity = dataSource.entities.values[0];
expect(entity.position.getValue(time1)).toEqualEpsilon(
Cartesian3.fromDegrees(1, 2, 3),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time2)).toEqualEpsilon(
Cartesian3.fromDegrees(4, 5, 6),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time3)).toBeUndefined();
// heightReference should be constant so its available all the time
expect(entity.billboard.heightReference.getValue(time1)).toEqual(
HeightReference.CLAMP_TO_GROUND,
);
expect(entity.billboard.heightReference.getValue(time2)).toEqual(
HeightReference.CLAMP_TO_GROUND,
);
expect(entity.billboard.heightReference.getValue(time3)).toEqual(
HeightReference.CLAMP_TO_GROUND,
);
expect(entity.availability.start).toEqual(time1);
expect(entity.availability.stop).toEqual(time2);
});
});
it("Geometry gx:MultiTrack: sets position and availability without interpolate", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:MultiTrack>\
<gx:Track>\
<when>2000-01-01T00:00:00Z</when>\
<gx:coord>1 2 3</gx:coord>\
<when>2000-01-01T00:00:01Z</when>\
<gx:coord>4 5 6</gx:coord>\
</gx:Track>\
<gx:Track>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>6 5 4</gx:coord>\
<when>2000-01-01T00:00:03Z</when>\
<gx:coord>3 2 1</gx:coord>\
</gx:Track>\
</gx:MultiTrack>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const time1 = JulianDate.fromIso8601("2000-01-01T00:00:00Z");
const time2 = JulianDate.fromIso8601("2000-01-01T00:00:01Z");
const time3 = JulianDate.fromIso8601("2000-01-01T00:00:02Z");
const time4 = JulianDate.fromIso8601("2000-01-01T00:00:03Z");
const entity = dataSource.entities.values[0];
expect(entity.availability.length).toEqual(2);
expect(entity.availability.get(0).start).toEqual(time1);
expect(entity.availability.get(0).stop).toEqual(time2);
expect(entity.availability.get(1).start).toEqual(time3);
expect(entity.availability.get(1).stop).toEqual(time4);
expect(entity.position.getValue(time1)).toEqualEpsilon(
Cartesian3.fromDegrees(1, 2),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time2)).toEqualEpsilon(
Cartesian3.fromDegrees(4, 5),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time3)).toEqualEpsilon(
Cartesian3.fromDegrees(6, 5),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time4)).toEqualEpsilon(
Cartesian3.fromDegrees(3, 2),
CesiumMath.EPSILON12,
);
});
});
it("Geometry gx:MultiTrack: sets position and availability with interpolate", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:MultiTrack>\
<gx:interpolate>1</gx:interpolate>\
<gx:Track>\
<when>2000-01-01T00:00:00Z</when>\
<gx:coord>1 2 3</gx:coord>\
<when>2000-01-01T00:00:01Z</when>\
<gx:coord>4 5 6</gx:coord>\
</gx:Track>\
<gx:Track>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>6 5 4</gx:coord>\
<when>2000-01-01T00:00:03Z</when>\
<gx:coord>3 2 1</gx:coord>\
</gx:Track>\
</gx:MultiTrack>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const time1 = JulianDate.fromIso8601("2000-01-01T00:00:00Z");
const time2 = JulianDate.fromIso8601("2000-01-01T00:00:01Z");
const time3 = JulianDate.fromIso8601("2000-01-01T00:00:02Z");
const time4 = JulianDate.fromIso8601("2000-01-01T00:00:03Z");
const entity = dataSource.entities.values[0];
expect(entity.availability.length).toEqual(1);
expect(entity.availability.get(0).start).toEqual(time1);
expect(entity.availability.get(0).stop).toEqual(time4);
expect(entity.position.getValue(time1)).toEqualEpsilon(
Cartesian3.fromDegrees(1, 2),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time2)).toEqualEpsilon(
Cartesian3.fromDegrees(4, 5),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time3)).toEqualEpsilon(
Cartesian3.fromDegrees(6, 5),
CesiumMath.EPSILON12,
);
expect(entity.position.getValue(time4)).toEqualEpsilon(
Cartesian3.fromDegrees(3, 2),
CesiumMath.EPSILON12,
);
});
});
it("Geometry gx:MultiTrack: sets position and availability altitudeMode absolute, extrude, with interpolate", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:MultiTrack>\
<gx:interpolate>1</gx:interpolate>\
<gx:Track>\
<altitudeMode>absolute</altitudeMode>\
<extrude>1</extrude>\
<when>2000-01-01T00:00:00Z</when>\
<gx:coord>1 2 3</gx:coord>\
<when>2000-01-01T00:00:01Z</when>\
<gx:coord>4 5 6</gx:coord>\
</gx:Track>\
<gx:Track>\
<altitudeMode>absolute</altitudeMode>\
<extrude>1</extrude>\
<when>2000-01-01T00:00:02Z</when>\
<gx:coord>6 5 4</gx:coord>\
<when>2000-01-01T00:00:03Z</when>\
<gx:coord>3 2 1</gx:coord>\
</gx:Track>\
</gx:MultiTrack>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const time1 = JulianDate.fromIso8601("2000-01-01T00:00:00Z");
const time2 = JulianDate.fromIso8601("2000-01-01T00:00:01Z");
const time3 = JulianDate.fromIso8601("2000-01-01T00:00:02Z");
const time4 = JulianDate.fromIso8601("2000-01-01T00:00:03Z");
const entity = dataSource.entities.values[0];
expect(entity.availability.length).toEqual(1);
expect(entity.availability.get(0).start).toEqual(time1);
expect(entity.availability.get(0).stop).toEqual(time4);
expect(entity.position.getValue(time1)).toEqual(
Cartesian3.fromDegrees(1, 2, 3),
);
expect(entity.position.getValue(time2)).toEqual(
Cartesian3.fromDegrees(4, 5, 6),
);
expect(entity.position.getValue(time3)).toEqual(
Cartesian3.fromDegrees(6, 5, 4),
);
expect(entity.position.getValue(time4)).toEqual(
Cartesian3.fromDegrees(3, 2, 1),
);
});
});
it("Geometry MultiGeometry: sets expected properties", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark id="testID">\
<MultiGeometry>\
<name>TheName</name>\
<description>TheDescription</description>\
<Point id="point1">\
<coordinates>1,2</coordinates>\
</Point>\
<Point id="point2">\
<coordinates>3,4</coordinates>\
</Point>\
</MultiGeometry>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities;
expect(entities.values.length).toEqual(3);
const multi = entities.getById("testID");
expect(multi).toBeDefined();
const point1 = entities.getById("testIDpoint1");
expect(point1).toBeDefined();
expect(point1.parent).toBe(multi);
expect(point1.name).toBe(multi.name);
expect(point1.description).toBe(multi.description);
expect(point1.kml).toBe(multi.kml);
expect(point1.position.getValue(Iso8601.MINIMUM_VALUE)).toEqualEpsilon(
Cartesian3.fromDegrees(1, 2),
CesiumMath.EPSILON13,
);
const point2 = entities.getById("testIDpoint2");
expect(point2).toBeDefined();
expect(point2.parent).toBe(multi);
expect(point2.name).toBe(multi.name);
expect(point2.description).toBe(multi.description);
expect(point2.kml).toBe(multi.kml);
expect(point2.position.getValue(Iso8601.MINIMUM_VALUE)).toEqualEpsilon(
Cartesian3.fromDegrees(3, 4),
CesiumMath.EPSILON13,
);
});
});
it("NetworkLink: Loads data", function () {
return KmlDataSource.load("Data/KML/networkLink.kml", options).then(
function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(2);
expect(entities[0].id).toEqual("link");
expect(entities[1].parent).toBe(entities[0]);
},
);
});
it("NetworkLink: onInterval", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
<refreshMode>onInterval</refreshMode>\
<refreshInterval>1</refreshInterval>\
</Link>\
</NetworkLink>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(3);
const link1 = entities[0];
const folder1 = entities[1];
const placemark1 = entities[2];
expect(link1.id).toEqual("link");
expect(folder1.parent).toBe(link1);
expect(placemark1.parent).toBe(folder1);
const spy = jasmine.createSpy("refreshEvent");
dataSource.refreshEvent.addEventListener(spy);
return pollToPromise(function () {
dataSource.update(0);
return spy.calls.count() > 0;
}).then(function () {
expect(spy).toHaveBeenCalledWith(dataSource, expectedRefreshLinkHref);
expect(entities.length).toEqual(3);
const link2 = entities[0];
const folder2 = entities[1];
const placemark2 = entities[2];
expect(link2.id).toEqual("link");
expect(folder2.parent).toBe(link2);
expect(placemark2.parent).toBe(folder2);
expect(link2).toEqual(link2);
expect(folder2).not.toEqual(folder1);
expect(placemark2).not.toEqual(placemark1);
});
});
});
it("NetworkLink: onExpire", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
<refreshMode>onExpire</refreshMode>\
</Link>\
</NetworkLink>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(3);
const link1 = entities[0];
const folder1 = entities[1];
const placemark1 = entities[2];
expect(link1.id).toEqual("link");
expect(folder1.parent).toBe(link1);
expect(placemark1.parent).toBe(folder1);
const spy = jasmine.createSpy("refreshEvent");
dataSource.refreshEvent.addEventListener(spy);
dataSource.update(0); // Document is already expired so call once
return pollToPromise(function () {
return spy.calls.count() > 0;
}).then(function () {
expect(spy).toHaveBeenCalledWith(dataSource, expectedRefreshLinkHref);
expect(entities.length).toEqual(3);
const link2 = entities[0];
const folder2 = entities[1];
const placemark2 = entities[2];
expect(link2.id).toEqual("link");
expect(folder2.parent).toBe(link2);
expect(placemark2.parent).toBe(folder2);
expect(link2).toEqual(link2);
expect(folder2).not.toEqual(folder1);
expect(placemark2).not.toEqual(placemark1);
});
});
});
it("NetworkLink: onStop", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
<viewRefreshMode>onStop</viewRefreshMode>\
</Link>\
</NetworkLink>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
Object.assign({
canvas: uberCanvas,
camera: uberCamera,
options: options,
}),
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(3);
const link1 = entities[0];
const folder1 = entities[1];
const placemark1 = entities[2];
expect(link1.id).toEqual("link");
expect(folder1.parent).toBe(link1);
expect(placemark1.parent).toBe(folder1);
const spy = jasmine.createSpy("refreshEvent");
dataSource.refreshEvent.addEventListener(spy);
// Move the camera and call update to set the last camera view
uberCamera.positionWC.x = 1.0;
dataSource.update(0);
return pollToPromise(function () {
return spy.calls.count() > 0;
}).then(function () {
expect(spy).toHaveBeenCalledWith(
dataSource,
`${expectedRefreshLinkHref}?BBOX=-180%2C-90%2C180%2C90`,
);
expect(entities.length).toEqual(3);
const link2 = entities[0];
const folder2 = entities[1];
const placemark2 = entities[2];
expect(link2.id).toEqual("link");
expect(folder2.parent).toBe(link2);
expect(placemark2.parent).toBe(folder2);
expect(link2).toEqual(link2);
expect(folder2).not.toEqual(folder1);
expect(placemark2).not.toEqual(placemark1);
});
});
});
it("NetworkLink: Url is correct on initial load", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
</Link>\
</NetworkLink>';
const requestNetworkLink = defer();
spyOn(Resource._Implementations, "loadWithXhr").and.callFake(
function (
url,
responseType,
method,
data,
headers,
deferred,
overrideMimeType,
) {
requestNetworkLink.resolve(url);
deferred.reject();
},
);
KmlDataSource.load(parser.parseFromString(kml, "text/xml"), options);
return requestNetworkLink.promise.then(function (url) {
expect(url).toEqual(expectedRefreshLinkHref);
});
});
it("NetworkLink can accept invalid but common URL tag instead of Link", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Url>\
<href>./Data/KML/refresh.kml</href>\
</Url>\
</NetworkLink>';
const requestNetworkLink = defer();
spyOn(Resource._Implementations, "loadWithXhr").and.callFake(
function (
url,
responseType,
method,
data,
headers,
deferred,
overrideMimeType,
) {
requestNetworkLink.resolve(url);
deferred.reject();
},
);
KmlDataSource.load(parser.parseFromString(kml, "text/xml"), options);
return requestNetworkLink.promise.then(function (url) {
expect(url).toEqual(expectedRefreshLinkHref);
});
});
it("NetworkLink: Url is correct on initial load with onStop defaults", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
<viewRefreshMode>onStop</viewRefreshMode>\
</Link>\
</NetworkLink>';
const requestNetworkLink = defer();
spyOn(Resource._Implementations, "loadWithXhr").and.callFake(
function (
url,
responseType,
method,
data,
headers,
deferred,
overrideMimeType,
) {
requestNetworkLink.resolve(url);
deferred.reject();
},
);
KmlDataSource.load(parser.parseFromString(kml, "text/xml"), options);
return requestNetworkLink.promise.then(function (url) {
expect(url).toEqual(
`${expectedRefreshLinkHref}?BBOX=-180%2C-90%2C180%2C90`,
);
});
});
it("NetworkLink: Url is correct on initial load with httpQuery without a ?", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
<viewRefreshMode>onInterval</viewRefreshMode>\
<httpQuery>client=[clientName]-v[clientVersion]&amp;v=[kmlVersion]&amp;lang=[language]</httpQuery>\
</Link>\
</NetworkLink>';
const requestNetworkLink = defer();
spyOn(Resource._Implementations, "loadWithXhr").and.callFake(
function (
url,
responseType,
method,
data,
headers,
deferred,
overrideMimeType,
) {
requestNetworkLink.resolve(url);
deferred.reject();
},
);
KmlDataSource.load(parser.parseFromString(kml, "text/xml"), options);
return requestNetworkLink.promise.then(function (url) {
expect(url).toEqual(
`${expectedRefreshLinkHref}?client=Cesium-v1&v=2.2&lang=English`,
);
});
});
it("NetworkLink: Url is correct on initial load with httpQuery with a ?", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
<viewRefreshMode>onInterval</viewRefreshMode>\
<httpQuery>?client=[clientName]-v[clientVersion]&amp;v=[kmlVersion]&amp;lang=[language]</httpQuery>\
</Link>\
</NetworkLink>';
const requestNetworkLink = defer();
spyOn(Resource._Implementations, "loadWithXhr").and.callFake(
function (
url,
responseType,
method,
data,
headers,
deferred,
overrideMimeType,
) {
requestNetworkLink.resolve(url);
deferred.reject();
},
);
KmlDataSource.load(parser.parseFromString(kml, "text/xml"), options);
return requestNetworkLink.promise.then(function (url) {
expect(url).toEqual(
`${expectedRefreshLinkHref}?client=Cesium-v1&v=2.2&lang=English`,
);
});
});
it("NetworkLink: Url is correct on initial load with viewFormat without a ?", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
<viewRefreshMode>onInterval</viewRefreshMode>\
<viewFormat>BBOX=[bboxWest],[bboxSouth],[bboxEast],[bboxNorth];CAMERA=\
[lookatLon],[lookatLat],[lookatRange],[lookatTilt],[lookatHeading];VIEW=\
[horizFov],[vertFov],[horizPixels],[vertPixels],[terrainEnabled]</viewFormat>\
</Link>\
</NetworkLink>';
const requestNetworkLink = defer();
spyOn(Resource._Implementations, "loadWithXhr").and.callFake(
function (
url,
responseType,
method,
data,
headers,
deferred,
overrideMimeType,
) {
requestNetworkLink.resolve(url);
deferred.reject();
},
);
KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
Object.assign({ camera: uberCamera, canvas: uberCanvas }, options),
);
return requestNetworkLink.promise.then(function (url) {
expect(url).toEqual(
`${expectedRefreshLinkHref}?BBOX=-180%2C-90%2C180%2C90&CAMERA=0%2C0%2C6378137%2C0%2C0&VIEW=45%2C45%2C512%2C512%2C1`,
);
});
});
it("NetworkLink: Url is correct on initial load with viewFormat with a ?", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
<viewRefreshMode>onInterval</viewRefreshMode>\
<viewFormat>?BBOX=[bboxWest],[bboxSouth],[bboxEast],[bboxNorth];CAMERA=\
[lookatLon],[lookatLat],[lookatRange],[lookatTilt],[lookatHeading];VIEW=\
[horizFov],[vertFov],[horizPixels],[vertPixels],[terrainEnabled]</viewFormat>\
</Link>\
</NetworkLink>';
const requestNetworkLink = defer();
spyOn(Resource._Implementations, "loadWithXhr").and.callFake(
function (
url,
responseType,
method,
data,
headers,
deferred,
overrideMimeType,
) {
requestNetworkLink.resolve(url);
deferred.reject();
},
);
const src = new KmlDataSource();
src.camera = uberCamera;
src.canvas = uberCanvas;
src.load(parser.parseFromString(kml, "text/xml"), options);
return requestNetworkLink.promise.then(function (url) {
expect(url).toEqual(
`${expectedRefreshLinkHref}?BBOX=-180%2C-90%2C180%2C90&CAMERA=0%2C0%2C6378137%2C0%2C0&VIEW=45%2C45%2C512%2C512%2C1`,
);
});
});
it("NetworkLink: Url is correct on initial load with viewFormat and httpQuery", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
<viewRefreshMode>onInterval</viewRefreshMode>\
<viewFormat>vf=1</viewFormat>\
<httpQuery>hq=1</httpQuery>\
</Link>\
</NetworkLink>';
const requestNetworkLink = defer();
spyOn(Resource._Implementations, "loadWithXhr").and.callFake(
function (
url,
responseType,
method,
data,
headers,
deferred,
overrideMimeType,
) {
requestNetworkLink.resolve(url);
deferred.reject();
},
);
KmlDataSource.load(parser.parseFromString(kml, "text/xml"), options);
return requestNetworkLink.promise.then(function (url) {
expect(url).toEqual(`${expectedRefreshLinkHref}?hq=1&vf=1`);
});
});
it("NetworkLink: onStop when no globe is in view", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/refresh.kml</href>\
<viewRefreshMode>onStop</viewRefreshMode>\
</Link>\
</NetworkLink>';
const ourOptions = {
camera: {
positionWC: new Cartesian3(0.0, 0.0, 0.0),
directionWC: new Cartesian3(0.0, 0.0, 1.0),
upWC: new Cartesian3(0.0, 1.0, 0.0),
pitch: 0.0,
heading: 0.0,
frustum: {
aspectRatio: 1.0,
fov: CesiumMath.PI_OVER_FOUR,
},
computeViewRectangle: function () {
return undefined;
},
pickEllipsoid: function () {
return undefined;
},
},
canvas: uberCanvas,
};
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
ourOptions,
).then(function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toEqual(3);
const link1 = entities[0];
const folder1 = entities[1];
const placemark1 = entities[2];
expect(link1.id).toEqual("link");
expect(folder1.parent).toBe(link1);
expect(placemark1.parent).toBe(folder1);
const spy = jasmine.createSpy("refreshEvent");
dataSource.refreshEvent.addEventListener(spy);
// Move the camera and call update to set the last camera view
ourOptions.camera.positionWC.x = 1.0;
dataSource.update(0);
return pollToPromise(function () {
return spy.calls.count() > 0;
}).then(function () {
expect(spy).toHaveBeenCalledWith(
dataSource,
`${expectedRefreshLinkHref}?BBOX=0%2C0%2C0%2C0`,
);
expect(entities.length).toEqual(3);
const link2 = entities[0];
const folder2 = entities[1];
const placemark2 = entities[2];
expect(link2.id).toEqual("link");
expect(folder2.parent).toBe(link2);
expect(placemark2.parent).toBe(folder2);
expect(link2).toEqual(link2);
expect(folder2).not.toEqual(folder1);
expect(placemark2).not.toEqual(placemark1);
});
});
});
it("NetworkLink: timespan for network link", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/simple.kml</href>\
</Link>\
<TimeSpan>\
<begin>2000-01-01</begin>\
<end>2000-01-03</end>\
</TimeSpan>\
</NetworkLink>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const start = JulianDate.fromIso8601("2000-01-01");
const stop = JulianDate.fromIso8601("2000-01-03");
const entities = dataSource.entities.values;
expect(entities.length).toBe(2);
const interval = entities[0].availability;
expect(interval.start).toEqual(start);
expect(interval.stop).toEqual(stop);
expect(entities[0].availability).toEqual(entities[1].availability);
});
});
it("NetworkLink: timestamp for network link", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/simple.kml</href>\
</Link>\
<TimeStamp>\
<when>2000-01-03</when>\
</TimeStamp>\
</NetworkLink>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const start = JulianDate.fromIso8601("2000-01-03");
const entities = dataSource.entities.values;
expect(entities.length).toBe(2);
const interval = entities[0].availability;
expect(interval.start).toEqual(start);
expect(interval.stop).toEqual(Iso8601.MAXIMUM_VALUE);
expect(entities[0].availability).toEqual(entities[1].availability);
});
});
it("NetworkLink: within a kmz file", function () {
return KmlDataSource.load("Data/KML/multilevel.kmz", options).then(
function (dataSource) {
const entities = dataSource.entities.values;
expect(entities.length).toBe(3);
expect(entities[1].billboard).not.toBeNull();
expect(entities[1].position.getValue(Iso8601.MINIMUM_VALUE)).toEqual(
Cartesian3.fromDegrees(1, 2, 3),
);
// The root network link is loaded, then the children
// since its done recursively the lowest level entities
// end up in the collection first.
expect(entities[0].parent).toBeUndefined();
expect(entities[2].parent).toBe(entities[0]);
expect(entities[1].parent).toBe(entities[2]);
},
);
});
it("can load a KML file with explicit namespaces", function () {
return KmlDataSource.load("Data/KML/namespaced.kml", options).then(
function (dataSource) {
expect(dataSource.entities.values.length).toBe(3);
},
);
});
it("can load styles from a KML file with namespaces", function () {
return KmlDataSource.load("Data/KML/namespaced.kml", options).then(
function (dataSource) {
console.debug(dataSource.entities.values[2]);
const polyline = dataSource.entities.values[2].polyline;
const expectedColor = Color.fromBytes(0xff, 0x00, 0xff, 0x00);
const polylineColor = polyline.material.color.getValue();
expect(polylineColor).toEqual(expectedColor);
expect(polyline.width.getValue()).toEqual(10);
},
);
});
it("Boolean values can use true string", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<altitudeMode>relativeToGround</altitudeMode>\
<extrude>1</extrude>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.perPositionHeight.getValue()).toEqual(true);
});
});
it("Boolean values can use false string", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<altitudeMode>relativeToGround</altitudeMode>\
<extrude>1</extrude>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.perPositionHeight.getValue()).toEqual(true);
});
});
it("Properly finds the root feature node when it is not the first child of the KML node", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<kml xmlns="http://www.opengis.net/kml/2.2">\
<NetworkLinkControl>\
</NetworkLinkControl>\
<Placemark>\
<name>bob</name>\
</Placemark>\
</kml>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.name).toBe("bob");
expect(entity.label).toBeDefined();
expect(entity.label.text.getValue()).toBe("bob");
});
});
it("Has entity collection with link to data source", function () {
const dataSource = new KmlDataSource(options);
const entityCollection = dataSource.entities;
expect(entityCollection.owner).toEqual(dataSource);
});
it("Has entity with link to entity collection", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<altitudeMode>relativeToGround</altitudeMode>\
<extrude>1</extrude>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entityCollection = dataSource.entities;
const entity = entityCollection.values[0];
expect(entity.entityCollection).toEqual(entityCollection);
});
});
it("GroundOverlay Icon with refreshMode shows warning", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay>\
<Icon>\
<href>http://test.invalid/image.png</href>\
<refreshMode>onInterval</refreshMode>\
</Icon>\
</GroundOverlay>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - Unsupported Icon refreshMode: onInterval",
);
});
});
it("GroundOverly Icon with viewRefreshMode shows warning", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<GroundOverlay>\
<Icon>\
<href>http://test.invalid/image.png</href>\
<viewRefreshMode>onStop</viewRefreshMode>\
</Icon>\
</GroundOverlay>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - Unsupported Icon viewRefreshMode: onStop",
);
});
});
it("GroundOverly Icon with gx:x, gx:y, gx:w, gx:h shows warning", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<GroundOverlay>\
<Icon>\
<href>http://test.invalid/image.png</href>\
<gx:x>1</gx:x>\
</Icon>\
</GroundOverlay>\
</Document>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - gx:x, gx:y, gx:w, gx:h aren't supported for GroundOverlays",
);
});
});
it("LineStyle with gx extensions show warnings", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>\
<Style>\
<LineStyle>\
<gx:outerColor>dddddddd</gx:outerColor>\
<gx:outerWidth>0.0</gx:outerWidth>\
<gx:physicalWidth>0.0</gx:physicalWidth>\
<gx:labelVisibility>0</gx:labelVisibility>\
</LineStyle>\
</Style>\
<LineString>\
<coordinates>1,2,3 \
4,5,6 \
</coordinates>\
</LineString>\
</Placemark>\
</Document>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(4);
expect(console.warn.calls.argsFor(0)[0]).toBe(
"KML - gx:outerColor is not supported in a LineStyle",
);
expect(console.warn.calls.argsFor(1)[0]).toBe(
"KML - gx:outerWidth is not supported in a LineStyle",
);
expect(console.warn.calls.argsFor(2)[0]).toBe(
"KML - gx:physicalWidth is not supported in a LineStyle",
);
expect(console.warn.calls.argsFor(3)[0]).toBe(
"KML - gx:labelVisibility is not supported in a LineStyle",
);
});
});
it("Folder with radioFolder listItemType shows warning", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Folder>\
<Style>\
<ListStyle>\
<listItemType>radioFolder</listItemType>\
</ListStyle>\
</Style>\
</Folder>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - Unsupported ListStyle with listItemType: radioFolder",
);
});
});
it("StyleMap with highlighted key shows warning", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<StyleMap>\
<Pair>\
<key>normal</key>\
<Style>\
</Style>\
</Pair>\
<Pair>\
<key>highlighted</key>\
<Style>\
</Style>\
</Pair>\
</StyleMap>\
</Placemark>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - Unsupported StyleMap key: highlighted",
);
});
});
it("Linestrings with gx:drawOrder shows warning", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>\
<LineString>\
<gx:drawOrder>1</gx:drawOrder>\
<coordinates>1,2,3 \
4,5,6 \
</coordinates>\
</LineString>\
</Placemark>\
</Document>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - gx:drawOrder is not supported in LineStrings when clampToGround is false",
);
});
});
it("gx:Track with gx:angles shows warning)", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:Track>\
<when>2000-01-01T00:00:00Z</when>\
<gx:coord>1 2 3</gx:coord>\
<gx:angles>4 5 6</gx:angles>\
</gx:Track>\
</Placemark>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - gx:angles are not supported in gx:Tracks",
);
});
});
it("Model geometry shows warning)", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Model></Model>\
</Placemark>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - Unsupported geometry: Model",
);
});
});
it("ExtendedData with SchemaData or custom XML show warnings", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<LineString>\
<coordinates>1,2,3 \
4,5,6 \
</coordinates>\
</LineString>\
<ExtendedData xmlns:prefix="test">\
<SchemaData></SchemaData>\
</ExtendedData>\
</Placemark>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(2);
expect(console.warn.calls.argsFor(0)[0]).toBe(
"KML - SchemaData is unsupported",
);
expect(console.warn.calls.argsFor(1)[0]).toBe(
"KML - ExtendedData with xmlns:prefix is unsupported",
);
});
});
it("Parse Camera and LookAt on features", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<LineString>\
<coordinates>1,2,3 \
4,5,6 \
</coordinates>\
</LineString>\
<Camera></Camera>\
<LookAt>\
<longitude>-120</longitude>\
<latitude>40</latitude>\
<altitude>100</altitude>\
<heading>90</heading>\
<tilt>30</tilt>\
<range>1250</range>\
</LookAt>\
</Placemark>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
const placemark = dataSource.entities.values[0];
expect(placemark.kml.camera).toBeInstanceOf(KmlCamera);
expect(placemark.kml.lookAt).toBeInstanceOf(KmlLookAt);
expect(placemark.kml.lookAt.position).toEqual(
Cartesian3.fromDegrees(-120, 40, 100),
);
expect(placemark.kml.lookAt.headingPitchRange).toEqualEpsilon(
new HeadingPitchRange(
CesiumMath.toRadians(90),
CesiumMath.toRadians(30 - 90),
1250,
),
CesiumMath.EPSILON10,
);
});
});
it("Features with a Region shows warning", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<LineString>\
<coordinates>1,2,3 \
4,5,6 \
</coordinates>\
</LineString>\
<Region></Region>\
</Placemark>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - Placemark Regions are unsupported",
);
});
});
it("NetworkLink with a viewRefreshMode=onRegion shows warning", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/simple.kml</href>\
<viewRefreshMode>onRegion</viewRefreshMode>\
</Link>\
</NetworkLink>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(1);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - Unsupported viewRefreshMode: onRegion",
);
});
});
it("Tour: reads gx:Tour)", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:Tour id="id_123">\
<name>Tour 1</name>\
<gx:Playlist>\
<gx:Wait>\
<gx:duration>2</gx:duration>\
</gx:Wait>\
<gx:FlyTo>\
<gx:duration>3</gx:duration>\
</gx:FlyTo>\
</gx:Playlist>\
</gx:Tour>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.kmlTours.length).toEqual(1);
const tour = dataSource.kmlTours[0];
expect(tour).toBeInstanceOf(KmlTour);
expect(tour.name).toEqual("Tour 1");
expect(tour.id).toEqual("id_123");
expect(tour.playlist.length).toEqual(2);
expect(tour.playlist[0].duration).toEqual(2);
expect(tour.playlist[0]).toBeInstanceOf(KmlTourWait);
expect(tour.playlist[1].duration).toEqual(3);
expect(tour.playlist[1]).toBeInstanceOf(KmlTourFlyTo);
});
});
it("Tour: reads LookAt and Camera", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:Tour>\
<gx:Playlist>\
<gx:FlyTo>\
<gx:duration>5</gx:duration>\
<gx:flyToMode>bounce</gx:flyToMode>\
<LookAt>\
<longitude>10</longitude>\
<latitude>20</latitude>\
<altitude>30</altitude>\
<range>40</range>\
<tilt>50</tilt>\
<heading>60</heading>\
</LookAt>\
</gx:FlyTo>\
<gx:FlyTo>\
<gx:duration>4.1</gx:duration>\
<Camera>\
<longitude>170.0</longitude>\
<latitude>-43.0</latitude>\
<altitude>9700</altitude>\
<heading>-10.0</heading>\
<tilt>33.5</tilt>\
<roll>20</roll>\
</Camera>\
</gx:FlyTo>\
</gx:Playlist>\
</gx:Tour>\
</Document>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.kmlTours.length).toEqual(1);
const tour = dataSource.kmlTours[0];
expect(tour.playlist.length).toEqual(2);
const flyto1 = tour.playlist[0];
const flyto2 = tour.playlist[1];
expect(flyto1.flyToMode).toEqual("bounce");
expect(flyto1.duration).toEqual(5);
expect(flyto1.view).toBeInstanceOf(KmlLookAt);
expect(flyto2.duration).toEqual(4.1);
expect(flyto2.view).toBeInstanceOf(KmlCamera);
expect(flyto1.view.position).toBeInstanceOf(Cartesian3);
expect(flyto2.view.position).toBeInstanceOf(Cartesian3);
expect(flyto1.view.headingPitchRange).toBeInstanceOf(HeadingPitchRange);
expect(flyto2.view.headingPitchRoll).toBeInstanceOf(HeadingPitchRoll);
});
});
it("Tour: log KML Tour unsupported nodes", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<gx:Tour>\
<gx:Playlist>\
<gx:AnimatedUpdate>\
</gx:AnimatedUpdate>\
<gx:TourControl>\
</gx:TourControl>\
<gx:SoundCue>\
</gx:SoundCue>\
</gx:Playlist>\
</gx:Tour>\
</Document>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.kmlTours.length).toEqual(1);
expect(dataSource.kmlTours[0].playlist.length).toEqual(0);
expect(console.warn).toHaveBeenCalledWith(
"KML Tour unsupported node AnimatedUpdate",
);
expect(console.warn).toHaveBeenCalledWith(
"KML Tour unsupported node TourControl",
);
expect(console.warn).toHaveBeenCalledWith(
"KML Tour unsupported node SoundCue",
);
});
});
it("NetworkLink: onExpire without an expires shows warning", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/simple.kml</href>\
<refreshMode>onExpire</refreshMode>\
</Link>\
</NetworkLink>';
spyOn(console, "warn").and.callThrough();
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(2);
expect(console.warn.calls.count()).toEqual(1);
expect(console.warn).toHaveBeenCalledWith(
"KML - refreshMode of onExpire requires the NetworkLinkControl to have an expires element",
);
});
});
it("NetworkLink: Heading and pitch can be undefined if the camera is in morphing mode", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<NetworkLink id="link">\
<Link>\
<href>./Data/KML/simple.kml</href>\
<refreshMode>onExpire</refreshMode>\
</Link>\
</NetworkLink>';
const camera = createCamera({
offset: Cartesian3.fromDegrees(-110, 30, 1000),
});
Camera.clone(uberCamera, camera);
const kmlOptions = {
camera: camera,
canvas: uberCanvas,
};
camera._mode = SceneMode.MORPHING;
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
kmlOptions,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(2);
});
});
it("when clampToGround is false, height isn't set if the polygon is extrudable", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<altitudeMode>relativeToGround</altitudeMode>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.perPositionHeight.getValue()).toEqual(true);
expect(entity.polygon.height).toBeUndefined();
});
});
it("when clampToGround is false, height is set to 0 if polygon isn't extrudable", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Polygon>\
<altitudeMode>clampToGround</altitudeMode>\
</Polygon>\
</Placemark>';
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polygon.perPositionHeight).toBeUndefined();
expect(entity.polygon.height.getValue()).toEqual(0);
});
});
it("when a LineString is clamped to ground and tesselated, entity has a polyline geometry and ColorProperty", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Placemark>\
<Style>\
<LineStyle>\
<color>FFFF0000</color>\
</LineStyle>\
</Style>\
<LineString>\
<altitudeMode>clampToGround</altitudeMode>\
<tessellate>true</tessellate>\
<coordinates>1,2,3\
4,5,6\
</coordinates>\
</LineString>\
</Placemark>';
const clampToGroundOptions = combine(options, { clampToGround: true });
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
clampToGroundOptions,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polyline).toBeDefined();
expect(entity.polyline.clampToGround.getValue()).toEqual(true);
expect(entity.polyline.material).toBeInstanceOf(ColorMaterialProperty);
});
});
it("when a LineString is clamped to ground and tesselated with z draworder, entity has a polyline geometry and ColorProperty", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\
<Document xmlns="http://www.opengis.net/kml/2.2"\
xmlns:gx="http://www.google.com/kml/ext/2.2">\
<Placemark>\
<Style>\
<LineStyle>\
<color>FFFF0000</color>\
</LineStyle>\
</Style>\
<LineString>\
<altitudeMode>clampToGround</altitudeMode>\
<tessellate>true</tessellate>\
<coordinates>1,2,3\
4,5,6\
</coordinates>\
<gx:drawOrder>2</gx:drawOrder>\
</LineString>\
</Placemark>\
</Document>';
const clampToGroundOptions = combine(options, { clampToGround: true });
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
clampToGroundOptions,
).then(function (dataSource) {
const entity = dataSource.entities.values[0];
expect(entity.polyline).toBeDefined();
expect(entity.polyline.clampToGround.getValue()).toEqual(true);
expect(entity.polyline.zIndex.getValue()).toBe(2);
});
});
it("correctly uses random colors", function () {
const kml =
'<?xml version="1.0" encoding="UTF-8"?>\n' +
"<Document>\n" +
'<Style id="color2">\n' +
"<PolyStyle>\n" +
"<color>00000000</color>\n" +
"<colorMode>random</colorMode>\n" +
"</PolyStyle>\n" +
"</Style>\n" +
'<Style id="color3">\n' +
"<PolyStyle>\n" +
"<color>9991ccfa</color>\n" +
"<colorMode>random</colorMode>\n" +
"</PolyStyle>\n" +
"</Style>\n" +
'<Style id="color4">\n' +
"<PolyStyle>\n" +
"<color>9991ccfa</color>\n" +
"<colorMode>random</colorMode>\n" +
"</PolyStyle>\n" +
"</Style>\n" +
"<Folder>\n" +
"<Placemark>\n" +
"<styleUrl>#color2</styleUrl>\n" +
"<Polygon>\n" +
"<outerBoundaryIs>\n" +
"<LinearRing>\n" +
"<coordinates>\n" +
"-89.52101149718367,44.23294548447373,0 -91.98408516538682,47.97512147591338,0 -96.92502046753899,45.42403256080313,0 -89.52101149718367,44.23294548447373,0 \n" +
"</coordinates>\n" +
"</LinearRing>\n" +
"</outerBoundaryIs>\n" +
"</Polygon>\n" +
"</Placemark>\n" +
"<Placemark>\n" +
"<styleUrl>#color3</styleUrl>\n" +
"<Polygon>\n" +
"<outerBoundaryIs>\n" +
"<LinearRing>\n" +
"<coordinates>\n" +
"-87.93623144974434,37.55775744070508,0 -89.46217805511263,41.26028180023913,0 -96.5546236391081,37.07597093222066,0 -87.93623144974434,37.55775744070508,0 \n" +
"</coordinates>\n" +
"</LinearRing>\n" +
"</outerBoundaryIs>\n" +
"</Polygon>\n" +
"</Placemark>\n" +
"<Placemark>\n" +
"<styleUrl>#color4</styleUrl>\n" +
"<Polygon>\n" +
"<outerBoundaryIs>\n" +
"<LinearRing>\n" +
"<coordinates>\n" +
"-104.347585776386,32.33288590150301,0 -103.8767557883591,37.6658714706182,0 -109.2704409486033,39.04706243328442,0 -104.347585776386,32.33288590150301,0 \n" +
"</coordinates>\n" +
"</LinearRing>\n" +
"</outerBoundaryIs>\n" +
"</Polygon>\n" +
"</Placemark>\n" +
"</Folder>\n" +
"</Document>\n";
CesiumMath.setRandomNumberSeed(0);
return KmlDataSource.load(
parser.parseFromString(kml, "text/xml"),
options,
).then(function (dataSource) {
expect(dataSource.entities.values.length).toEqual(4);
expect(
dataSource.entities.values[2].polygon.material.color.getValue(),
).not.toEqual(
dataSource.entities.values[3].polygon.material.color.getValue(),
);
});
});
});