astro/packages/webapi/src/polyfill.ts

350 lines
6.9 KiB
TypeScript
Raw Normal View History

import {
AbortController,
AbortSignal,
2022-06-06 16:49:53 +00:00
alert,
atob,
Blob,
2022-06-06 16:49:53 +00:00
btoa,
ByteLengthQueuingStrategy,
2022-06-06 16:49:53 +00:00
cancelAnimationFrame,
cancelIdleCallback,
CanvasRenderingContext2D,
CharacterData,
2022-06-06 16:49:53 +00:00
clearTimeout,
Comment,
CountQueuingStrategy,
CSSStyleSheet,
CustomElementRegistry,
CustomEvent,
Document,
DocumentFragment,
2022-06-06 16:49:53 +00:00
DOMException,
Element,
Event,
EventTarget,
2022-06-06 16:49:53 +00:00
fetch,
File,
FormData,
2022-06-06 16:49:53 +00:00
Headers,
HTMLBodyElement,
HTMLCanvasElement,
HTMLDivElement,
2022-06-06 16:49:53 +00:00
HTMLDocument,
HTMLElement,
HTMLHeadElement,
HTMLHtmlElement,
HTMLImageElement,
HTMLSpanElement,
HTMLStyleElement,
HTMLTemplateElement,
HTMLUnknownElement,
Image,
ImageData,
2022-06-06 16:49:53 +00:00
initCustomElementRegistry,
initDocument,
initMediaQueryList,
initObject,
initPromise,
initRelativeIndexingMethod,
initStorage,
initString,
initWindow,
IntersectionObserver,
MediaQueryList,
MutationObserver,
Node,
NodeFilter,
NodeIterator,
OffscreenCanvas,
ReadableByteStreamController,
ReadableStream,
ReadableStreamBYOBReader,
ReadableStreamBYOBRequest,
ReadableStreamDefaultController,
ReadableStreamDefaultReader,
Request,
2022-06-06 16:49:53 +00:00
requestAnimationFrame,
requestIdleCallback,
ResizeObserver,
Response,
2022-06-06 16:49:53 +00:00
setTimeout,
ShadowRoot,
Storage,
2022-06-06 16:49:53 +00:00
structuredClone,
StyleSheet,
Text,
TransformStream,
TreeWalker,
URLPattern,
2022-06-06 16:49:53 +00:00
Window,
WritableStream,
WritableStreamDefaultController,
WritableStreamDefaultWriter,
} from './ponyfill'
import { exclusions } from './exclusions'
import { inheritence } from './inheritence'
2022-06-06 16:49:53 +00:00
export { pathToPosix } from './lib/utils'
export {
AbortController,
AbortSignal,
2022-06-06 16:49:53 +00:00
alert,
atob,
Blob,
2022-06-06 16:49:53 +00:00
btoa,
ByteLengthQueuingStrategy,
2022-06-06 16:49:53 +00:00
cancelAnimationFrame,
cancelIdleCallback,
CanvasRenderingContext2D,
CharacterData,
2022-06-06 16:49:53 +00:00
clearTimeout,
Comment,
CountQueuingStrategy,
CSSStyleSheet,
CustomElementRegistry,
CustomEvent,
Document,
DocumentFragment,
2022-06-06 16:49:53 +00:00
DOMException,
Element,
Event,
EventTarget,
2022-06-06 16:49:53 +00:00
fetch,
File,
FormData,
2022-06-06 16:49:53 +00:00
Headers,
HTMLBodyElement,
HTMLCanvasElement,
HTMLDivElement,
2022-06-06 16:49:53 +00:00
HTMLDocument,
HTMLElement,
HTMLHeadElement,
HTMLHtmlElement,
HTMLImageElement,
HTMLSpanElement,
HTMLStyleElement,
HTMLTemplateElement,
HTMLUnknownElement,
Image,
ImageData,
2022-06-06 16:49:53 +00:00
IntersectionObserver,
MediaQueryList,
MutationObserver,
Node,
NodeFilter,
NodeIterator,
OffscreenCanvas,
ReadableByteStreamController,
ReadableStream,
ReadableStreamBYOBReader,
ReadableStreamBYOBRequest,
ReadableStreamDefaultController,
ReadableStreamDefaultReader,
Request,
2022-06-06 16:49:53 +00:00
requestAnimationFrame,
requestIdleCallback,
ResizeObserver,
Response,
2022-06-06 16:49:53 +00:00
setTimeout,
ShadowRoot,
2022-06-06 16:49:53 +00:00
structuredClone,
StyleSheet,
Text,
TransformStream,
TreeWalker,
URLPattern,
2022-06-06 16:49:53 +00:00
Window,
WritableStream,
WritableStreamDefaultController,
WritableStreamDefaultWriter,
} from './ponyfill.js'
export const polyfill = (target: any, options?: PolyfillOptions) => {
const webAPIs = {
AbortController,
AbortSignal,
Blob,
ByteLengthQueuingStrategy,
CanvasRenderingContext2D,
CharacterData,
Comment,
CountQueuingStrategy,
CSSStyleSheet,
CustomElementRegistry,
CustomEvent,
Document,
DocumentFragment,
DOMException,
Element,
Event,
EventTarget,
File,
FormData,
HTMLDocument,
HTMLElement,
HTMLBodyElement,
HTMLCanvasElement,
HTMLDivElement,
HTMLHeadElement,
HTMLHtmlElement,
HTMLImageElement,
HTMLSpanElement,
HTMLStyleElement,
HTMLTemplateElement,
HTMLUnknownElement,
Headers,
IntersectionObserver,
Image,
ImageData,
MediaQueryList,
MutationObserver,
Node,
NodeFilter,
NodeIterator,
OffscreenCanvas,
ReadableByteStreamController,
ReadableStream,
ReadableStreamBYOBReader,
ReadableStreamBYOBRequest,
ReadableStreamDefaultController,
ReadableStreamDefaultReader,
Request,
ResizeObserver,
Response,
ShadowRoot,
Storage,
StyleSheet,
Text,
TransformStream,
TreeWalker,
URLPattern,
WritableStream,
WritableStreamDefaultController,
WritableStreamDefaultWriter,
Window,
alert,
atob,
btoa,
cancelAnimationFrame,
cancelIdleCallback,
clearTimeout,
fetch,
requestAnimationFrame,
requestIdleCallback,
setTimeout,
structuredClone,
}
// initialize exclude options
const excludeOptions = new Set(
typeof Object(options).exclude === 'string'
? String(Object(options).exclude).trim().split(/\s+/)
2022-03-07 21:37:50 +00:00
: Array.isArray(Object(options).exclude)
? Object(options).exclude.reduce(
2022-03-07 21:37:50 +00:00
(array: string[], entry: unknown) =>
array.splice(
array.length,
0,
...(typeof entry === 'string' ? entry.trim().split(/\s+/) : [])
) && array,
[]
)
: []
) as Set<string>
// expand exclude options using exclusion shorthands
for (const excludeOption of excludeOptions) {
if (excludeOption in exclusions) {
2022-03-07 21:37:50 +00:00
for (const exclusion of exclusions[
excludeOption as keyof typeof exclusions
]) {
excludeOptions.add(exclusion)
}
}
}
// apply each WebAPI
for (const name of Object.keys(webAPIs)) {
// skip WebAPIs that are excluded
if (excludeOptions.has(name)) continue
2022-03-07 21:37:50 +00:00
// skip WebAPIs that are built-in
if (Object.hasOwnProperty.call(target, name)) continue
// define WebAPIs on the target
2022-03-07 21:37:50 +00:00
Object.defineProperty(target, name, {
configurable: true,
enumerable: true,
writable: true,
value: webAPIs[name as keyof typeof webAPIs],
})
}
// ensure WebAPIs correctly inherit other WebAPIs
for (const name of Object.keys(webAPIs)) {
// skip WebAPIs that are excluded
if (excludeOptions.has(name)) continue
// skip WebAPIs that do not extend other WebAPIs
if (!Object.hasOwnProperty.call(inheritence, name)) continue
const Class = target[name]
const Super = target[inheritence[name as keyof typeof inheritence]]
// skip WebAPIs that are not available
if (!Class || !Super) continue
2022-03-07 21:37:50 +00:00
// skip WebAPIs that are already inherited correctly
if (Object.getPrototypeOf(Class.prototype) === Super.prototype) continue
// define WebAPIs inheritence
Object.setPrototypeOf(Class.prototype, Super.prototype)
}
2022-03-07 21:37:50 +00:00
if (
!excludeOptions.has('HTMLDocument') &&
!excludeOptions.has('HTMLElement')
) {
initDocument(target, excludeOptions)
if (!excludeOptions.has('CustomElementRegistry')) {
initCustomElementRegistry(target, excludeOptions)
}
}
initObject(target, excludeOptions)
initMediaQueryList(target, excludeOptions)
initPromise(target, excludeOptions)
initRelativeIndexingMethod(target, excludeOptions)
initStorage(target, excludeOptions)
initString(target, excludeOptions)
initWindow(target, excludeOptions)
return target
}
polyfill.internals = (target: any, name: string) => {
const init = {
CustomElementRegistry: initCustomElementRegistry,
Document: initDocument,
MediaQueryList: initMediaQueryList,
Object: initObject,
Promise: initPromise,
RelativeIndexingMethod: initRelativeIndexingMethod,
Storage: initStorage,
String: initString,
Window: initWindow,
}
init[name as keyof typeof init](target, new Set<string>())
return target
}
interface PolyfillOptions {
exclude?: string | string[]
override?: Record<string, { (...args: any[]): any }>
}