|  | // Copyright (c) 2012, the Dart project authors.  Please see the AUTHORS file | 
|  | // for details. All rights reserved. Use of this source code is governed by a | 
|  | // BSD-style license that can be found in the LICENSE file. | 
|  |  | 
|  | // Conversions for IDBKey. | 
|  | // | 
|  | // Per http://www.w3.org/TR/IndexedDB/#key-construct | 
|  | // | 
|  | // "A value is said to be a valid key if it is one of the following types: Array | 
|  | // JavaScript objects [ECMA-262], DOMString [WEBIDL], Date [ECMA-262] or float | 
|  | // [WEBIDL]. However Arrays are only valid keys if every item in the array is | 
|  | // defined and is a valid key (i.e. sparse arrays can not be valid keys) and if | 
|  | // the Array doesn't directly or indirectly contain itself. Any non-numeric | 
|  | // properties are ignored, and thus does not affect whether the Array is a valid | 
|  | // key. Additionally, if the value is of type float, it is only a valid key if | 
|  | // it is not NaN, and if the value is of type Date it is only a valid key if its | 
|  | // [[PrimitiveValue]] internal property, as defined by [ECMA-262], is not NaN." | 
|  |  | 
|  | // What is required is to ensure that an Lists in the key are actually | 
|  | // JavaScript arrays, and any Dates are JavaScript Dates. | 
|  |  | 
|  | // Conversions for Window.  These check if the window is the local | 
|  | // window, and if it's not, wraps or unwraps it with a secure wrapper. | 
|  | // We need to test for EventTarget here as well as it's a base type. | 
|  | // We omit an unwrapper for Window as no methods take a non-local | 
|  | // window as a parameter. | 
|  |  | 
|  | part of html_common; | 
|  |  | 
|  | /// Converts a Dart value into a JavaScript SerializedScriptValue. | 
|  | convertDartToNative_SerializedScriptValue(value) { | 
|  | return convertDartToNative_PrepareForStructuredClone(value); | 
|  | } | 
|  |  | 
|  | /// Since the source object may be viewed via a JavaScript event listener the | 
|  | /// original may not be modified. | 
|  | convertNativeToDart_SerializedScriptValue(object) { | 
|  | return convertNativeToDart_AcceptStructuredClone(object, mustCopy: true); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Converts a Dart value into a JavaScript SerializedScriptValue.  Returns the | 
|  | * original input or a functional 'copy'.  Does not mutate the original. | 
|  | * | 
|  | * The main transformation is the translation of Dart Maps are converted to | 
|  | * JavaScript Objects. | 
|  | * | 
|  | * The algorithm is essentially a dry-run of the structured clone algorithm | 
|  | * described at | 
|  | * http://www.whatwg.org/specs/web-apps/current-work/multipage/common-dom-interfaces.html#structured-clone | 
|  | * https://www.khronos.org/registry/typedarray/specs/latest/#9 | 
|  | * | 
|  | * Since the result of this function is expected to be passed only to JavaScript | 
|  | * operations that perform the structured clone algorithm which does not mutate | 
|  | * its output, the result may share structure with the input [value]. | 
|  | */ | 
|  | abstract class _StructuredClone { | 
|  | // TODO(sra): Replace slots with identity hash table. | 
|  | var values = []; | 
|  | var copies = []; // initially 'null', 'true' during initial DFS, then a copy. | 
|  |  | 
|  | int findSlot(value) { | 
|  | int length = values.length; | 
|  | for (int i = 0; i < length; i++) { | 
|  | if (identical(values[i], value)) return i; | 
|  | } | 
|  | values.add(value); | 
|  | copies.add(null); | 
|  | return length; | 
|  | } | 
|  |  | 
|  | readSlot(int i) => copies[i]; | 
|  | writeSlot(int i, x) { | 
|  | copies[i] = x; | 
|  | } | 
|  |  | 
|  | cleanupSlots() {} // Will be needed if we mark objects with a property. | 
|  | bool cloneNotRequired(object); | 
|  | JSObject newJsObject(); | 
|  | void forEachObjectKey(object, action(key, value)); | 
|  | void putIntoObject(object, key, value); | 
|  | newJsMap(); | 
|  | List newJsList(length); | 
|  | void putIntoMap(map, key, value); | 
|  |  | 
|  | // Returns the input, or a clone of the input. | 
|  | walk(e) { | 
|  | if (e == null) return e; | 
|  | if (e is bool) return e; | 
|  | if (e is num) return e; | 
|  | if (e is String) return e; | 
|  | if (e is DateTime) { | 
|  | return convertDartToNative_DateTime(e); | 
|  | } | 
|  | if (e is RegExp) { | 
|  | // TODO(sra). | 
|  | throw new UnimplementedError('structured clone of RegExp'); | 
|  | } | 
|  |  | 
|  | // The browser's internal structured cloning algorithm will copy certain | 
|  | // types of object, but it will copy only its own implementations and not | 
|  | // just any Dart implementations of the interface. | 
|  |  | 
|  | // TODO(sra): The JavaScript objects suitable for direct cloning by the | 
|  | // structured clone algorithm could be tagged with an private interface. | 
|  |  | 
|  | if (e is File) return e; | 
|  | if (e is Blob) return e; | 
|  | if (e is FileList) return e; | 
|  |  | 
|  | // TODO(sra): Firefox: How to convert _TypedImageData on the other end? | 
|  | if (e is ImageData) return e; | 
|  | if (cloneNotRequired(e)) return e; | 
|  |  | 
|  | if (e is Map) { | 
|  | var slot = findSlot(e); | 
|  | var copy = readSlot(slot); | 
|  | if (copy != null) return copy; | 
|  | copy = newJsMap(); | 
|  | writeSlot(slot, copy); | 
|  | e.forEach((key, value) { | 
|  | putIntoMap(copy, key, walk(value)); | 
|  | }); | 
|  | return copy; | 
|  | } | 
|  |  | 
|  | if (e is List) { | 
|  | // Since a JavaScript Array is an instance of Dart List it is tempting | 
|  | // in dart2js to avoid making a copy of the list if there is no need | 
|  | // to copy anything reachable from the array.  However, the list may have | 
|  | // non-native properties or methods from interceptors and such, e.g. | 
|  | // an immutability marker. So we  had to stop doing that. | 
|  | var slot = findSlot(e); | 
|  | var copy = JS('returns:List|Null;creates:;', '#', readSlot(slot)); | 
|  | if (copy != null) return copy; | 
|  | copy = copyList(e, slot); | 
|  | return copy; | 
|  | } | 
|  |  | 
|  | if (e is JSObject) { | 
|  | var slot = findSlot(e); | 
|  | var copy = readSlot(slot); | 
|  | if (copy != null) return copy; | 
|  | copy = newJsObject(); | 
|  | writeSlot(slot, copy); | 
|  | // TODO: Consider inlining this so we don't allocate a closure. | 
|  | forEachObjectKey(e, (key, value) { | 
|  | putIntoObject(copy, key, walk(value)); | 
|  | }); | 
|  | return copy; | 
|  | } | 
|  |  | 
|  | throw new UnimplementedError('structured clone of other type'); | 
|  | } | 
|  |  | 
|  | List copyList(List e, int slot) { | 
|  | int i = 0; | 
|  | int length = e.length; | 
|  | var copy = newJsList(length); | 
|  | writeSlot(slot, copy); | 
|  | for (; i < length; i++) { | 
|  | copy[i] = walk(e[i]); | 
|  | } | 
|  | return copy; | 
|  | } | 
|  |  | 
|  | convertDartToNative_PrepareForStructuredClone(value) { | 
|  | var copy = walk(value); | 
|  | cleanupSlots(); | 
|  | return copy; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Converts a native value into a Dart object. | 
|  | * | 
|  | * If [mustCopy] is [:false:], may return the original input.  May mutate the | 
|  | * original input (but will be idempotent if mutation occurs).  It is assumed | 
|  | * that this conversion happens on native serializable script values such values | 
|  | * from native DOM calls. | 
|  | * | 
|  | * [object] is the result of a structured clone operation. | 
|  | * | 
|  | * If necessary, JavaScript Dates are converted into Dart Dates. | 
|  | * | 
|  | * If [mustCopy] is [:true:], the entire object is copied and the original input | 
|  | * is not mutated.  This should be the case where Dart and JavaScript code can | 
|  | * access the value, for example, via multiple event listeners for | 
|  | * MessageEvents.  Mutating the object to make it more 'Dart-like' would corrupt | 
|  | * the value as seen from the JavaScript listeners. | 
|  | */ | 
|  | abstract class _AcceptStructuredClone { | 
|  | // TODO(sra): Replace slots with identity hash table. | 
|  | var values = []; | 
|  | var copies = []; // initially 'null', 'true' during initial DFS, then a copy. | 
|  | bool mustCopy = false; | 
|  |  | 
|  | int findSlot(value) { | 
|  | int length = values.length; | 
|  | for (int i = 0; i < length; i++) { | 
|  | if (identicalInJs(values[i], value)) return i; | 
|  | } | 
|  | values.add(value); | 
|  | copies.add(null); | 
|  | return length; | 
|  | } | 
|  |  | 
|  | /// Are the two objects identical, but taking into account that two JsObject | 
|  | /// wrappers may not be identical, but their underlying Js Object might be. | 
|  | bool identicalInJs(a, b); | 
|  | readSlot(int i) => copies[i]; | 
|  | writeSlot(int i, x) { | 
|  | copies[i] = x; | 
|  | } | 
|  |  | 
|  | /// Iterate over the JS properties. | 
|  | forEachJsField(object, action(key, value)); | 
|  |  | 
|  | /// Create a new Dart list of the given length. May create a native List or | 
|  | /// a JsArray, depending if we're in Dartium or dart2js. | 
|  | List newDartList(length); | 
|  |  | 
|  | walk(e) { | 
|  | if (e == null) return e; | 
|  | if (e is bool) return e; | 
|  | if (e is num) return e; | 
|  | if (e is String) return e; | 
|  |  | 
|  | if (isJavaScriptDate(e)) { | 
|  | return convertNativeToDart_DateTime(e); | 
|  | } | 
|  |  | 
|  | if (isJavaScriptRegExp(e)) { | 
|  | // TODO(sra). | 
|  | throw new UnimplementedError('structured clone of RegExp'); | 
|  | } | 
|  |  | 
|  | if (isJavaScriptPromise(e)) { | 
|  | return promiseToFuture(e); | 
|  | } | 
|  |  | 
|  | if (isJavaScriptSimpleObject(e)) { | 
|  | // TODO(sra): If mustCopy is false, swizzle the prototype for one of a Map | 
|  | // implementation that uses the properties as storage. | 
|  | var slot = findSlot(e); | 
|  | var copy = readSlot(slot); | 
|  | if (copy != null) return copy; | 
|  | var map = {}; | 
|  |  | 
|  | writeSlot(slot, map); | 
|  | forEachJsField(e, (key, value) => map[key] = walk(value)); | 
|  | return map; | 
|  | } | 
|  |  | 
|  | if (isJavaScriptArray(e)) { | 
|  | var l = JS<List>('returns:List;creates:;', '#', e); | 
|  | var slot = findSlot(l); | 
|  | var copy = JS<List?>('returns:List|Null;creates:;', '#', readSlot(slot)); | 
|  | if (copy != null) return copy; | 
|  |  | 
|  | int length = l.length; | 
|  | // Since a JavaScript Array is an instance of Dart List, we can modify it | 
|  | // in-place unless we must copy. | 
|  | copy = mustCopy ? newDartList(length) : l; | 
|  | writeSlot(slot, copy); | 
|  |  | 
|  | for (int i = 0; i < length; i++) { | 
|  | copy[i] = walk(l[i]); | 
|  | } | 
|  | return copy; | 
|  | } | 
|  |  | 
|  | // Assume anything else is already a valid Dart object, either by having | 
|  | // already been processed, or e.g. a cloneable native class. | 
|  | return e; | 
|  | } | 
|  |  | 
|  | convertNativeToDart_AcceptStructuredClone(object, {mustCopy = false}) { | 
|  | this.mustCopy = mustCopy; | 
|  | var copy = walk(object); | 
|  | return copy; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Conversions for ContextAttributes. | 
|  | // | 
|  | // On Firefox, the returned ContextAttributes is a plain object. | 
|  | class ContextAttributes { | 
|  | bool alpha; | 
|  | bool antialias; | 
|  | bool depth; | 
|  | bool premultipliedAlpha; | 
|  | bool preserveDrawingBuffer; | 
|  | bool stencil; | 
|  | bool failIfMajorPerformanceCaveat; | 
|  |  | 
|  | ContextAttributes( | 
|  | this.alpha, | 
|  | this.antialias, | 
|  | this.depth, | 
|  | this.failIfMajorPerformanceCaveat, | 
|  | this.premultipliedAlpha, | 
|  | this.preserveDrawingBuffer, | 
|  | this.stencil, | 
|  | ); | 
|  | } | 
|  |  | 
|  | convertNativeToDart_ContextAttributes(nativeContextAttributes) { | 
|  | // On Firefox the above test fails because ContextAttributes is a plain | 
|  | // object so we create a _TypedContextAttributes. | 
|  |  | 
|  | return new ContextAttributes( | 
|  | JS('var', '#.alpha', nativeContextAttributes), | 
|  | JS('var', '#.antialias', nativeContextAttributes), | 
|  | JS('var', '#.depth', nativeContextAttributes), | 
|  | JS('var', '#.failIfMajorPerformanceCaveat', nativeContextAttributes), | 
|  | JS('var', '#.premultipliedAlpha', nativeContextAttributes), | 
|  | JS('var', '#.preserveDrawingBuffer', nativeContextAttributes), | 
|  | JS('var', '#.stencil', nativeContextAttributes), | 
|  | ); | 
|  | } | 
|  |  | 
|  | // Conversions for ImageData | 
|  | // | 
|  | // On Firefox, the returned ImageData is a plain object. | 
|  |  | 
|  | class _TypedImageData implements ImageData { | 
|  | final Uint8ClampedList data; | 
|  | final int height; | 
|  | final int width; | 
|  |  | 
|  | _TypedImageData(this.data, this.height, this.width); | 
|  | } | 
|  |  | 
|  | ImageData convertNativeToDart_ImageData(nativeImageData) { | 
|  | // None of the native getters that return ImageData are declared as returning | 
|  | // [ImageData] since that is incorrect for FireFox, which returns a plain | 
|  | // Object.  So we need something that tells the compiler that the ImageData | 
|  | // class has been instantiated. | 
|  | // TODO(sra): Remove this when all the ImageData returning APIs have been | 
|  | // annotated as returning the union ImageData + Object. | 
|  | JS('ImageData', '0'); | 
|  |  | 
|  | if (nativeImageData is ImageData) { | 
|  | // Fix for Issue 16069: on IE, the `data` field is a CanvasPixelArray which | 
|  | // has Array as the constructor property.  This interferes with finding the | 
|  | // correct interceptor.  Fix it by overwriting the constructor property. | 
|  | var data = nativeImageData.data; | 
|  | if (JS('bool', '#.constructor === Array', data)) { | 
|  | if (JS('bool', 'typeof CanvasPixelArray !== "undefined"')) { | 
|  | JS('void', '#.constructor = CanvasPixelArray', data); | 
|  | // This TypedArray property is missing from CanvasPixelArray. | 
|  | JS('void', '#.BYTES_PER_ELEMENT = 1', data); | 
|  | } | 
|  | } | 
|  |  | 
|  | return nativeImageData; | 
|  | } | 
|  |  | 
|  | // On Firefox the above test fails because [nativeImageData] is a plain | 
|  | // object.  So we create a _TypedImageData. | 
|  |  | 
|  | return new _TypedImageData( | 
|  | JS('NativeUint8ClampedList', '#.data', nativeImageData), | 
|  | JS('var', '#.height', nativeImageData), | 
|  | JS('var', '#.width', nativeImageData), | 
|  | ); | 
|  | } | 
|  |  | 
|  | // We can get rid of this conversion if _TypedImageData implements the fields | 
|  | // with native names. | 
|  | convertDartToNative_ImageData(ImageData imageData) { | 
|  | if (imageData is _TypedImageData) { | 
|  | return JS( | 
|  | '', | 
|  | '{data: #, height: #, width: #}', | 
|  | imageData.data, | 
|  | imageData.height, | 
|  | imageData.width, | 
|  | ); | 
|  | } | 
|  | return imageData; | 
|  | } |