diff --git a/Event.js b/Event.js
new file mode 100644
index 0000000..cf53ef9
--- /dev/null
+++ b/Event.js
@@ -0,0 +1,164 @@
+import Check from "./Check.js";
+import defined from "./defined.js";
+
+/**
+ * A generic utility class for managing subscribers for a particular event.
+ * This class is usually instantiated inside of a container class and
+ * exposed as a property for others to subscribe to.
+ *
+ * @alias Event
+ * @template Listener extends (...args: any[]) => void = (...args: any[]) => void
+ * @constructor
+ * @example
+ * MyObject.prototype.myListener = function(arg1, arg2) {
+ * this.myArg1Copy = arg1;
+ * this.myArg2Copy = arg2;
+ * }
+ *
+ * const myObjectInstance = new MyObject();
+ * const evt = new Cesium.Event();
+ * evt.addEventListener(MyObject.prototype.myListener, myObjectInstance);
+ * evt.raiseEvent('1', '2');
+ * evt.removeEventListener(MyObject.prototype.myListener);
+ */
+function Event() {
+ this._listeners = [];
+ this._scopes = [];
+ this._toRemove = [];
+ this._insideRaiseEvent = false;
+}
+
+Object.defineProperties(Event.prototype, {
+ /**
+ * The number of listeners currently subscribed to the event.
+ * @memberof Event.prototype
+ * @type {number}
+ * @readonly
+ */
+ numberOfListeners: {
+ get: function () {
+ return this._listeners.length - this._toRemove.length;
+ },
+ },
+});
+
+/**
+ * Registers a callback function to be executed whenever the event is raised.
+ * An optional scope can be provided to serve as the this pointer
+ * in which the function will execute.
+ *
+ * @param {Listener} listener The function to be executed when the event is raised.
+ * @param {object} [scope] An optional object scope to serve as the this
+ * pointer in which the listener function will execute.
+ * @returns {Event.RemoveCallback} A function that will remove this event listener when invoked.
+ *
+ * @see Event#raiseEvent
+ * @see Event#removeEventListener
+ */
+Event.prototype.addEventListener = function (listener, scope) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.func("listener", listener);
+ //>>includeEnd('debug');
+
+ this._listeners.push(listener);
+ this._scopes.push(scope);
+
+ const event = this;
+ return function () {
+ event.removeEventListener(listener, scope);
+ };
+};
+
+/**
+ * Unregisters a previously registered callback.
+ *
+ * @param {Listener} listener The function to be unregistered.
+ * @param {object} [scope] The scope that was originally passed to addEventListener.
+ * @returns {boolean} true if the listener was removed; false if the listener and scope are not registered with the event.
+ *
+ * @see Event#addEventListener
+ * @see Event#raiseEvent
+ */
+Event.prototype.removeEventListener = function (listener, scope) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.func("listener", listener);
+ //>>includeEnd('debug');
+
+ const listeners = this._listeners;
+ const scopes = this._scopes;
+
+ let index = -1;
+ for (let i = 0; i < listeners.length; i++) {
+ if (listeners[i] === listener && scopes[i] === scope) {
+ index = i;
+ break;
+ }
+ }
+
+ if (index !== -1) {
+ if (this._insideRaiseEvent) {
+ //In order to allow removing an event subscription from within
+ //a callback, we don't actually remove the items here. Instead
+ //remember the index they are at and undefined their value.
+ this._toRemove.push(index);
+ listeners[index] = undefined;
+ scopes[index] = undefined;
+ } else {
+ listeners.splice(index, 1);
+ scopes.splice(index, 1);
+ }
+ return true;
+ }
+
+ return false;
+};
+
+function compareNumber(a, b) {
+ return b - a;
+}
+
+/**
+ * Raises the event by calling each registered listener with all supplied arguments.
+ *
+ * @param {...Parameters} arguments This method takes any number of parameters and passes them through to the listener functions.
+ *
+ * @see Event#addEventListener
+ * @see Event#removeEventListener
+ */
+Event.prototype.raiseEvent = function () {
+ this._insideRaiseEvent = true;
+
+ let i;
+ const listeners = this._listeners;
+ const scopes = this._scopes;
+ let length = listeners.length;
+
+ for (i = 0; i < length; i++) {
+ const listener = listeners[i];
+ if (defined(listener)) {
+ listeners[i].apply(scopes[i], arguments);
+ }
+ }
+
+ //Actually remove items removed in removeEventListener.
+ const toRemove = this._toRemove;
+ length = toRemove.length;
+ if (length > 0) {
+ toRemove.sort(compareNumber);
+ for (i = 0; i < length; i++) {
+ const index = toRemove[i];
+ listeners.splice(index, 1);
+ scopes.splice(index, 1);
+ }
+ toRemove.length = 0;
+ }
+
+ this._insideRaiseEvent = false;
+};
+
+/**
+ * A function that removes a listener.
+ * @callback Event.RemoveCallback
+ */
+
+export default Event;
diff --git a/Resource.js b/Resource.js
new file mode 100644
index 0000000..cec4680
--- /dev/null
+++ b/Resource.js
@@ -0,0 +1,2275 @@
+import Uri from "urijs";
+import appendForwardSlash from "./appendForwardSlash.js";
+import Check from "./Check.js";
+import clone from "./clone.js";
+import combine from "./combine.js";
+import defaultValue from "./defaultValue.js";
+import defer from "./defer.js";
+import defined from "./defined.js";
+import DeveloperError from "./DeveloperError.js";
+import getAbsoluteUri from "./getAbsoluteUri.js";
+import getBaseUri from "./getBaseUri.js";
+import getExtensionFromUri from "./getExtensionFromUri.js";
+import getImagePixels from "./getImagePixels.js";
+import isBlobUri from "./isBlobUri.js";
+import isCrossOriginUrl from "./isCrossOriginUrl.js";
+import isDataUri from "./isDataUri.js";
+import loadAndExecuteScript from "./loadAndExecuteScript.js";
+import CesiumMath from "./Math.js";
+import objectToQuery from "./objectToQuery.js";
+import queryToObject from "./queryToObject.js";
+import Request from "./Request.js";
+import RequestErrorEvent from "./RequestErrorEvent.js";
+import RequestScheduler from "./RequestScheduler.js";
+import RequestState from "./RequestState.js";
+import RuntimeError from "./RuntimeError.js";
+import TrustedServers from "./TrustedServers.js";
+
+const xhrBlobSupported = (function () {
+ try {
+ const xhr = new XMLHttpRequest();
+ xhr.open("GET", "#", true);
+ xhr.responseType = "blob";
+ return xhr.responseType === "blob";
+ } catch (e) {
+ return false;
+ }
+})();
+
+/**
+ * @typedef {object} Resource.ConstructorOptions
+ *
+ * Initialization options for the Resource constructor
+ *
+ * @property {string} url The url of the resource.
+ * @property {object} [queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @property {object} [templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @property {object} [headers={}] Additional HTTP headers that will be sent.
+ * @property {Proxy} [proxy] A proxy to be used when loading the resource.
+ * @property {Resource.RetryCallback} [retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @property {number} [retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @property {Request} [request] A Request object that will be used. Intended for internal use only.
+ * @property {boolean} [parseUrl=true] If true, parse the url for query parameters; otherwise store the url without change
+ */
+
+/**
+ * A resource that includes the location and any other parameters we need to retrieve it or create derived resources. It also provides the ability to retry requests.
+ *
+ * @alias Resource
+ * @constructor
+ *
+ * @param {string|Resource.ConstructorOptions} options A url or an object describing initialization options
+ *
+ * @example
+ * function refreshTokenRetryCallback(resource, error) {
+ * if (error.statusCode === 403) {
+ * // 403 status code means a new token should be generated
+ * return getNewAccessToken()
+ * .then(function(token) {
+ * resource.queryParameters.access_token = token;
+ * return true;
+ * })
+ * .catch(function() {
+ * return false;
+ * });
+ * }
+ *
+ * return false;
+ * }
+ *
+ * const resource = new Resource({
+ * url: 'http://server.com/path/to/resource.json',
+ * proxy: new DefaultProxy('/proxy/'),
+ * headers: {
+ * 'X-My-Header': 'valueOfHeader'
+ * },
+ * queryParameters: {
+ * 'access_token': '123-435-456-000'
+ * },
+ * retryCallback: refreshTokenRetryCallback,
+ * retryAttempts: 1
+ * });
+ */
+function Resource(options) {
+ options = defaultValue(options, defaultValue.EMPTY_OBJECT);
+ if (typeof options === "string") {
+ options = {
+ url: options,
+ };
+ }
+
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.string("options.url", options.url);
+ //>>includeEnd('debug');
+
+ this._url = undefined;
+ this._templateValues = defaultClone(options.templateValues, {});
+ this._queryParameters = defaultClone(options.queryParameters, {});
+
+ /**
+ * Additional HTTP headers that will be sent with the request.
+ *
+ * @type {object}
+ */
+ this.headers = defaultClone(options.headers, {});
+
+ /**
+ * A Request object that will be used. Intended for internal use only.
+ *
+ * @type {Request}
+ */
+ this.request = defaultValue(options.request, new Request());
+
+ /**
+ * A proxy to be used when loading the resource.
+ *
+ * @type {Proxy}
+ */
+ this.proxy = options.proxy;
+
+ /**
+ * Function to call when a request for this resource fails. If it returns true or a Promise that resolves to true, the request will be retried.
+ *
+ * @type {Function}
+ */
+ this.retryCallback = options.retryCallback;
+
+ /**
+ * The number of times the retryCallback should be called before giving up.
+ *
+ * @type {number}
+ */
+ this.retryAttempts = defaultValue(options.retryAttempts, 0);
+ this._retryCount = 0;
+
+ const parseUrl = defaultValue(options.parseUrl, true);
+ if (parseUrl) {
+ this.parseUrl(options.url, true, true);
+ } else {
+ this._url = options.url;
+ }
+
+ this._credits = options.credits;
+}
+
+/**
+ * Clones a value if it is defined, otherwise returns the default value
+ *
+ * @param {object} [value] The value to clone.
+ * @param {object} [defaultValue] The default value.
+ *
+ * @returns {object} A clone of value or the defaultValue.
+ *
+ * @private
+ */
+function defaultClone(value, defaultValue) {
+ return defined(value) ? clone(value) : defaultValue;
+}
+
+/**
+ * A helper function to create a resource depending on whether we have a String or a Resource
+ *
+ * @param {Resource|string} resource A Resource or a String to use when creating a new Resource.
+ *
+ * @returns {Resource} If resource is a String, a Resource constructed with the url and options. Otherwise the resource parameter is returned.
+ *
+ * @private
+ */
+Resource.createIfNeeded = function (resource) {
+ if (resource instanceof Resource) {
+ // Keep existing request object. This function is used internally to duplicate a Resource, so that it can't
+ // be modified outside of a class that holds it (eg. an imagery or terrain provider). Since the Request objects
+ // are managed outside of the providers, by the tile loading code, we want to keep the request property the same so if it is changed
+ // in the underlying tiling code the requests for this resource will use it.
+ return resource.getDerivedResource({
+ request: resource.request,
+ });
+ }
+
+ if (typeof resource !== "string") {
+ return resource;
+ }
+
+ return new Resource({
+ url: resource,
+ });
+};
+
+let supportsImageBitmapOptionsPromise;
+/**
+ * A helper function to check whether createImageBitmap supports passing ImageBitmapOptions.
+ *
+ * @returns {Promise} A promise that resolves to true if this browser supports creating an ImageBitmap with options.
+ *
+ * @private
+ */
+Resource.supportsImageBitmapOptions = function () {
+ // Until the HTML folks figure out what to do about this, we need to actually try loading an image to
+ // know if this browser supports passing options to the createImageBitmap function.
+ // https://github.com/whatwg/html/pull/4248
+ //
+ // We also need to check whether the colorSpaceConversion option is supported.
+ // We do this by loading a PNG with an embedded color profile, first with
+ // colorSpaceConversion: "none" and then with colorSpaceConversion: "default".
+ // If the pixel color is different then we know the option is working.
+ // As of Webkit 17612.3.6.1.6 the createImageBitmap promise resolves but the
+ // option is not actually supported.
+ if (defined(supportsImageBitmapOptionsPromise)) {
+ return supportsImageBitmapOptionsPromise;
+ }
+
+ if (typeof createImageBitmap !== "function") {
+ supportsImageBitmapOptionsPromise = Promise.resolve(false);
+ return supportsImageBitmapOptionsPromise;
+ }
+
+ const imageDataUri =
+ "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAIAAACQd1PeAAAABGdBTUEAAE4g3rEiDgAAACBjSFJNAAB6JgAAgIQAAPoAAACA6AAAdTAAAOpgAAA6mAAAF3CculE8AAAADElEQVQI12Ng6GAAAAEUAIngE3ZiAAAAAElFTkSuQmCC";
+
+ supportsImageBitmapOptionsPromise = Resource.fetchBlob({
+ url: imageDataUri,
+ })
+ .then(function (blob) {
+ const imageBitmapOptions = {
+ imageOrientation: "flipY", // default is "none"
+ premultiplyAlpha: "none", // default is "default"
+ colorSpaceConversion: "none", // default is "default"
+ };
+ return Promise.all([
+ createImageBitmap(blob, imageBitmapOptions),
+ createImageBitmap(blob),
+ ]);
+ })
+ .then(function (imageBitmaps) {
+ // Check whether the colorSpaceConversion option had any effect on the green channel
+ const colorWithOptions = getImagePixels(imageBitmaps[0]);
+ const colorWithDefaults = getImagePixels(imageBitmaps[1]);
+ return colorWithOptions[1] !== colorWithDefaults[1];
+ })
+ .catch(function () {
+ return false;
+ });
+
+ return supportsImageBitmapOptionsPromise;
+};
+
+Object.defineProperties(Resource, {
+ /**
+ * Returns true if blobs are supported.
+ *
+ * @memberof Resource
+ * @type {boolean}
+ *
+ * @readonly
+ */
+ isBlobSupported: {
+ get: function () {
+ return xhrBlobSupported;
+ },
+ },
+});
+
+Object.defineProperties(Resource.prototype, {
+ /**
+ * Query parameters appended to the url.
+ *
+ * @memberof Resource.prototype
+ * @type {object}
+ *
+ * @readonly
+ */
+ queryParameters: {
+ get: function () {
+ return this._queryParameters;
+ },
+ },
+
+ /**
+ * The key/value pairs used to replace template parameters in the url.
+ *
+ * @memberof Resource.prototype
+ * @type {object}
+ *
+ * @readonly
+ */
+ templateValues: {
+ get: function () {
+ return this._templateValues;
+ },
+ },
+
+ /**
+ * The url to the resource with template values replaced, query string appended and encoded by proxy if one was set.
+ *
+ * @memberof Resource.prototype
+ * @type {string}
+ */
+ url: {
+ get: function () {
+ return this.getUrlComponent(true, true);
+ },
+ set: function (value) {
+ this.parseUrl(value, false, false);
+ },
+ },
+
+ /**
+ * The file extension of the resource.
+ *
+ * @memberof Resource.prototype
+ * @type {string}
+ *
+ * @readonly
+ */
+ extension: {
+ get: function () {
+ return getExtensionFromUri(this._url);
+ },
+ },
+
+ /**
+ * True if the Resource refers to a data URI.
+ *
+ * @memberof Resource.prototype
+ * @type {boolean}
+ */
+ isDataUri: {
+ get: function () {
+ return isDataUri(this._url);
+ },
+ },
+
+ /**
+ * True if the Resource refers to a blob URI.
+ *
+ * @memberof Resource.prototype
+ * @type {boolean}
+ */
+ isBlobUri: {
+ get: function () {
+ return isBlobUri(this._url);
+ },
+ },
+
+ /**
+ * True if the Resource refers to a cross origin URL.
+ *
+ * @memberof Resource.prototype
+ * @type {boolean}
+ */
+ isCrossOriginUrl: {
+ get: function () {
+ return isCrossOriginUrl(this._url);
+ },
+ },
+
+ /**
+ * True if the Resource has request headers. This is equivalent to checking if the headers property has any keys.
+ *
+ * @memberof Resource.prototype
+ * @type {boolean}
+ */
+ hasHeaders: {
+ get: function () {
+ return Object.keys(this.headers).length > 0;
+ },
+ },
+
+ /**
+ * Gets the credits required for attribution of an asset.
+ * @private
+ */
+ credits: {
+ get: function () {
+ return this._credits;
+ },
+ },
+});
+
+/**
+ * Override Object#toString so that implicit string conversion gives the
+ * complete URL represented by this Resource.
+ *
+ * @returns {string} The URL represented by this Resource
+ */
+Resource.prototype.toString = function () {
+ return this.getUrlComponent(true, true);
+};
+
+/**
+ * Parse a url string, and store its info
+ *
+ * @param {string} url The input url string.
+ * @param {boolean} merge If true, we'll merge with the resource's existing queryParameters. Otherwise they will be replaced.
+ * @param {boolean} preserveQuery If true duplicate parameters will be concatenated into an array. If false, keys in url will take precedence.
+ * @param {string} [baseUrl] If supplied, and input url is a relative url, it will be made absolute relative to baseUrl
+ *
+ * @private
+ */
+Resource.prototype.parseUrl = function (url, merge, preserveQuery, baseUrl) {
+ let uri = new Uri(url);
+ const query = parseQueryString(uri.query());
+
+ this._queryParameters = merge
+ ? combineQueryParameters(query, this.queryParameters, preserveQuery)
+ : query;
+
+ // Remove unneeded info from the Uri
+ uri.search("");
+ uri.fragment("");
+
+ if (defined(baseUrl) && uri.scheme() === "") {
+ uri = uri.absoluteTo(getAbsoluteUri(baseUrl));
+ }
+
+ this._url = uri.toString();
+};
+
+/**
+ * Parses a query string and returns the object equivalent.
+ *
+ * @param {string} queryString The query string
+ * @returns {object}
+ *
+ * @private
+ */
+function parseQueryString(queryString) {
+ if (queryString.length === 0) {
+ return {};
+ }
+
+ // Special case where the querystring is just a string, not key/value pairs
+ if (queryString.indexOf("=") === -1) {
+ return { [queryString]: undefined };
+ }
+
+ return queryToObject(queryString);
+}
+
+/**
+ * This combines a map of query parameters.
+ *
+ * @param {object} q1 The first map of query parameters. Values in this map will take precedence if preserveQueryParameters is false.
+ * @param {object} q2 The second map of query parameters.
+ * @param {boolean} preserveQueryParameters If true duplicate parameters will be concatenated into an array. If false, keys in q1 will take precedence.
+ *
+ * @returns {object} The combined map of query parameters.
+ *
+ * @example
+ * const q1 = {
+ * a: 1,
+ * b: 2
+ * };
+ * const q2 = {
+ * a: 3,
+ * c: 4
+ * };
+ * const q3 = {
+ * b: [5, 6],
+ * d: 7
+ * }
+ *
+ * // Returns
+ * // {
+ * // a: [1, 3],
+ * // b: 2,
+ * // c: 4
+ * // };
+ * combineQueryParameters(q1, q2, true);
+ *
+ * // Returns
+ * // {
+ * // a: 1,
+ * // b: 2,
+ * // c: 4
+ * // };
+ * combineQueryParameters(q1, q2, false);
+ *
+ * // Returns
+ * // {
+ * // a: 1,
+ * // b: [2, 5, 6],
+ * // d: 7
+ * // };
+ * combineQueryParameters(q1, q3, true);
+ *
+ * // Returns
+ * // {
+ * // a: 1,
+ * // b: 2,
+ * // d: 7
+ * // };
+ * combineQueryParameters(q1, q3, false);
+ *
+ * @private
+ */
+function combineQueryParameters(q1, q2, preserveQueryParameters) {
+ if (!preserveQueryParameters) {
+ return combine(q1, q2);
+ }
+
+ const result = clone(q1, true);
+ for (const param in q2) {
+ if (q2.hasOwnProperty(param)) {
+ let value = result[param];
+ const q2Value = q2[param];
+ if (defined(value)) {
+ if (!Array.isArray(value)) {
+ value = result[param] = [value];
+ }
+
+ result[param] = value.concat(q2Value);
+ } else {
+ result[param] = Array.isArray(q2Value) ? q2Value.slice() : q2Value;
+ }
+ }
+ }
+
+ return result;
+}
+
+/**
+ * Returns the url, optional with the query string and processed by a proxy.
+ *
+ * @param {boolean} [query=false] If true, the query string is included.
+ * @param {boolean} [proxy=false] If true, the url is processed by the proxy object, if defined.
+ *
+ * @returns {string} The url with all the requested components.
+ */
+Resource.prototype.getUrlComponent = function (query, proxy) {
+ if (this.isDataUri) {
+ return this._url;
+ }
+
+ let url = this._url;
+ if (query) {
+ url = `${url}${stringifyQuery(this.queryParameters)}`;
+ }
+
+ // Restore the placeholders, which may have been escaped in objectToQuery or elsewhere
+ url = url.replace(/%7B/g, "{").replace(/%7D/g, "}");
+
+ const templateValues = this._templateValues;
+ if (Object.keys(templateValues).length > 0) {
+ url = url.replace(/{(.*?)}/g, function (match, key) {
+ const replacement = templateValues[key];
+ if (defined(replacement)) {
+ // use the replacement value from templateValues if there is one...
+ return encodeURIComponent(replacement);
+ }
+ // otherwise leave it unchanged
+ return match;
+ });
+ }
+
+ if (proxy && defined(this.proxy)) {
+ url = this.proxy.getURL(url);
+ }
+
+ return url;
+};
+
+/**
+ * Converts a query object into a string.
+ *
+ * @param {object} queryObject The object with query parameters
+ * @returns {string}
+ *
+ * @private
+ */
+function stringifyQuery(queryObject) {
+ const keys = Object.keys(queryObject);
+
+ if (keys.length === 0) {
+ return "";
+ }
+ if (keys.length === 1 && !defined(queryObject[keys[0]])) {
+ // We have 1 key with an undefined value, so this is just a string, not key/value pairs
+ return `?${keys[0]}`;
+ }
+
+ return `?${objectToQuery(queryObject)}`;
+}
+
+/**
+ * Combines the specified object and the existing query parameters. This allows you to add many parameters at once,
+ * as opposed to adding them one at a time to the queryParameters property. If a value is already set, it will be replaced with the new value.
+ *
+ * @param {object} params The query parameters
+ * @param {boolean} [useAsDefault=false] If true the params will be used as the default values, so they will only be set if they are undefined.
+ */
+Resource.prototype.setQueryParameters = function (params, useAsDefault) {
+ if (useAsDefault) {
+ this._queryParameters = combineQueryParameters(
+ this._queryParameters,
+ params,
+ false
+ );
+ } else {
+ this._queryParameters = combineQueryParameters(
+ params,
+ this._queryParameters,
+ false
+ );
+ }
+};
+
+/**
+ * Combines the specified object and the existing query parameters. This allows you to add many parameters at once,
+ * as opposed to adding them one at a time to the queryParameters property.
+ *
+ * @param {object} params The query parameters
+ */
+Resource.prototype.appendQueryParameters = function (params) {
+ this._queryParameters = combineQueryParameters(
+ params,
+ this._queryParameters,
+ true
+ );
+};
+
+/**
+ * Combines the specified object and the existing template values. This allows you to add many values at once,
+ * as opposed to adding them one at a time to the templateValues property. If a value is already set, it will become an array and the new value will be appended.
+ *
+ * @param {object} template The template values
+ * @param {boolean} [useAsDefault=false] If true the values will be used as the default values, so they will only be set if they are undefined.
+ */
+Resource.prototype.setTemplateValues = function (template, useAsDefault) {
+ if (useAsDefault) {
+ this._templateValues = combine(this._templateValues, template);
+ } else {
+ this._templateValues = combine(template, this._templateValues);
+ }
+};
+
+/**
+ * Returns a resource relative to the current instance. All properties remain the same as the current instance unless overridden in options.
+ *
+ * @param {object} options An object with the following properties
+ * @param {string} [options.url] The url that will be resolved relative to the url of the current instance.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be combined with those of the current instance.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}). These will be combined with those of the current instance.
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The function to call when loading the resource fails.
+ * @param {number} [options.retryAttempts] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @param {boolean} [options.preserveQueryParameters=false] If true, this will keep all query parameters from the current resource and derived resource. If false, derived parameters will replace those of the current resource.
+ *
+ * @returns {Resource} The resource derived from the current one.
+ */
+Resource.prototype.getDerivedResource = function (options) {
+ const resource = this.clone();
+ resource._retryCount = 0;
+
+ if (defined(options.url)) {
+ const preserveQuery = defaultValue(options.preserveQueryParameters, false);
+ resource.parseUrl(options.url, true, preserveQuery, this._url);
+ }
+
+ if (defined(options.queryParameters)) {
+ resource._queryParameters = combine(
+ options.queryParameters,
+ resource.queryParameters
+ );
+ }
+ if (defined(options.templateValues)) {
+ resource._templateValues = combine(
+ options.templateValues,
+ resource.templateValues
+ );
+ }
+ if (defined(options.headers)) {
+ resource.headers = combine(options.headers, resource.headers);
+ }
+ if (defined(options.proxy)) {
+ resource.proxy = options.proxy;
+ }
+ if (defined(options.request)) {
+ resource.request = options.request;
+ }
+ if (defined(options.retryCallback)) {
+ resource.retryCallback = options.retryCallback;
+ }
+ if (defined(options.retryAttempts)) {
+ resource.retryAttempts = options.retryAttempts;
+ }
+
+ return resource;
+};
+
+/**
+ * Called when a resource fails to load. This will call the retryCallback function if defined until retryAttempts is reached.
+ *
+ * @param {RequestErrorEvent} [error] The error that was encountered.
+ *
+ * @returns {Promise} A promise to a boolean, that if true will cause the resource request to be retried.
+ *
+ * @private
+ */
+Resource.prototype.retryOnError = function (error) {
+ const retryCallback = this.retryCallback;
+ if (
+ typeof retryCallback !== "function" ||
+ this._retryCount >= this.retryAttempts
+ ) {
+ return Promise.resolve(false);
+ }
+
+ const that = this;
+ return Promise.resolve(retryCallback(this, error)).then(function (result) {
+ ++that._retryCount;
+
+ return result;
+ });
+};
+
+/**
+ * Duplicates a Resource instance.
+ *
+ * @param {Resource} [result] The object onto which to store the result.
+ *
+ * @returns {Resource} The modified result parameter or a new Resource instance if one was not provided.
+ */
+Resource.prototype.clone = function (result) {
+ if (!defined(result)) {
+ return new Resource({
+ url: this._url,
+ queryParameters: this.queryParameters,
+ templateValues: this.templateValues,
+ headers: this.headers,
+ proxy: this.proxy,
+ retryCallback: this.retryCallback,
+ retryAttempts: this.retryAttempts,
+ request: this.request.clone(),
+ parseUrl: false,
+ credits: defined(this.credits) ? this.credits.slice() : undefined,
+ });
+ }
+
+ result._url = this._url;
+ result._queryParameters = clone(this._queryParameters);
+ result._templateValues = clone(this._templateValues);
+ result.headers = clone(this.headers);
+ result.proxy = this.proxy;
+ result.retryCallback = this.retryCallback;
+ result.retryAttempts = this.retryAttempts;
+ result._retryCount = 0;
+ result.request = this.request.clone();
+
+ return result;
+};
+
+/**
+ * Returns the base path of the Resource.
+ *
+ * @param {boolean} [includeQuery = false] Whether or not to include the query string and fragment form the uri
+ *
+ * @returns {string} The base URI of the resource
+ */
+Resource.prototype.getBaseUri = function (includeQuery) {
+ return getBaseUri(this.getUrlComponent(includeQuery), includeQuery);
+};
+
+/**
+ * Appends a forward slash to the URL.
+ */
+Resource.prototype.appendForwardSlash = function () {
+ this._url = appendForwardSlash(this._url);
+};
+
+/**
+ * Asynchronously loads the resource as raw binary data. Returns a promise that will resolve to
+ * an ArrayBuffer once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled.
+ *
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ * @example
+ * // load a single URL asynchronously
+ * resource.fetchArrayBuffer().then(function(arrayBuffer) {
+ * // use the data
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.fetchArrayBuffer = function () {
+ return this.fetch({
+ responseType: "arraybuffer",
+ });
+};
+
+/**
+ * Creates a Resource and calls fetchArrayBuffer() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.fetchArrayBuffer = function (options) {
+ const resource = new Resource(options);
+ return resource.fetchArrayBuffer();
+};
+
+/**
+ * Asynchronously loads the given resource as a blob. Returns a promise that will resolve to
+ * a Blob once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled.
+ *
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ * @example
+ * // load a single URL asynchronously
+ * resource.fetchBlob().then(function(blob) {
+ * // use the data
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.fetchBlob = function () {
+ return this.fetch({
+ responseType: "blob",
+ });
+};
+
+/**
+ * Creates a Resource and calls fetchBlob() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.fetchBlob = function (options) {
+ const resource = new Resource(options);
+ return resource.fetchBlob();
+};
+
+/**
+ * Asynchronously loads the given image resource. Returns a promise that will resolve to
+ * an {@link https://developer.mozilla.org/en-US/docs/Web/API/ImageBitmap|ImageBitmap} if preferImageBitmap is true and the browser supports createImageBitmap or otherwise an
+ * {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement|Image} once loaded, or reject if the image failed to load.
+ *
+ * @param {object} [options] An object with the following properties.
+ * @param {boolean} [options.preferBlob=false] If true, we will load the image via a blob.
+ * @param {boolean} [options.preferImageBitmap=false] If true, image will be decoded during fetch and an ImageBitmap is returned.
+ * @param {boolean} [options.flipY=false] If true, image will be vertically flipped during decode. Only applies if the browser supports createImageBitmap.
+ * @param {boolean} [options.skipColorSpaceConversion=false] If true, any custom gamma or color profiles in the image will be ignored. Only applies if the browser supports createImageBitmap.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * // load a single image asynchronously
+ * resource.fetchImage().then(function(image) {
+ * // use the loaded image
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * // load several images in parallel
+ * Promise.all([resource1.fetchImage(), resource2.fetchImage()]).then(function(images) {
+ * // images is an array containing all the loaded images
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.fetchImage = function (options) {
+ options = defaultValue(options, defaultValue.EMPTY_OBJECT);
+ const preferImageBitmap = defaultValue(options.preferImageBitmap, false);
+ const preferBlob = defaultValue(options.preferBlob, false);
+ const flipY = defaultValue(options.flipY, false);
+ const skipColorSpaceConversion = defaultValue(
+ options.skipColorSpaceConversion,
+ false
+ );
+
+ checkAndResetRequest(this.request);
+ // We try to load the image normally if
+ // 1. Blobs aren't supported
+ // 2. It's a data URI
+ // 3. It's a blob URI
+ // 4. It doesn't have request headers and we preferBlob is false
+ if (
+ !xhrBlobSupported ||
+ this.isDataUri ||
+ this.isBlobUri ||
+ (!this.hasHeaders && !preferBlob)
+ ) {
+ return fetchImage({
+ resource: this,
+ flipY: flipY,
+ skipColorSpaceConversion: skipColorSpaceConversion,
+ preferImageBitmap: preferImageBitmap,
+ });
+ }
+
+ const blobPromise = this.fetchBlob();
+ if (!defined(blobPromise)) {
+ return;
+ }
+
+ let supportsImageBitmap;
+ let useImageBitmap;
+ let generatedBlobResource;
+ let generatedBlob;
+ return Resource.supportsImageBitmapOptions()
+ .then(function (result) {
+ supportsImageBitmap = result;
+ useImageBitmap = supportsImageBitmap && preferImageBitmap;
+ return blobPromise;
+ })
+ .then(function (blob) {
+ if (!defined(blob)) {
+ return;
+ }
+ generatedBlob = blob;
+ if (useImageBitmap) {
+ return Resource.createImageBitmapFromBlob(blob, {
+ flipY: flipY,
+ premultiplyAlpha: false,
+ skipColorSpaceConversion: skipColorSpaceConversion,
+ });
+ }
+ const blobUrl = window.URL.createObjectURL(blob);
+ generatedBlobResource = new Resource({
+ url: blobUrl,
+ });
+
+ return fetchImage({
+ resource: generatedBlobResource,
+ flipY: flipY,
+ skipColorSpaceConversion: skipColorSpaceConversion,
+ preferImageBitmap: false,
+ });
+ })
+ .then(function (image) {
+ if (!defined(image)) {
+ return;
+ }
+
+ // The blob object may be needed for use by a TileDiscardPolicy,
+ // so attach it to the image.
+ image.blob = generatedBlob;
+
+ if (useImageBitmap) {
+ return image;
+ }
+
+ window.URL.revokeObjectURL(generatedBlobResource.url);
+ return image;
+ })
+ .catch(function (error) {
+ if (defined(generatedBlobResource)) {
+ window.URL.revokeObjectURL(generatedBlobResource.url);
+ }
+
+ // If the blob load succeeded but the image decode failed, attach the blob
+ // to the error object for use by a TileDiscardPolicy.
+ // In particular, BingMapsImageryProvider uses this to detect the
+ // zero-length response that is returned when a tile is not available.
+ error.blob = generatedBlob;
+
+ return Promise.reject(error);
+ });
+};
+
+/**
+ * Fetches an image and returns a promise to it.
+ *
+ * @param {object} [options] An object with the following properties.
+ * @param {Resource} [options.resource] Resource object that points to an image to fetch.
+ * @param {boolean} [options.preferImageBitmap] If true, image will be decoded during fetch and an ImageBitmap is returned.
+ * @param {boolean} [options.flipY] If true, image will be vertically flipped during decode. Only applies if the browser supports createImageBitmap.
+ * @param {boolean} [options.skipColorSpaceConversion=false] If true, any custom gamma or color profiles in the image will be ignored. Only applies if the browser supports createImageBitmap.
+ * @private
+ */
+function fetchImage(options) {
+ const resource = options.resource;
+ const flipY = options.flipY;
+ const skipColorSpaceConversion = options.skipColorSpaceConversion;
+ const preferImageBitmap = options.preferImageBitmap;
+
+ const request = resource.request;
+ request.url = resource.url;
+ request.requestFunction = function () {
+ let crossOrigin = false;
+
+ // data URIs can't have crossorigin set.
+ if (!resource.isDataUri && !resource.isBlobUri) {
+ crossOrigin = resource.isCrossOriginUrl;
+ }
+
+ const deferred = defer();
+ Resource._Implementations.createImage(
+ request,
+ crossOrigin,
+ deferred,
+ flipY,
+ skipColorSpaceConversion,
+ preferImageBitmap
+ );
+
+ return deferred.promise;
+ };
+
+ const promise = RequestScheduler.request(request);
+ if (!defined(promise)) {
+ return;
+ }
+
+ return promise.catch(function (e) {
+ // Don't retry cancelled or otherwise aborted requests
+ if (request.state !== RequestState.FAILED) {
+ return Promise.reject(e);
+ }
+ return resource.retryOnError(e).then(function (retry) {
+ if (retry) {
+ // Reset request so it can try again
+ request.state = RequestState.UNISSUED;
+ request.deferred = undefined;
+
+ return fetchImage({
+ resource: resource,
+ flipY: flipY,
+ skipColorSpaceConversion: skipColorSpaceConversion,
+ preferImageBitmap: preferImageBitmap,
+ });
+ }
+ return Promise.reject(e);
+ });
+ });
+}
+
+/**
+ * Creates a Resource and calls fetchImage() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {boolean} [options.flipY=false] Whether to vertically flip the image during fetch and decode. Only applies when requesting an image and the browser supports createImageBitmap.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @param {boolean} [options.preferBlob=false] If true, we will load the image via a blob.
+ * @param {boolean} [options.preferImageBitmap=false] If true, image will be decoded during fetch and an ImageBitmap is returned.
+ * @param {boolean} [options.skipColorSpaceConversion=false] If true, any custom gamma or color profiles in the image will be ignored. Only applies when requesting an image and the browser supports createImageBitmap.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.fetchImage = function (options) {
+ const resource = new Resource(options);
+ return resource.fetchImage({
+ flipY: options.flipY,
+ skipColorSpaceConversion: options.skipColorSpaceConversion,
+ preferBlob: options.preferBlob,
+ preferImageBitmap: options.preferImageBitmap,
+ });
+};
+
+/**
+ * Asynchronously loads the given resource as text. Returns a promise that will resolve to
+ * a String once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled.
+ *
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ * @example
+ * // load text from a URL, setting a custom header
+ * const resource = new Resource({
+ * url: 'http://someUrl.com/someJson.txt',
+ * headers: {
+ * 'X-Custom-Header' : 'some value'
+ * }
+ * });
+ * resource.fetchText().then(function(text) {
+ * // Do something with the text
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest|XMLHttpRequest}
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.fetchText = function () {
+ return this.fetch({
+ responseType: "text",
+ });
+};
+
+/**
+ * Creates a Resource and calls fetchText() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.fetchText = function (options) {
+ const resource = new Resource(options);
+ return resource.fetchText();
+};
+
+// note: */* below is */* but that ends the comment block early
+/**
+ * Asynchronously loads the given resource as JSON. Returns a promise that will resolve to
+ * a JSON object once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled. This function
+ * adds 'Accept: application/json,*/*;q=0.01' to the request headers, if not
+ * already specified.
+ *
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * resource.fetchJson().then(function(jsonData) {
+ * // Do something with the JSON object
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.fetchJson = function () {
+ const promise = this.fetch({
+ responseType: "text",
+ headers: {
+ Accept: "application/json,*/*;q=0.01",
+ },
+ });
+
+ if (!defined(promise)) {
+ return undefined;
+ }
+
+ return promise.then(function (value) {
+ if (!defined(value)) {
+ return;
+ }
+ return JSON.parse(value);
+ });
+};
+
+/**
+ * Creates a Resource and calls fetchJson() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.fetchJson = function (options) {
+ const resource = new Resource(options);
+ return resource.fetchJson();
+};
+
+/**
+ * Asynchronously loads the given resource as XML. Returns a promise that will resolve to
+ * an XML Document once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled.
+ *
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * // load XML from a URL, setting a custom header
+ * Cesium.loadXML('http://someUrl.com/someXML.xml', {
+ * 'X-Custom-Header' : 'some value'
+ * }).then(function(document) {
+ * // Do something with the document
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest|XMLHttpRequest}
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.fetchXML = function () {
+ return this.fetch({
+ responseType: "document",
+ overrideMimeType: "text/xml",
+ });
+};
+
+/**
+ * Creates a Resource and calls fetchXML() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.fetchXML = function (options) {
+ const resource = new Resource(options);
+ return resource.fetchXML();
+};
+
+/**
+ * Requests a resource using JSONP.
+ *
+ * @param {string} [callbackParameterName='callback'] The callback parameter name that the server expects.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * // load a data asynchronously
+ * resource.fetchJsonp().then(function(data) {
+ * // use the loaded data
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.fetchJsonp = function (callbackParameterName) {
+ callbackParameterName = defaultValue(callbackParameterName, "callback");
+
+ checkAndResetRequest(this.request);
+
+ //generate a unique function name
+ let functionName;
+ do {
+ functionName = `loadJsonp${CesiumMath.nextRandomNumber()
+ .toString()
+ .substring(2, 8)}`;
+ } while (defined(window[functionName]));
+
+ return fetchJsonp(this, callbackParameterName, functionName);
+};
+
+function fetchJsonp(resource, callbackParameterName, functionName) {
+ const callbackQuery = {};
+ callbackQuery[callbackParameterName] = functionName;
+ resource.setQueryParameters(callbackQuery);
+
+ const request = resource.request;
+ const url = resource.url;
+ request.url = url;
+ request.requestFunction = function () {
+ const deferred = defer();
+
+ //assign a function with that name in the global scope
+ window[functionName] = function (data) {
+ deferred.resolve(data);
+
+ try {
+ delete window[functionName];
+ } catch (e) {
+ window[functionName] = undefined;
+ }
+ };
+
+ Resource._Implementations.loadAndExecuteScript(url, functionName, deferred);
+ return deferred.promise;
+ };
+
+ const promise = RequestScheduler.request(request);
+ if (!defined(promise)) {
+ return;
+ }
+
+ return promise.catch(function (e) {
+ if (request.state !== RequestState.FAILED) {
+ return Promise.reject(e);
+ }
+
+ return resource.retryOnError(e).then(function (retry) {
+ if (retry) {
+ // Reset request so it can try again
+ request.state = RequestState.UNISSUED;
+ request.deferred = undefined;
+
+ return fetchJsonp(resource, callbackParameterName, functionName);
+ }
+
+ return Promise.reject(e);
+ });
+ });
+}
+
+/**
+ * Creates a Resource from a URL and calls fetchJsonp() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @param {string} [options.callbackParameterName='callback'] The callback parameter name that the server expects.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.fetchJsonp = function (options) {
+ const resource = new Resource(options);
+ return resource.fetchJsonp(options.callbackParameterName);
+};
+
+/**
+ * @private
+ */
+Resource.prototype._makeRequest = function (options) {
+ const resource = this;
+ checkAndResetRequest(resource.request);
+
+ const request = resource.request;
+ const url = resource.url;
+ request.url = url;
+
+ request.requestFunction = function () {
+ const responseType = options.responseType;
+ const headers = combine(options.headers, resource.headers);
+ const overrideMimeType = options.overrideMimeType;
+ const method = options.method;
+ const data = options.data;
+ const deferred = defer();
+ const xhr = Resource._Implementations.loadWithXhr(
+ url,
+ responseType,
+ method,
+ data,
+ headers,
+ deferred,
+ overrideMimeType
+ );
+ if (defined(xhr) && defined(xhr.abort)) {
+ request.cancelFunction = function () {
+ xhr.abort();
+ };
+ }
+ return deferred.promise;
+ };
+
+ const promise = RequestScheduler.request(request);
+ if (!defined(promise)) {
+ return;
+ }
+
+ return promise
+ .then(function (data) {
+ // explicitly set to undefined to ensure GC of request response data. See #8843
+ request.cancelFunction = undefined;
+ return data;
+ })
+ .catch(function (e) {
+ request.cancelFunction = undefined;
+ if (request.state !== RequestState.FAILED) {
+ return Promise.reject(e);
+ }
+
+ return resource.retryOnError(e).then(function (retry) {
+ if (retry) {
+ // Reset request so it can try again
+ request.state = RequestState.UNISSUED;
+ request.deferred = undefined;
+
+ return resource.fetch(options);
+ }
+
+ return Promise.reject(e);
+ });
+ });
+};
+
+/**
+ * Checks to make sure the Resource isn't already being requested.
+ *
+ * @param {Request} request The request to check.
+ *
+ * @private
+ */
+function checkAndResetRequest(request) {
+ if (
+ request.state === RequestState.ISSUED ||
+ request.state === RequestState.ACTIVE
+ ) {
+ throw new RuntimeError("The Resource is already being fetched.");
+ }
+
+ request.state = RequestState.UNISSUED;
+ request.deferred = undefined;
+}
+
+const dataUriRegex = /^data:(.*?)(;base64)?,(.*)$/;
+
+function decodeDataUriText(isBase64, data) {
+ const result = decodeURIComponent(data);
+ if (isBase64) {
+ return atob(result);
+ }
+ return result;
+}
+
+function decodeDataUriArrayBuffer(isBase64, data) {
+ const byteString = decodeDataUriText(isBase64, data);
+ const buffer = new ArrayBuffer(byteString.length);
+ const view = new Uint8Array(buffer);
+ for (let i = 0; i < byteString.length; i++) {
+ view[i] = byteString.charCodeAt(i);
+ }
+ return buffer;
+}
+
+function decodeDataUri(dataUriRegexResult, responseType) {
+ responseType = defaultValue(responseType, "");
+ const mimeType = dataUriRegexResult[1];
+ const isBase64 = !!dataUriRegexResult[2];
+ const data = dataUriRegexResult[3];
+ let buffer;
+ let parser;
+
+ switch (responseType) {
+ case "":
+ case "text":
+ return decodeDataUriText(isBase64, data);
+ case "arraybuffer":
+ return decodeDataUriArrayBuffer(isBase64, data);
+ case "blob":
+ buffer = decodeDataUriArrayBuffer(isBase64, data);
+ return new Blob([buffer], {
+ type: mimeType,
+ });
+ case "document":
+ parser = new DOMParser();
+ return parser.parseFromString(
+ decodeDataUriText(isBase64, data),
+ mimeType
+ );
+ case "json":
+ return JSON.parse(decodeDataUriText(isBase64, data));
+ default:
+ //>>includeStart('debug', pragmas.debug);
+ throw new DeveloperError(`Unhandled responseType: ${responseType}`);
+ //>>includeEnd('debug');
+ }
+}
+
+/**
+ * Asynchronously loads the given resource. Returns a promise that will resolve to
+ * the result once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled. It's recommended that you use
+ * the more specific functions eg. fetchJson, fetchBlob, etc.
+ *
+ * @param {object} [options] Object with the following properties:
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {object} [options.headers] Additional HTTP headers to send with the request, if any.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * resource.fetch()
+ * .then(function(body) {
+ * // use the data
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.fetch = function (options) {
+ options = defaultClone(options, {});
+ options.method = "GET";
+
+ return this._makeRequest(options);
+};
+
+/**
+ * Creates a Resource from a URL and calls fetch() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.fetch = function (options) {
+ const resource = new Resource(options);
+ return resource.fetch({
+ // Make copy of just the needed fields because headers can be passed to both the constructor and to fetch
+ responseType: options.responseType,
+ overrideMimeType: options.overrideMimeType,
+ });
+};
+
+/**
+ * Asynchronously deletes the given resource. Returns a promise that will resolve to
+ * the result once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled.
+ *
+ * @param {object} [options] Object with the following properties:
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {object} [options.headers] Additional HTTP headers to send with the request, if any.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * resource.delete()
+ * .then(function(body) {
+ * // use the data
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.delete = function (options) {
+ options = defaultClone(options, {});
+ options.method = "DELETE";
+
+ return this._makeRequest(options);
+};
+
+/**
+ * Creates a Resource from a URL and calls delete() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.data] Data that is posted with the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.delete = function (options) {
+ const resource = new Resource(options);
+ return resource.delete({
+ // Make copy of just the needed fields because headers can be passed to both the constructor and to fetch
+ responseType: options.responseType,
+ overrideMimeType: options.overrideMimeType,
+ data: options.data,
+ });
+};
+
+/**
+ * Asynchronously gets headers the given resource. Returns a promise that will resolve to
+ * the result once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled.
+ *
+ * @param {object} [options] Object with the following properties:
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {object} [options.headers] Additional HTTP headers to send with the request, if any.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * resource.head()
+ * .then(function(headers) {
+ * // use the data
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.head = function (options) {
+ options = defaultClone(options, {});
+ options.method = "HEAD";
+
+ return this._makeRequest(options);
+};
+
+/**
+ * Creates a Resource from a URL and calls head() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.head = function (options) {
+ const resource = new Resource(options);
+ return resource.head({
+ // Make copy of just the needed fields because headers can be passed to both the constructor and to fetch
+ responseType: options.responseType,
+ overrideMimeType: options.overrideMimeType,
+ });
+};
+
+/**
+ * Asynchronously gets options the given resource. Returns a promise that will resolve to
+ * the result once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled.
+ *
+ * @param {object} [options] Object with the following properties:
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {object} [options.headers] Additional HTTP headers to send with the request, if any.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * resource.options()
+ * .then(function(headers) {
+ * // use the data
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.options = function (options) {
+ options = defaultClone(options, {});
+ options.method = "OPTIONS";
+
+ return this._makeRequest(options);
+};
+
+/**
+ * Creates a Resource from a URL and calls options() on it.
+ *
+ * @param {string|object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.options = function (options) {
+ const resource = new Resource(options);
+ return resource.options({
+ // Make copy of just the needed fields because headers can be passed to both the constructor and to fetch
+ responseType: options.responseType,
+ overrideMimeType: options.overrideMimeType,
+ });
+};
+
+/**
+ * Asynchronously posts data to the given resource. Returns a promise that will resolve to
+ * the result once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled.
+ *
+ * @param {object} data Data that is posted with the resource.
+ * @param {object} [options] Object with the following properties:
+ * @param {object} [options.data] Data that is posted with the resource.
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {object} [options.headers] Additional HTTP headers to send with the request, if any.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * resource.post(data)
+ * .then(function(result) {
+ * // use the result
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.post = function (data, options) {
+ Check.defined("data", data);
+
+ options = defaultClone(options, {});
+ options.method = "POST";
+ options.data = data;
+
+ return this._makeRequest(options);
+};
+
+/**
+ * Creates a Resource from a URL and calls post() on it.
+ *
+ * @param {object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} options.data Data that is posted with the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.post = function (options) {
+ const resource = new Resource(options);
+ return resource.post(options.data, {
+ // Make copy of just the needed fields because headers can be passed to both the constructor and to post
+ responseType: options.responseType,
+ overrideMimeType: options.overrideMimeType,
+ });
+};
+
+/**
+ * Asynchronously puts data to the given resource. Returns a promise that will resolve to
+ * the result once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled.
+ *
+ * @param {object} data Data that is posted with the resource.
+ * @param {object} [options] Object with the following properties:
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {object} [options.headers] Additional HTTP headers to send with the request, if any.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * resource.put(data)
+ * .then(function(result) {
+ * // use the result
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.put = function (data, options) {
+ Check.defined("data", data);
+
+ options = defaultClone(options, {});
+ options.method = "PUT";
+ options.data = data;
+
+ return this._makeRequest(options);
+};
+
+/**
+ * Creates a Resource from a URL and calls put() on it.
+ *
+ * @param {object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} options.data Data that is posted with the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.put = function (options) {
+ const resource = new Resource(options);
+ return resource.put(options.data, {
+ // Make copy of just the needed fields because headers can be passed to both the constructor and to post
+ responseType: options.responseType,
+ overrideMimeType: options.overrideMimeType,
+ });
+};
+
+/**
+ * Asynchronously patches data to the given resource. Returns a promise that will resolve to
+ * the result once loaded, or reject if the resource failed to load. The data is loaded
+ * using XMLHttpRequest, which means that in order to make requests to another origin,
+ * the server must have Cross-Origin Resource Sharing (CORS) headers enabled.
+ *
+ * @param {object} data Data that is posted with the resource.
+ * @param {object} [options] Object with the following properties:
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {object} [options.headers] Additional HTTP headers to send with the request, if any.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ *
+ *
+ * @example
+ * resource.patch(data)
+ * .then(function(result) {
+ * // use the result
+ * }).catch(function(error) {
+ * // an error occurred
+ * });
+ *
+ * @see {@link http://www.w3.org/TR/cors/|Cross-Origin Resource Sharing}
+ * @see {@link http://wiki.commonjs.org/wiki/Promises/A|CommonJS Promises/A}
+ */
+Resource.prototype.patch = function (data, options) {
+ Check.defined("data", data);
+
+ options = defaultClone(options, {});
+ options.method = "PATCH";
+ options.data = data;
+
+ return this._makeRequest(options);
+};
+
+/**
+ * Creates a Resource from a URL and calls patch() on it.
+ *
+ * @param {object} options A url or an object with the following properties
+ * @param {string} options.url The url of the resource.
+ * @param {object} options.data Data that is posted with the resource.
+ * @param {object} [options.queryParameters] An object containing query parameters that will be sent when retrieving the resource.
+ * @param {object} [options.templateValues] Key/Value pairs that are used to replace template values (eg. {x}).
+ * @param {object} [options.headers={}] Additional HTTP headers that will be sent.
+ * @param {Proxy} [options.proxy] A proxy to be used when loading the resource.
+ * @param {Resource.RetryCallback} [options.retryCallback] The Function to call when a request for this resource fails. If it returns true, the request will be retried.
+ * @param {number} [options.retryAttempts=0] The number of times the retryCallback should be called before giving up.
+ * @param {Request} [options.request] A Request object that will be used. Intended for internal use only.
+ * @param {string} [options.responseType] The type of response. This controls the type of item returned.
+ * @param {string} [options.overrideMimeType] Overrides the MIME type returned by the server.
+ * @returns {Promise|undefined} a promise that will resolve to the requested data when loaded. Returns undefined if request.throttle is true and the request does not have high enough priority.
+ */
+Resource.patch = function (options) {
+ const resource = new Resource(options);
+ return resource.patch(options.data, {
+ // Make copy of just the needed fields because headers can be passed to both the constructor and to post
+ responseType: options.responseType,
+ overrideMimeType: options.overrideMimeType,
+ });
+};
+
+/**
+ * Contains implementations of functions that can be replaced for testing
+ *
+ * @private
+ */
+Resource._Implementations = {};
+
+Resource._Implementations.loadImageElement = function (
+ url,
+ crossOrigin,
+ deferred
+) {
+ const image = new Image();
+
+ image.onload = function () {
+ // work-around a known issue with Firefox and dimensionless SVG, see:
+ // - https://github.com/whatwg/html/issues/3510
+ // - https://bugzilla.mozilla.org/show_bug.cgi?id=700533
+ if (
+ image.naturalWidth === 0 &&
+ image.naturalHeight === 0 &&
+ image.width === 0 &&
+ image.height === 0
+ ) {
+ // these values affect rasterization and will likely mar the content
+ // until Firefox takes a stance on the issue, marred content is better than no content
+ // Chromium uses a more refined heuristic about its choice given nil viewBox, and a better stance and solution is
+ // proposed later in the original issue thread:
+ // - Chromium behavior: https://github.com/CesiumGS/cesium/issues/9188#issuecomment-704400825
+ // - Cesium's stance/solve: https://github.com/CesiumGS/cesium/issues/9188#issuecomment-720645777
+ image.width = 300;
+ image.height = 150;
+ }
+ deferred.resolve(image);
+ };
+
+ image.onerror = function (e) {
+ deferred.reject(e);
+ };
+
+ if (crossOrigin) {
+ if (TrustedServers.contains(url)) {
+ image.crossOrigin = "use-credentials";
+ } else {
+ image.crossOrigin = "";
+ }
+ }
+
+ image.src = url;
+};
+
+Resource._Implementations.createImage = function (
+ request,
+ crossOrigin,
+ deferred,
+ flipY,
+ skipColorSpaceConversion,
+ preferImageBitmap
+) {
+ const url = request.url;
+ // Passing an Image to createImageBitmap will force it to run on the main thread
+ // since DOM elements don't exist on workers. We convert it to a blob so it's non-blocking.
+ // See:
+ // https://bugzilla.mozilla.org/show_bug.cgi?id=1044102#c38
+ // https://bugs.chromium.org/p/chromium/issues/detail?id=580202#c10
+ Resource.supportsImageBitmapOptions()
+ .then(function (supportsImageBitmap) {
+ // We can only use ImageBitmap if we can flip on decode.
+ // See: https://github.com/CesiumGS/cesium/pull/7579#issuecomment-466146898
+ if (!(supportsImageBitmap && preferImageBitmap)) {
+ Resource._Implementations.loadImageElement(url, crossOrigin, deferred);
+ return;
+ }
+ const responseType = "blob";
+ const method = "GET";
+ const xhrDeferred = defer();
+ const xhr = Resource._Implementations.loadWithXhr(
+ url,
+ responseType,
+ method,
+ undefined,
+ undefined,
+ xhrDeferred,
+ undefined,
+ undefined,
+ undefined
+ );
+
+ if (defined(xhr) && defined(xhr.abort)) {
+ request.cancelFunction = function () {
+ xhr.abort();
+ };
+ }
+ return xhrDeferred.promise
+ .then(function (blob) {
+ if (!defined(blob)) {
+ deferred.reject(
+ new RuntimeError(
+ `Successfully retrieved ${url} but it contained no content.`
+ )
+ );
+ return;
+ }
+
+ return Resource.createImageBitmapFromBlob(blob, {
+ flipY: flipY,
+ premultiplyAlpha: false,
+ skipColorSpaceConversion: skipColorSpaceConversion,
+ });
+ })
+ .then(function (image) {
+ deferred.resolve(image);
+ });
+ })
+ .catch(function (e) {
+ deferred.reject(e);
+ });
+};
+
+/**
+ * Wrapper for createImageBitmap
+ *
+ * @private
+ */
+Resource.createImageBitmapFromBlob = function (blob, options) {
+ Check.defined("options", options);
+ Check.typeOf.bool("options.flipY", options.flipY);
+ Check.typeOf.bool("options.premultiplyAlpha", options.premultiplyAlpha);
+ Check.typeOf.bool(
+ "options.skipColorSpaceConversion",
+ options.skipColorSpaceConversion
+ );
+
+ return createImageBitmap(blob, {
+ imageOrientation: options.flipY ? "flipY" : "none",
+ premultiplyAlpha: options.premultiplyAlpha ? "premultiply" : "none",
+ colorSpaceConversion: options.skipColorSpaceConversion ? "none" : "default",
+ });
+};
+
+function loadWithHttpRequest(
+ url,
+ responseType,
+ method,
+ data,
+ headers,
+ deferred,
+ overrideMimeType
+) {
+ // Note: only the 'json' and 'text' responseTypes transforms the loaded buffer
+ fetch(url, {
+ method,
+ headers,
+ })
+ .then(async (response) => {
+ if (!response.ok) {
+ const responseHeaders = {};
+ response.headers.forEach((value, key) => {
+ responseHeaders[key] = value;
+ });
+ deferred.reject(
+ new RequestErrorEvent(response.status, response, responseHeaders)
+ );
+ return;
+ }
+
+ switch (responseType) {
+ case "text":
+ deferred.resolve(response.text());
+ break;
+ case "json":
+ deferred.resolve(response.json());
+ break;
+ default:
+ deferred.resolve(new Uint8Array(await response.arrayBuffer()).buffer);
+ break;
+ }
+ })
+ .catch(() => {
+ deferred.reject(new RequestErrorEvent());
+ });
+}
+
+const noXMLHttpRequest = typeof XMLHttpRequest === "undefined";
+Resource._Implementations.loadWithXhr = function (
+ url,
+ responseType,
+ method,
+ data,
+ headers,
+ deferred,
+ overrideMimeType
+) {
+ const dataUriRegexResult = dataUriRegex.exec(url);
+ if (dataUriRegexResult !== null) {
+ deferred.resolve(decodeDataUri(dataUriRegexResult, responseType));
+ return;
+ }
+
+ if (noXMLHttpRequest) {
+ loadWithHttpRequest(
+ url,
+ responseType,
+ method,
+ data,
+ headers,
+ deferred,
+ overrideMimeType
+ );
+ return;
+ }
+
+ const xhr = new XMLHttpRequest();
+
+ if (TrustedServers.contains(url)) {
+ xhr.withCredentials = true;
+ }
+
+ xhr.open(method, url, true);
+
+ if (defined(overrideMimeType) && defined(xhr.overrideMimeType)) {
+ xhr.overrideMimeType(overrideMimeType);
+ }
+
+ if (defined(headers)) {
+ for (const key in headers) {
+ if (headers.hasOwnProperty(key)) {
+ xhr.setRequestHeader(key, headers[key]);
+ }
+ }
+ }
+
+ if (defined(responseType)) {
+ xhr.responseType = responseType;
+ }
+
+ // While non-standard, file protocol always returns a status of 0 on success
+ let localFile = false;
+ if (typeof url === "string") {
+ localFile =
+ url.indexOf("file://") === 0 ||
+ (typeof window !== "undefined" && window.location.origin === "file://");
+ }
+
+ xhr.onload = function () {
+ if (
+ (xhr.status < 200 || xhr.status >= 300) &&
+ !(localFile && xhr.status === 0)
+ ) {
+ deferred.reject(
+ new RequestErrorEvent(
+ xhr.status,
+ xhr.response,
+ xhr.getAllResponseHeaders()
+ )
+ );
+ return;
+ }
+
+ const response = xhr.response;
+ const browserResponseType = xhr.responseType;
+
+ if (method === "HEAD" || method === "OPTIONS") {
+ const responseHeaderString = xhr.getAllResponseHeaders();
+ const splitHeaders = responseHeaderString.trim().split(/[\r\n]+/);
+
+ const responseHeaders = {};
+ splitHeaders.forEach(function (line) {
+ const parts = line.split(": ");
+ const header = parts.shift();
+ responseHeaders[header] = parts.join(": ");
+ });
+
+ deferred.resolve(responseHeaders);
+ return;
+ }
+
+ //All modern browsers will go into either the first or second if block or last else block.
+ //Other code paths support older browsers that either do not support the supplied responseType
+ //or do not support the xhr.response property.
+ if (xhr.status === 204) {
+ // accept no content
+ deferred.resolve(undefined);
+ } else if (
+ defined(response) &&
+ (!defined(responseType) || browserResponseType === responseType)
+ ) {
+ deferred.resolve(response);
+ } else if (responseType === "json" && typeof response === "string") {
+ try {
+ deferred.resolve(JSON.parse(response));
+ } catch (e) {
+ deferred.reject(e);
+ }
+ } else if (
+ (browserResponseType === "" || browserResponseType === "document") &&
+ defined(xhr.responseXML) &&
+ xhr.responseXML.hasChildNodes()
+ ) {
+ deferred.resolve(xhr.responseXML);
+ } else if (
+ (browserResponseType === "" || browserResponseType === "text") &&
+ defined(xhr.responseText)
+ ) {
+ deferred.resolve(xhr.responseText);
+ } else {
+ deferred.reject(
+ new RuntimeError("Invalid XMLHttpRequest response type.")
+ );
+ }
+ };
+
+ xhr.onerror = function (e) {
+ deferred.reject(new RequestErrorEvent());
+ };
+
+ xhr.send(data);
+
+ return xhr;
+};
+
+Resource._Implementations.loadAndExecuteScript = function (
+ url,
+ functionName,
+ deferred
+) {
+ return loadAndExecuteScript(url, functionName).catch(function (e) {
+ deferred.reject(e);
+ });
+};
+
+/**
+ * The default implementations
+ *
+ * @private
+ */
+Resource._DefaultImplementations = {};
+Resource._DefaultImplementations.createImage =
+ Resource._Implementations.createImage;
+Resource._DefaultImplementations.loadWithXhr =
+ Resource._Implementations.loadWithXhr;
+Resource._DefaultImplementations.loadAndExecuteScript =
+ Resource._Implementations.loadAndExecuteScript;
+
+/**
+ * A resource instance initialized to the current browser location
+ *
+ * @type {Resource}
+ * @constant
+ */
+Resource.DEFAULT = Object.freeze(
+ new Resource({
+ url:
+ typeof document === "undefined"
+ ? ""
+ : document.location.href.split("?")[0],
+ })
+);
+
+/**
+ * A function that returns the value of the property.
+ * @callback Resource.RetryCallback
+ *
+ * @param {Resource} [resource] The resource that failed to load.
+ * @param {RequestErrorEvent} [error] The error that occurred during the loading of the resource.
+ * @returns {boolean|Promise} If true or a promise that resolved to true, the resource will be retried. Otherwise the failure will be returned.
+ */
+export default Resource;
diff --git a/TaskProcessor.js b/TaskProcessor.js
index 61798f4..9eca8d8 100644
--- a/TaskProcessor.js
+++ b/TaskProcessor.js
@@ -1,4 +1,4 @@
-import Uri from "https://cdnjs.cloudflare.com/ajax/libs/URI.js/1.19.11/URI.min.js";
+import Uri from "urijs";
import buildModuleUrl from "./buildModuleUrl.js";
import defaultValue from "./defaultValue.js";
import defined from "./defined.js";
diff --git a/destroyObject.js b/destroyObject.js
new file mode 100644
index 0000000..40093a8
--- /dev/null
+++ b/destroyObject.js
@@ -0,0 +1,57 @@
+import defaultValue from "./defaultValue.js";
+import DeveloperError from "./DeveloperError.js";
+
+function returnTrue() {
+ return true;
+}
+
+/**
+ * Destroys an object. Each of the object's functions, including functions in its prototype,
+ * is replaced with a function that throws a {@link DeveloperError}, except for the object's
+ * isDestroyed function, which is set to a function that returns true.
+ * The object's properties are removed with delete.
+ *
+ * This function is used by objects that hold native resources, e.g., WebGL resources, which
+ * need to be explicitly released. Client code calls an object's destroy function,
+ * which then releases the native resource and calls destroyObject to put itself
+ * in a destroyed state.
+ *
+ * @function
+ *
+ * @param {object} object The object to destroy.
+ * @param {string} [message] The message to include in the exception that is thrown if
+ * a destroyed object's function is called.
+ *
+ *
+ * @example
+ * // How a texture would destroy itself.
+ * this.destroy = function () {
+ * _gl.deleteTexture(_texture);
+ * return Cesium.destroyObject(this);
+ * };
+ *
+ * @see DeveloperError
+ */
+function destroyObject(object, message) {
+ message = defaultValue(
+ message,
+ "This object was destroyed, i.e., destroy() was called."
+ );
+
+ function throwOnDestroyed() {
+ //>>includeStart('debug', pragmas.debug);
+ throw new DeveloperError(message);
+ //>>includeEnd('debug');
+ }
+
+ for (const key in object) {
+ if (typeof object[key] === "function") {
+ object[key] = throwOnDestroyed;
+ }
+ }
+
+ object.isDestroyed = returnTrue;
+
+ return undefined;
+}
+export default destroyObject;
diff --git a/isCrossOriginUrl.js b/isCrossOriginUrl.js
new file mode 100644
index 0000000..4557038
--- /dev/null
+++ b/isCrossOriginUrl.js
@@ -0,0 +1,30 @@
+import defined from "./defined.js";
+
+let a;
+
+/**
+ * Given a URL, determine whether that URL is considered cross-origin to the current page.
+ *
+ * @private
+ */
+function isCrossOriginUrl(url) {
+ if (!defined(a)) {
+ a = document.createElement("a");
+ }
+
+ // copy window location into the anchor to get consistent results
+ // when the port is default for the protocol (e.g. 80 for HTTP)
+ a.href = window.location.href;
+
+ // host includes both hostname and port if the port is not standard
+ const host = a.host;
+ const protocol = a.protocol;
+
+ a.href = url;
+ // IE only absolutizes href on get, not set
+ // eslint-disable-next-line no-self-assign
+ a.href = a.href;
+
+ return protocol !== a.protocol || host !== a.host;
+}
+export default isCrossOriginUrl;