UNPKG

happy-dom

Version:

Happy DOM is a JavaScript implementation of a web browser without its graphical user interface. It includes many web standards from WHATWG DOM and HTML.

1,170 lines (1,169 loc) 64.6 kB
import { Buffer } from 'buffer'; import { webcrypto } from 'crypto'; import { TextEncoder, TextDecoder } from 'util'; import Stream from 'stream'; import { ReadableStream } from 'stream/web'; import { URLSearchParams } from 'url'; import VM from 'vm'; import * as PropertySymbol from '../PropertySymbol.js'; import Base64 from '../base64/Base64.js'; import BrowserErrorCaptureEnum from '../browser/enums/BrowserErrorCaptureEnum.js'; import Clipboard from '../clipboard/Clipboard.js'; import ClipboardItem from '../clipboard/ClipboardItem.js'; import CSS from '../css/CSS.js'; import CSSRule from '../css/CSSRule.js'; import CSSUnitValue from '../css/CSSUnitValue.js'; import CSSStyleDeclaration from '../css/declaration/CSSStyleDeclaration.js'; import CSSContainerRule from '../css/rules/CSSContainerRule.js'; import CSSFontFaceRule from '../css/rules/CSSFontFaceRule.js'; import CSSKeyframeRule from '../css/rules/CSSKeyframeRule.js'; import CSSKeyframesRule from '../css/rules/CSSKeyframesRule.js'; import CSSMediaRule from '../css/rules/CSSMediaRule.js'; import CSSStyleRule from '../css/rules/CSSStyleRule.js'; import CSSSupportsRule from '../css/rules/CSSSupportsRule.js'; import CustomElementRegistry from '../custom-element/CustomElementRegistry.js'; import DataTransfer from '../event/DataTransfer.js'; import DataTransferItem from '../event/DataTransferItem.js'; import DataTransferItemList from '../event/DataTransferItemList.js'; import Event from '../event/Event.js'; import EventTarget from '../event/EventTarget.js'; import Touch from '../event/Touch.js'; import UIEvent from '../event/UIEvent.js'; import AnimationEvent from '../event/events/AnimationEvent.js'; import ClipboardEvent from '../event/events/ClipboardEvent.js'; import CustomEvent from '../event/events/CustomEvent.js'; import ErrorEvent from '../event/events/ErrorEvent.js'; import FocusEvent from '../event/events/FocusEvent.js'; import HashChangeEvent from '../event/events/HashChangeEvent.js'; import InputEvent from '../event/events/InputEvent.js'; import KeyboardEvent from '../event/events/KeyboardEvent.js'; import MediaQueryListEvent from '../event/events/MediaQueryListEvent.js'; import MessageEvent from '../event/events/MessageEvent.js'; import MouseEvent from '../event/events/MouseEvent.js'; import PointerEvent from '../event/events/PointerEvent.js'; import ProgressEvent from '../event/events/ProgressEvent.js'; import StorageEvent from '../event/events/StorageEvent.js'; import SubmitEvent from '../event/events/SubmitEvent.js'; import TouchEvent from '../event/events/TouchEvent.js'; import WheelEvent from '../event/events/WheelEvent.js'; import DOMExceptionNameEnum from '../exception/DOMExceptionNameEnum.js'; import Fetch from '../fetch/Fetch.js'; import Headers from '../fetch/Headers.js'; import Blob from '../file/Blob.js'; import File from '../file/File.js'; import History from '../history/History.js'; import IntersectionObserver from '../intersection-observer/IntersectionObserver.js'; import IntersectionObserverEntry from '../intersection-observer/IntersectionObserverEntry.js'; import Location from '../location/Location.js'; import MediaQueryList from '../match-media/MediaQueryList.js'; import MutationRecord from '../mutation-observer/MutationRecord.js'; import MimeType from '../navigator/MimeType.js'; import MimeTypeArray from '../navigator/MimeTypeArray.js'; import Navigator from '../navigator/Navigator.js'; import Plugin from '../navigator/Plugin.js'; import PluginArray from '../navigator/PluginArray.js'; import Attr from '../nodes/attr/Attr.js'; import CharacterData from '../nodes/character-data/CharacterData.js'; import DocumentType from '../nodes/document-type/DocumentType.js'; import DocumentReadyStateEnum from '../nodes/document/DocumentReadyStateEnum.js'; import DocumentReadyStateManager from '../nodes/document/DocumentReadyStateManager.js'; import DOMRect from '../dom/DOMRect.js'; import DOMRectReadOnly from '../dom/DOMRectReadOnly.js'; import Element from '../nodes/element/Element.js'; import HTMLCollection from '../nodes/element/HTMLCollection.js'; import NamedNodeMap from '../nodes/element/NamedNodeMap.js'; import HTMLAnchorElement from '../nodes/html-anchor-element/HTMLAnchorElement.js'; import HTMLAreaElement from '../nodes/html-area-element/HTMLAreaElement.js'; import HTMLAudioElement from '../nodes/html-audio-element/HTMLAudioElement.js'; import HTMLBaseElement from '../nodes/html-base-element/HTMLBaseElement.js'; import HTMLBodyElement from '../nodes/html-body-element/HTMLBodyElement.js'; import HTMLBRElement from '../nodes/html-br-element/HTMLBRElement.js'; import HTMLButtonElement from '../nodes/html-button-element/HTMLButtonElement.js'; import HTMLCanvasElement from '../nodes/html-canvas-element/HTMLCanvasElement.js'; import HTMLDListElement from '../nodes/html-d-list-element/HTMLDListElement.js'; import HTMLDataElement from '../nodes/html-data-element/HTMLDataElement.js'; import HTMLDataListElement from '../nodes/html-data-list-element/HTMLDataListElement.js'; import HTMLDetailsElement from '../nodes/html-details-element/HTMLDetailsElement.js'; import HTMLDialogElement from '../nodes/html-dialog-element/HTMLDialogElement.js'; import HTMLDivElement from '../nodes/html-div-element/HTMLDivElement.js'; import HTMLElement from '../nodes/html-element/HTMLElement.js'; import HTMLEmbedElement from '../nodes/html-embed-element/HTMLEmbedElement.js'; import HTMLFieldSetElement from '../nodes/html-field-set-element/HTMLFieldSetElement.js'; import HTMLFormControlsCollection from '../nodes/html-form-element/HTMLFormControlsCollection.js'; import HTMLFormElement from '../nodes/html-form-element/HTMLFormElement.js'; import RadioNodeList from '../nodes/html-form-element/RadioNodeList.js'; import HTMLHeadElement from '../nodes/html-head-element/HTMLHeadElement.js'; import HTMLHeadingElement from '../nodes/html-heading-element/HTMLHeadingElement.js'; import HTMLHRElement from '../nodes/html-hr-element/HTMLHRElement.js'; import HTMLHtmlElement from '../nodes/html-html-element/HTMLHtmlElement.js'; import HTMLIFrameElement from '../nodes/html-iframe-element/HTMLIFrameElement.js'; import HTMLImageElement from '../nodes/html-image-element/HTMLImageElement.js'; import FileList from '../nodes/html-input-element/FileList.js'; import HTMLInputElement from '../nodes/html-input-element/HTMLInputElement.js'; import HTMLLabelElement from '../nodes/html-label-element/HTMLLabelElement.js'; import HTMLLegendElement from '../nodes/html-legend-element/HTMLLegendElement.js'; import HTMLLIElement from '../nodes/html-li-element/HTMLLIElement.js'; import HTMLLinkElement from '../nodes/html-link-element/HTMLLinkElement.js'; import HTMLMapElement from '../nodes/html-map-element/HTMLMapElement.js'; import HTMLMediaElement from '../nodes/html-media-element/HTMLMediaElement.js'; import TextTrackCueList from '../nodes/html-media-element/TextTrackCueList.js'; import TimeRanges from '../nodes/html-media-element/TimeRanges.js'; import HTMLMenuElement from '../nodes/html-menu-element/HTMLMenuElement.js'; import HTMLMetaElement from '../nodes/html-meta-element/HTMLMetaElement.js'; import HTMLMeterElement from '../nodes/html-meter-element/HTMLMeterElement.js'; import HTMLModElement from '../nodes/html-mod-element/HTMLModElement.js'; import HTMLOListElement from '../nodes/html-o-list-element/HTMLOListElement.js'; import HTMLObjectElement from '../nodes/html-object-element/HTMLObjectElement.js'; import HTMLOptGroupElement from '../nodes/html-opt-group-element/HTMLOptGroupElement.js'; import HTMLOptionElement from '../nodes/html-option-element/HTMLOptionElement.js'; import HTMLOutputElement from '../nodes/html-output-element/HTMLOutputElement.js'; import HTMLParagraphElement from '../nodes/html-paragraph-element/HTMLParagraphElement.js'; import HTMLParamElement from '../nodes/html-param-element/HTMLParamElement.js'; import HTMLPictureElement from '../nodes/html-picture-element/HTMLPictureElement.js'; import HTMLPreElement from '../nodes/html-pre-element/HTMLPreElement.js'; import HTMLProgressElement from '../nodes/html-progress-element/HTMLProgressElement.js'; import HTMLQuoteElement from '../nodes/html-quote-element/HTMLQuoteElement.js'; import HTMLScriptElement from '../nodes/html-script-element/HTMLScriptElement.js'; import HTMLSelectElement from '../nodes/html-select-element/HTMLSelectElement.js'; import HTMLSlotElement from '../nodes/html-slot-element/HTMLSlotElement.js'; import HTMLSourceElement from '../nodes/html-source-element/HTMLSourceElement.js'; import HTMLSpanElement from '../nodes/html-span-element/HTMLSpanElement.js'; import HTMLStyleElement from '../nodes/html-style-element/HTMLStyleElement.js'; import HTMLTableCaptionElement from '../nodes/html-table-caption-element/HTMLTableCaptionElement.js'; import HTMLTableCellElement from '../nodes/html-table-cell-element/HTMLTableCellElement.js'; import HTMLTableColElement from '../nodes/html-table-col-element/HTMLTableColElement.js'; import HTMLTableElement from '../nodes/html-table-element/HTMLTableElement.js'; import HTMLTableRowElement from '../nodes/html-table-row-element/HTMLTableRowElement.js'; import HTMLTableSectionElement from '../nodes/html-table-section-element/HTMLTableSectionElement.js'; import HTMLTemplateElement from '../nodes/html-template-element/HTMLTemplateElement.js'; import HTMLTextAreaElement from '../nodes/html-text-area-element/HTMLTextAreaElement.js'; import HTMLTimeElement from '../nodes/html-time-element/HTMLTimeElement.js'; import HTMLTitleElement from '../nodes/html-title-element/HTMLTitleElement.js'; import HTMLTrackElement from '../nodes/html-track-element/HTMLTrackElement.js'; import HTMLUListElement from '../nodes/html-u-list-element/HTMLUListElement.js'; import HTMLUnknownElement from '../nodes/html-unknown-element/HTMLUnknownElement.js'; import HTMLVideoElement from '../nodes/html-video-element/HTMLVideoElement.js'; import Node from '../nodes/node/Node.js'; import NodeList from '../nodes/node/NodeList.js'; import ProcessingInstruction from '../nodes/processing-instruction/ProcessingInstruction.js'; import ShadowRoot from '../nodes/shadow-root/ShadowRoot.js'; import SVGElement from '../nodes/svg-element/SVGElement.js'; import Permissions from '../permissions/Permissions.js'; import ResizeObserver from '../resize-observer/ResizeObserver.js'; import Screen from '../screen/Screen.js'; import Selection from '../selection/Selection.js'; import Storage from '../storage/Storage.js'; import NodeFilter from '../tree-walker/NodeFilter.js'; import NodeIterator from '../tree-walker/NodeIterator.js'; import TreeWalker from '../tree-walker/TreeWalker.js'; import URL from '../url/URL.js'; import ValidityState from '../validity-state/ValidityState.js'; import XMLSerializer from '../xml-serializer/XMLSerializer.js'; import VMGlobalPropertyScript from './VMGlobalPropertyScript.js'; import WindowPageOpenUtility from './WindowPageOpenUtility.js'; import { PerformanceObserver, PerformanceEntry } from 'node:perf_hooks'; import EventPhaseEnum from '../event/EventPhaseEnum.js'; import HTMLOptionsCollection from '../nodes/html-select-element/HTMLOptionsCollection.js'; import WindowContextClassExtender from './WindowContextClassExtender.js'; import WindowBrowserContext from './WindowBrowserContext.js'; import SVGSVGElement from '../nodes/svg-svg-element/SVGSVGElement.js'; import SVGGraphicsElement from '../nodes/svg-graphics-element/SVGGraphicsElement.js'; import SVGAnimateElement from '../nodes/svg-animate-element/SVGAnimateElement.js'; import SVGAnimateMotionElement from '../nodes/svg-animate-motion-element/SVGAnimateMotionElement.js'; import SVGAnimateTransformElement from '../nodes/svg-animate-transform-element/SVGAnimateTransformElement.js'; import SVGCircleElement from '../nodes/svg-circle-element/SVGCircleElement.js'; import SVGClipPathElement from '../nodes/svg-clip-path-element/SVGClipPathElement.js'; import SVGDefsElement from '../nodes/svg-defs-element/SVGDefsElement.js'; import SVGDescElement from '../nodes/svg-desc-element/SVGDescElement.js'; import SVGEllipseElement from '../nodes/svg-ellipse-element/SVGEllipseElement.js'; import SVGFEBlendElement from '../nodes/svg-fe-blend-element/SVGFEBlendElement.js'; import SVGFEColorMatrixElement from '../nodes/svg-fe-color-matrix-element/SVGFEColorMatrixElement.js'; import SVGFEComponentTransferElement from '../nodes/svg-fe-component-transfer-element/SVGFEComponentTransferElement.js'; import SVGFECompositeElement from '../nodes/svg-fe-composite-element/SVGFECompositeElement.js'; import SVGFEConvolveMatrixElement from '../nodes/svg-fe-convolve-matrix-element/SVGFEConvolveMatrixElement.js'; import SVGFEDiffuseLightingElement from '../nodes/svg-fe-diffuse-lighting-element/SVGFEDiffuseLightingElement.js'; import SVGFEDisplacementMapElement from '../nodes/svg-fe-displacement-map-element/SVGFEDisplacementMapElement.js'; import SVGFEDistantLightElement from '../nodes/svg-fe-distant-light-element/SVGFEDistantLightElement.js'; import SVGFEDropShadowElement from '../nodes/svg-fe-drop-shadow-element/SVGFEDropShadowElement.js'; import SVGFEFloodElement from '../nodes/svg-fe-flood-element/SVGFEFloodElement.js'; import SVGFEFuncAElement from '../nodes/svg-fe-func-a-element/SVGFEFuncAElement.js'; import SVGFEFuncBElement from '../nodes/svg-fe-func-b-element/SVGFEFuncBElement.js'; import SVGFEFuncGElement from '../nodes/svg-fe-func-g-element/SVGFEFuncGElement.js'; import SVGFEFuncRElement from '../nodes/svg-fe-func-r-element/SVGFEFuncRElement.js'; import SVGFEGaussianBlurElement from '../nodes/svg-fe-gaussian-blur-element/SVGFEGaussianBlurElement.js'; import SVGFEImageElement from '../nodes/svg-fe-image-element/SVGFEImageElement.js'; import SVGFEMergeElement from '../nodes/svg-fe-merge-element/SVGFEMergeElement.js'; import SVGFEMergeNodeElement from '../nodes/svg-fe-merge-node-element/SVGFEMergeNodeElement.js'; import SVGFEMorphologyElement from '../nodes/svg-fe-morphology-element/SVGFEMorphologyElement.js'; import SVGFEOffsetElement from '../nodes/svg-fe-offset-element/SVGFEOffsetElement.js'; import SVGFEPointLightElement from '../nodes/svg-fe-point-light-element/SVGFEPointLightElement.js'; import SVGFESpecularLightingElement from '../nodes/svg-fe-specular-lighting-element/SVGFESpecularLightingElement.js'; import SVGFESpotLightElement from '../nodes/svg-fe-spot-light-element/SVGFESpotLightElement.js'; import SVGFETileElement from '../nodes/svg-fe-tile-element/SVGFETileElement.js'; import SVGFETurbulenceElement from '../nodes/svg-fe-turbulence-element/SVGFETurbulenceElement.js'; import SVGFilterElement from '../nodes/svg-filter-element/SVGFilterElement.js'; import SVGForeignObjectElement from '../nodes/svg-foreign-object-element/SVGForeignObjectElement.js'; import SVGGElement from '../nodes/svg-g-element/SVGGElement.js'; import SVGImageElement from '../nodes/svg-image-element/SVGImageElement.js'; import SVGLineElement from '../nodes/svg-line-element/SVGLineElement.js'; import SVGLinearGradientElement from '../nodes/svg-linear-gradient-element/SVGLinearGradientElement.js'; import SVGMarkerElement from '../nodes/svg-marker-element/SVGMarkerElement.js'; import SVGMaskElement from '../nodes/svg-mask-element/SVGMaskElement.js'; import SVGMetadataElement from '../nodes/svg-metadata-element/SVGMetadataElement.js'; import SVGMPathElement from '../nodes/svg-m-path-element/SVGMPathElement.js'; import SVGPathElement from '../nodes/svg-path-element/SVGPathElement.js'; import SVGPatternElement from '../nodes/svg-pattern-element/SVGPatternElement.js'; import SVGPolygonElement from '../nodes/svg-polygon-element/SVGPolygonElement.js'; import SVGPolylineElement from '../nodes/svg-polyline-element/SVGPolylineElement.js'; import SVGRadialGradientElement from '../nodes/svg-radial-gradient-element/SVGRadialGradientElement.js'; import SVGRectElement from '../nodes/svg-rect-element/SVGRectElement.js'; import SVGScriptElement from '../nodes/svg-script-element/SVGScriptElement.js'; import SVGSetElement from '../nodes/svg-set-element/SVGSetElement.js'; import SVGStopElement from '../nodes/svg-stop-element/SVGStopElement.js'; import SVGStyleElement from '../nodes/svg-style-element/SVGStyleElement.js'; import SVGSwitchElement from '../nodes/svg-switch-element/SVGSwitchElement.js'; import SVGSymbolElement from '../nodes/svg-symbol-element/SVGSymbolElement.js'; import SVGTextElement from '../nodes/svg-text-element/SVGTextElement.js'; import SVGTextPathElement from '../nodes/svg-text-path-element/SVGTextPathElement.js'; import SVGTitleElement from '../nodes/svg-title-element/SVGTitleElement.js'; import SVGTSpanElement from '../nodes/svg-t-span-element/SVGTSpanElement.js'; import SVGUseElement from '../nodes/svg-use-element/SVGUseElement.js'; import SVGViewElement from '../nodes/svg-view-element/SVGViewElement.js'; import SVGAnimationElement from '../nodes/svg-animation-element/SVGAnimationElement.js'; import SVGComponentTransferFunctionElement from '../nodes/svg-component-transfer-function-element/SVGComponentTransferFunctionElement.js'; import SVGGeometryElement from '../nodes/svg-geometry-element/SVGGeometryElement.js'; import SVGGradientElement from '../nodes/svg-gradient-element/SVGGradientElement.js'; import SVGTextPositioningElement from '../nodes/svg-text-positioning-element/SVGTextPositioningElement.js'; import DOMMatrixReadOnly from '../dom/dom-matrix/DOMMatrixReadOnly.js'; import DOMMatrix from '../dom/dom-matrix/DOMMatrix.js'; import SVGAngle from '../svg/SVGAngle.js'; import SVGAnimatedAngle from '../svg/SVGAnimatedAngle.js'; import SVGAnimatedBoolean from '../svg/SVGAnimatedBoolean.js'; import SVGAnimatedEnumeration from '../svg/SVGAnimatedEnumeration.js'; import SVGAnimatedInteger from '../svg/SVGAnimatedInteger.js'; import SVGAnimatedLength from '../svg/SVGAnimatedLength.js'; import SVGLength from '../svg/SVGLength.js'; import SVGAnimatedNumber from '../svg/SVGAnimatedNumber.js'; import SVGAnimatedNumberList from '../svg/SVGAnimatedNumberList.js'; import SVGAnimatedPreserveAspectRatio from '../svg/SVGAnimatedPreserveAspectRatio.js'; import SVGAnimatedRect from '../svg/SVGAnimatedRect.js'; import SVGAnimatedString from '../svg/SVGAnimatedString.js'; import SVGAnimatedTransformList from '../svg/SVGAnimatedTransformList.js'; import SVGLengthList from '../svg/SVGLengthList.js'; import SVGMatrix from '../svg/SVGMatrix.js'; import SVGNumber from '../svg/SVGNumber.js'; import SVGNumberList from '../svg/SVGNumberList.js'; import SVGPoint from '../svg/SVGPoint.js'; import SVGPointList from '../svg/SVGPointList.js'; import SVGPreserveAspectRatio from '../svg/SVGPreserveAspectRatio.js'; import SVGRect from '../svg/SVGRect.js'; import SVGStringList from '../svg/SVGStringList.js'; import SVGTransform from '../svg/SVGTransform.js'; import SVGTransformList from '../svg/SVGTransformList.js'; import SVGUnitTypes from '../svg/SVGUnitTypes.js'; import DOMPoint from '../dom/DOMPoint.js'; import SVGAnimatedLengthList from '../svg/SVGAnimatedLengthList.js'; import CustomElementReactionStack from '../custom-element/CustomElementReactionStack.js'; const TIMER = { setTimeout: globalThis.setTimeout.bind(globalThis), clearTimeout: globalThis.clearTimeout.bind(globalThis), setInterval: globalThis.setInterval.bind(globalThis), clearInterval: globalThis.clearInterval.bind(globalThis), queueMicrotask: globalThis.queueMicrotask.bind(globalThis), setImmediate: globalThis.setImmediate.bind(globalThis), clearImmediate: globalThis.clearImmediate.bind(globalThis) }; const IS_NODE_JS_TIMEOUT_ENVIRONMENT = setTimeout.toString().includes('new Timeout'); /** * Class for PerformanceObserverEntryList as it is only available as an interface from Node.js. */ class PerformanceObserverEntryList { /** * Constructor. */ constructor() { throw new TypeError('Illegal constructor'); } } /** * Zero Timeout. */ class Timeout { callback; /** * Constructor. * @param callback Callback. */ constructor(callback) { this.callback = callback; } } /** * Browser window. * * Reference: * https://developer.mozilla.org/en-US/docs/Web/API/Window. */ export default class BrowserWindow extends EventTarget { // Nodes Node = Node; Attr = Attr; ShadowRoot = ShadowRoot; ProcessingInstruction = ProcessingInstruction; Element = Element; CharacterData = CharacterData; DocumentType = DocumentType; // HTML Element classes HTMLAnchorElement = HTMLAnchorElement; HTMLButtonElement = HTMLButtonElement; HTMLOptGroupElement = HTMLOptGroupElement; HTMLOptionElement = HTMLOptionElement; HTMLElement = HTMLElement; HTMLUnknownElement = HTMLUnknownElement; HTMLTemplateElement = HTMLTemplateElement; HTMLInputElement = HTMLInputElement; HTMLSelectElement = HTMLSelectElement; HTMLTextAreaElement = HTMLTextAreaElement; HTMLImageElement = HTMLImageElement; HTMLStyleElement = HTMLStyleElement; HTMLLabelElement = HTMLLabelElement; HTMLSlotElement = HTMLSlotElement; HTMLMetaElement = HTMLMetaElement; HTMLMediaElement = HTMLMediaElement; HTMLAudioElement = HTMLAudioElement; HTMLVideoElement = HTMLVideoElement; HTMLBaseElement = HTMLBaseElement; HTMLDialogElement = HTMLDialogElement; HTMLScriptElement = HTMLScriptElement; HTMLLinkElement = HTMLLinkElement; HTMLIFrameElement = HTMLIFrameElement; HTMLFormElement = HTMLFormElement; HTMLUListElement = HTMLUListElement; HTMLTrackElement = HTMLTrackElement; HTMLTableRowElement = HTMLTableRowElement; HTMLTitleElement = HTMLTitleElement; HTMLTimeElement = HTMLTimeElement; HTMLTableSectionElement = HTMLTableSectionElement; HTMLTableCellElement = HTMLTableCellElement; HTMLTableElement = HTMLTableElement; HTMLSpanElement = HTMLSpanElement; HTMLSourceElement = HTMLSourceElement; HTMLQuoteElement = HTMLQuoteElement; HTMLProgressElement = HTMLProgressElement; HTMLPreElement = HTMLPreElement; HTMLPictureElement = HTMLPictureElement; HTMLParamElement = HTMLParamElement; HTMLParagraphElement = HTMLParagraphElement; HTMLOutputElement = HTMLOutputElement; HTMLOListElement = HTMLOListElement; HTMLObjectElement = HTMLObjectElement; HTMLMeterElement = HTMLMeterElement; HTMLMenuElement = HTMLMenuElement; HTMLMapElement = HTMLMapElement; HTMLLIElement = HTMLLIElement; HTMLLegendElement = HTMLLegendElement; HTMLModElement = HTMLModElement; HTMLHtmlElement = HTMLHtmlElement; HTMLHRElement = HTMLHRElement; HTMLHeadElement = HTMLHeadElement; HTMLHeadingElement = HTMLHeadingElement; HTMLFieldSetElement = HTMLFieldSetElement; HTMLEmbedElement = HTMLEmbedElement; HTMLDListElement = HTMLDListElement; HTMLDivElement = HTMLDivElement; HTMLDetailsElement = HTMLDetailsElement; HTMLDataListElement = HTMLDataListElement; HTMLDataElement = HTMLDataElement; HTMLTableColElement = HTMLTableColElement; HTMLTableCaptionElement = HTMLTableCaptionElement; HTMLCanvasElement = HTMLCanvasElement; HTMLBRElement = HTMLBRElement; HTMLBodyElement = HTMLBodyElement; HTMLAreaElement = HTMLAreaElement; // SVG Element classes SVGSVGElement = SVGSVGElement; SVGAnimateElement = SVGAnimateElement; SVGAnimateMotionElement = SVGAnimateMotionElement; SVGAnimateTransformElement = SVGAnimateTransformElement; SVGCircleElement = SVGCircleElement; SVGClipPathElement = SVGClipPathElement; SVGDefsElement = SVGDefsElement; SVGDescElement = SVGDescElement; SVGEllipseElement = SVGEllipseElement; SVGFEBlendElement = SVGFEBlendElement; SVGFEColorMatrixElement = SVGFEColorMatrixElement; SVGFEComponentTransferElement = SVGFEComponentTransferElement; SVGFECompositeElement = SVGFECompositeElement; SVGFEConvolveMatrixElement = SVGFEConvolveMatrixElement; SVGFEDiffuseLightingElement = SVGFEDiffuseLightingElement; SVGFEDisplacementMapElement = SVGFEDisplacementMapElement; SVGFEDistantLightElement = SVGFEDistantLightElement; SVGFEDropShadowElement = SVGFEDropShadowElement; SVGFEFloodElement = SVGFEFloodElement; SVGFEFuncAElement = SVGFEFuncAElement; SVGFEFuncBElement = SVGFEFuncBElement; SVGFEFuncGElement = SVGFEFuncGElement; SVGFEFuncRElement = SVGFEFuncRElement; SVGFEGaussianBlurElement = SVGFEGaussianBlurElement; SVGFEImageElement = SVGFEImageElement; SVGFEMergeElement = SVGFEMergeElement; SVGFEMergeNodeElement = SVGFEMergeNodeElement; SVGFEMorphologyElement = SVGFEMorphologyElement; SVGFEOffsetElement = SVGFEOffsetElement; SVGFEPointLightElement = SVGFEPointLightElement; SVGFESpecularLightingElement = SVGFESpecularLightingElement; SVGFESpotLightElement = SVGFESpotLightElement; SVGFETileElement = SVGFETileElement; SVGFETurbulenceElement = SVGFETurbulenceElement; SVGFilterElement = SVGFilterElement; SVGForeignObjectElement = SVGForeignObjectElement; SVGGElement = SVGGElement; SVGImageElement = SVGImageElement; SVGLineElement = SVGLineElement; SVGLinearGradientElement = SVGLinearGradientElement; SVGMarkerElement = SVGMarkerElement; SVGMaskElement = SVGMaskElement; SVGMetadataElement = SVGMetadataElement; SVGMPathElement = SVGMPathElement; SVGPathElement = SVGPathElement; SVGPatternElement = SVGPatternElement; SVGPolygonElement = SVGPolygonElement; SVGPolylineElement = SVGPolylineElement; SVGRadialGradientElement = SVGRadialGradientElement; SVGRectElement = SVGRectElement; SVGScriptElement = SVGScriptElement; SVGSetElement = SVGSetElement; SVGStopElement = SVGStopElement; SVGStyleElement = SVGStyleElement; SVGSwitchElement = SVGSwitchElement; SVGSymbolElement = SVGSymbolElement; SVGTextElement = SVGTextElement; SVGTextPathElement = SVGTextPathElement; SVGTitleElement = SVGTitleElement; SVGTSpanElement = SVGTSpanElement; SVGUseElement = SVGUseElement; SVGViewElement = SVGViewElement; // Abstract SVG Element classes SVGElement = SVGElement; SVGAnimationElement = SVGAnimationElement; SVGComponentTransferFunctionElement = SVGComponentTransferFunctionElement; SVGGeometryElement = SVGGeometryElement; SVGGradientElement = SVGGradientElement; SVGTextPositioningElement = SVGTextPositioningElement; SVGGraphicsElement = SVGGraphicsElement; // Event classes Event = Event; UIEvent = UIEvent; CustomEvent = CustomEvent; AnimationEvent = AnimationEvent; KeyboardEvent = KeyboardEvent; MessageEvent = MessageEvent; MouseEvent = MouseEvent; PointerEvent = PointerEvent; FocusEvent = FocusEvent; WheelEvent = WheelEvent; InputEvent = InputEvent; ErrorEvent = ErrorEvent; StorageEvent = StorageEvent; SubmitEvent = SubmitEvent; ProgressEvent = ProgressEvent; MediaQueryListEvent = MediaQueryListEvent; HashChangeEvent = HashChangeEvent; ClipboardEvent = ClipboardEvent; TouchEvent = TouchEvent; Touch = Touch; // Non-implemented event classes AudioProcessingEvent = Event; BeforeInputEvent = Event; BeforeUnloadEvent = Event; BlobEvent = Event; CloseEvent = Event; CompositionEvent = Event; CSSFontFaceLoadEvent = Event; DeviceLightEvent = Event; DeviceMotionEvent = Event; DeviceOrientationEvent = Event; DeviceProximityEvent = Event; DOMTransactionEvent = Event; DragEvent = Event; EditingBeforeInputEvent = Event; FetchEvent = Event; GamepadEvent = Event; IDBVersionChangeEvent = Event; MediaStreamEvent = Event; MutationEvent = Event; OfflineAudioCompletionEvent = Event; OverconstrainedError = Event; PageTransitionEvent = Event; PaymentRequestUpdateEvent = Event; PopStateEvent = Event; RelatedEvent = Event; RTCDataChannelEvent = Event; RTCIdentityErrorEvent = Event; RTCIdentityEvent = Event; RTCPeerConnectionIceEvent = Event; SensorEvent = Event; SVGEvent = Event; SVGZoomEvent = Event; TimeEvent = Event; TrackEvent = Event; TransitionEvent = Event; UserProximityEvent = Event; WebGLContextEvent = Event; TextEvent = Event; // Other classes that don't have to be bound to the Window context Permissions = Permissions; History = History; Navigator = Navigator; Clipboard = Clipboard; TimeRanges = TimeRanges; TextTrackCueList = TextTrackCueList; ValidityState = ValidityState; MutationRecord = MutationRecord; IntersectionObserver = IntersectionObserver; IntersectionObserverEntry = IntersectionObserverEntry; CSSStyleDeclaration = CSSStyleDeclaration; CSSRule = CSSRule; CSSContainerRule = CSSContainerRule; CSSFontFaceRule = CSSFontFaceRule; CSSKeyframeRule = CSSKeyframeRule; CSSKeyframesRule = CSSKeyframesRule; CSSMediaRule = CSSMediaRule; CSSStyleRule = CSSStyleRule; CSSSupportsRule = CSSSupportsRule; DOMRect = DOMRect; DOMRectReadOnly = DOMRectReadOnly; Plugin = Plugin; PluginArray = PluginArray; Location = Location; CustomElementRegistry = CustomElementRegistry; ResizeObserver = ResizeObserver; URL = URL; Blob = Blob; File = File; Storage = Storage; MimeType = MimeType; MimeTypeArray = MimeTypeArray; NodeFilter = NodeFilter; HTMLCollection = HTMLCollection; HTMLFormControlCollection = HTMLFormControlsCollection; HTMLOptionsCollection = HTMLOptionsCollection; NodeList = NodeList; RadioNodeList = RadioNodeList; FileList = FileList; Screen = Screen; DOMMatrixReadOnly = DOMMatrixReadOnly; DOMMatrix = DOMMatrix; NamedNodeMap = NamedNodeMap; TreeWalker = TreeWalker; NodeIterator = NodeIterator; DataTransfer = DataTransfer; DataTransferItem = DataTransferItem; DataTransferItemList = DataTransferItemList; Headers = Headers; XMLSerializer = XMLSerializer; ClipboardItem = ClipboardItem; Selection = Selection; CSSUnitValue = CSSUnitValue; SVGAngle = SVGAngle; SVGAnimatedAngle = SVGAnimatedAngle; SVGAnimatedBoolean = SVGAnimatedBoolean; SVGAnimatedEnumeration = SVGAnimatedEnumeration; SVGAnimatedInteger = SVGAnimatedInteger; SVGAnimatedLength = SVGAnimatedLength; SVGAnimatedNumber = SVGAnimatedNumber; SVGAnimatedNumberList = SVGAnimatedNumberList; SVGAnimatedPreserveAspectRatio = SVGAnimatedPreserveAspectRatio; SVGAnimatedRect = SVGAnimatedRect; SVGAnimatedString = SVGAnimatedString; SVGAnimatedTransformList = SVGAnimatedTransformList; SVGLength = SVGLength; SVGLengthList = SVGLengthList; SVGMatrix = SVGMatrix; SVGNumber = SVGNumber; SVGNumberList = SVGNumberList; SVGPoint = SVGPoint; SVGPointList = SVGPointList; SVGPreserveAspectRatio = SVGPreserveAspectRatio; SVGRect = SVGRect; SVGStringList = SVGStringList; SVGTransform = SVGTransform; SVGTransformList = SVGTransformList; SVGAnimatedLengthList = SVGAnimatedLengthList; SVGUnitTypes = SVGUnitTypes; DOMPoint = DOMPoint; Window = this.constructor; // Node.js Classes URLSearchParams = URLSearchParams; WritableStream = Stream.Writable; ReadableStream = ReadableStream; TransformStream = Stream.Transform; PerformanceObserver = PerformanceObserver; PerformanceEntry = PerformanceEntry; PerformanceObserverEntryList = PerformanceObserverEntryList; // Events onload = null; onerror = null; // Public properties. document; customElements = new CustomElementRegistry(this); window = this; globalThis = this; performance = performance; screenLeft = 0; screenTop = 0; screenX = 0; screenY = 0; crypto = webcrypto; TextEncoder = TextEncoder; TextDecoder = TextDecoder; closed = false; console; name = ''; Buffer = Buffer; // Public internal properties // Used for tracking capture event listeners to improve performance when they are not used. // See EventTarget class. [PropertySymbol.mutationObservers] = []; [PropertySymbol.readyStateManager] = new DocumentReadyStateManager(this); [PropertySymbol.location]; [PropertySymbol.history]; [PropertySymbol.navigator]; [PropertySymbol.screen]; [PropertySymbol.sessionStorage]; [PropertySymbol.localStorage]; [PropertySymbol.self] = this; [PropertySymbol.top] = this; [PropertySymbol.parent] = this; [PropertySymbol.window] = this; [PropertySymbol.internalId] = -1; [PropertySymbol.customElementReactionStack] = new CustomElementReactionStack(this); [PropertySymbol.modules] = { json: new Map(), css: new Map(), esm: new Map() }; [PropertySymbol.moduleImportMap] = null; // Private properties #browserFrame; #innerWidth = null; #innerHeight = null; #outerWidth = null; #outerHeight = null; #devicePixelRatio = null; #zeroDelayTimeout = { timeouts: null }; #timerLoopStacks = []; /** * Constructor. * * @param browserFrame Browser frame. * @param [options] Options. * @param [options.url] URL. */ constructor(browserFrame, options) { super(); this.#browserFrame = browserFrame; this.console = browserFrame.page.console; this[PropertySymbol.navigator] = new Navigator(this); this[PropertySymbol.screen] = new Screen(); this[PropertySymbol.sessionStorage] = new Storage(); this[PropertySymbol.localStorage] = new Storage(); this[PropertySymbol.location] = new Location(this.#browserFrame, options?.url ?? 'about:blank'); this[PropertySymbol.history] = new History(this.#browserFrame, this); WindowBrowserContext.setWindowBrowserFrameRelation(this, this.#browserFrame); this[PropertySymbol.setupVMContext](); WindowContextClassExtender.extendClasses(this); // Document this.document = new this.HTMLDocument(); this.document[PropertySymbol.defaultView] = this; // Ready state manager this[PropertySymbol.readyStateManager].waitUntilComplete().then(() => { this.document[PropertySymbol.readyState] = DocumentReadyStateEnum.complete; this.document.dispatchEvent(new Event('readystatechange')); // Not sure why target is set to document here, but this is how it works in the browser const loadEvent = new Event('load'); loadEvent[PropertySymbol.currentTarget] = this; loadEvent[PropertySymbol.target] = this.document; loadEvent[PropertySymbol.eventPhase] = EventPhaseEnum.atTarget; this.dispatchEvent(loadEvent); loadEvent[PropertySymbol.currentTarget] = null; loadEvent[PropertySymbol.eventPhase] = EventPhaseEnum.none; loadEvent[PropertySymbol.dispatching] = false; }); this[PropertySymbol.bindMethods](); } /** * Returns self. * * @returns Self. */ get self() { return this[PropertySymbol.self]; } /** * Returns self. * * @param self Self. */ set self(self) { this[PropertySymbol.self] = self; } /** * Returns top. * * @returns Top. */ get top() { return this[PropertySymbol.top]; } /** * Returns parent. * * @returns Parent. */ get parent() { return this[PropertySymbol.parent]; } /** * Returns parent. * * @param parent Parent. */ set parent(parent) { this[PropertySymbol.parent] = parent; } /** * Returns location. */ get location() { return this[PropertySymbol.location]; } /** * Returns location. * * @param href Href. */ set location(href) { this[PropertySymbol.location].href = href; } /** * Returns history. */ get history() { return this[PropertySymbol.history]; } /** * Returns navigator. */ get navigator() { return this[PropertySymbol.navigator]; } /** * Returns screen. */ get screen() { return this[PropertySymbol.screen]; } /** * Returns session storage. */ get sessionStorage() { return this[PropertySymbol.sessionStorage]; } /** * Returns local storage. */ get localStorage() { return this[PropertySymbol.localStorage]; } /** * Returns opener. * * @returns Opener. */ get opener() { return this.#browserFrame[PropertySymbol.openerWindow]; } /** * The number of pixels that the document is currently scrolled horizontally. * * @returns Scroll X. */ get scrollX() { return this.document?.documentElement?.scrollLeft ?? 0; } /** * The read-only Window property pageXOffset is an alias for scrollX. * * @returns Scroll X. */ get pageXOffset() { return this.scrollX; } /** * The number of pixels that the document is currently scrolled vertically. * * @returns Scroll Y. */ get scrollY() { return this.document?.documentElement?.scrollTop ?? 0; } /** * The read-only Window property pageYOffset is an alias for scrollY. * * @returns Scroll Y. */ get pageYOffset() { return this.scrollY; } /** * The CSS interface holds useful CSS-related methods. * * @returns CSS interface. */ get CSS() { return new CSS(); } /** * Returns inner width. * * @returns Inner width. */ get innerWidth() { if (this.#innerWidth === null) { return this.#browserFrame.page.viewport.width; } return this.#innerWidth; } /** * Sets inner width. * * @param value Inner width. */ set innerWidth(value) { this.#innerWidth = value; } /** * Returns inner height. * * @returns Inner height. */ get innerHeight() { // It seems like this value can be defined according to spec, but changing it has no effect on the actual viewport. if (this.#innerHeight === null) { return this.#browserFrame.page.viewport.height; } return this.#innerHeight; } /** * Sets inner height. * * @param value Inner height. */ set innerHeight(value) { this.#innerHeight = value; } /** * Returns outer width. * * @returns Outer width. */ get outerWidth() { // It seems like this value can be defined according to spec, but changing it has no effect on the actual viewport. if (this.#outerWidth === null) { return this.#browserFrame.page.viewport.width; } return this.#outerWidth; } /** * Sets outer width. * * @param value Outer width. */ set outerWidth(value) { this.#outerWidth = value; } /** * Returns outer height. * * @returns Outer height. */ get outerHeight() { if (this.#outerHeight === null) { return this.#browserFrame.page.viewport.height; } return this.#outerHeight; } /** * Sets outer height. * * @param value Outer height. */ set outerHeight(value) { this.#outerHeight = value; } /** * Returns device pixel ratio. * * @returns Device pixel ratio. */ get devicePixelRatio() { // It seems like this value can be defined according to spec, but changing it has no effect on the actual viewport. if (this.#devicePixelRatio === null) { return this.#browserFrame.page.viewport.devicePixelRatio; } return this.#devicePixelRatio; } /** * Sets device pixel ratio. * * @param value Device pixel ratio. */ set devicePixelRatio(value) { this.#devicePixelRatio = value; } /** * Returns an object containing the values of all CSS properties of an element. * * @param element Element. * @returns CSS style declaration. */ getComputedStyle(element) { element[PropertySymbol.computedStyle] = element[PropertySymbol.computedStyle] || new CSSStyleDeclaration(PropertySymbol.illegalConstructor, this, { element, computed: true }); return element[PropertySymbol.computedStyle]; } /** * Returns selection. * * @returns Selection. */ getSelection() { return this.document.getSelection(); } /** * Scrolls to a particular set of coordinates. * * @param x X position or options object. * @param y Y position. */ scroll(x, y) { if (typeof x !== 'object' && arguments.length === 1) { throw new this.TypeError("Failed to execute 'scroll' on 'Window': The provided value is not of type 'ScrollToOptions'."); } const options = typeof x === 'object' ? x : { left: x, top: y }; if (options.behavior === 'smooth') { this.setTimeout(() => { if (options.top !== undefined) { const top = Number(options.top); this.document.documentElement.scrollTop = isNaN(top) ? 0 : top; } if (options.left !== undefined) { const left = Number(options.left); this.document.documentElement.scrollLeft = isNaN(left) ? 0 : left; } }); } else { if (options.top !== undefined) { const top = Number(options.top); this.document.documentElement.scrollTop = isNaN(top) ? 0 : top; } if (options.left !== undefined) { const left = Number(options.left); this.document.documentElement.scrollLeft = isNaN(left) ? 0 : left; } } } /** * Scrolls to a particular set of coordinates. * * @param x X position or options object. * @param y Y position. */ scrollTo(x, y) { if (typeof x !== 'object' && arguments.length === 1) { throw new this.TypeError("Failed to execute 'scrollTo' on 'Window': The provided value is not of type 'ScrollToOptions'."); } this.scroll(x, y); } /** * Scrolls by a relative amount from the current position. * * @param x Pixels to scroll by from top or scroll options object. * @param y Pixels to scroll by from left. */ scrollBy(x, y) { if (typeof x !== 'object' && arguments.length === 1) { throw new this.TypeError("Failed to execute 'scrollBy' on 'Window': The provided value is not of type 'ScrollToOptions'."); } const options = typeof x === 'object' ? x : { left: x, top: y }; this.scroll({ left: this.document.documentElement.scrollLeft + (options.left ?? 0), top: this.document.documentElement.scrollTop + (options.top ?? 0), behavior: options.behavior }); } /** * Shifts focus away from the window. */ blur() { // TODO: Implement. } /** * Gives focus to the window. */ focus() { // TODO: Implement. } /** * Loads a specified resource into a new or existing browsing context (that is, a tab, a window, or an iframe) under a specified name. * * @param [url] URL. * @param [target] Target. * @param [features] Window features. * @returns Window. */ open(url, target, features) { return WindowPageOpenUtility.openPage(this.#browserFrame, { url, target, features }); } /** * Closes the window. */ close() { // Users can only close the main frame of a page (not child frames such as iframes). if (this.#browserFrame.page.mainFrame === this.#browserFrame && // Scripts may close only the windows that were opened by them. // In this case, this means windows that have been opened by another window this.#browserFrame[PropertySymbol.openerWindow]) { this[PropertySymbol.destroy](); this.#browserFrame.page.close(); } } /** * Returns a new MediaQueryList object that can then be used to determine if the document matches the media query string. * * @param mediaQueryString A string specifying the media query to parse into a MediaQueryList. * @returns A new MediaQueryList. */ matchMedia(mediaQueryString) { return new MediaQueryList({ window: this, media: mediaQueryString }); } /** * Sets a timer which executes a function once the timer expires. * * @param callback Function to be executed. * @param [delay=0] Delay in ms. * @param args Arguments passed to the callback function. * @returns Timeout ID. */ setTimeout(callback, delay = 0, ...args) { if (this.closed) { return {}; } const settings = this.#browserFrame.page.context.browser.settings; if (settings.timer.preventTimerLoops) { const stack = new Error().stack; const timerLoopStacks = this.#timerLoopStacks; if (timerLoopStacks.includes(stack)) { return {}; } timerLoopStacks.push(stack); } // We can group timeouts with a delay of 0 into one timeout to improve performance. // Grouping timeouts will also improve the performance of the async task manager. // It also makes the async task manager more stable as many timeouts may cause waitUntilComplete() to be resolved too early. if (!delay) { const zeroDelayTimeout = this.#zeroDelayTimeout; if (!zeroDelayTimeout.timeouts) { const useTryCatch = !settings || (!settings.disableErrorCapturing && settings.errorCapture === BrowserErrorCaptureEnum.tryAndCatch); const id = TIMER.setTimeout(() => { // We need to call endTimer() before the callback as the callback might throw an error. this.#browserFrame[PropertySymbol.asyncTaskManager].endTimer(id); const timeouts = zeroDelayTimeout.timeouts; zeroDelayTimeout.timeouts = null; for (const timeout of timeouts) { if (useTryCatch) { let result; try { result = timeout.callback(); } catch (error) { this[PropertySymbol.dispatchError](error); } if (result instanceof Promise) { result.catch((error) => this[PropertySymbol.dispatchError](error)); } } else { timeout.callback(); } } }, 0); zeroDelayTimeout.timeouts = []; this.#browserFrame[PropertySymbol.asyncTaskManager].startTimer(id); } const timeout = new Timeout(() => callback(...args)); zeroDelayTimeout.timeouts.push(timeout); return timeout; } const useTryCatch = !settings || (!settings.disableErrorCapturing && settings.errorCapture === BrowserErrorCaptureEnum.tryAndCatch); const id = TIMER.setTimeout(() => { // We need to call endTimer() before the callback as the callback might throw an error. this.#browserFrame[PropertySymbol.asyncTaskManager].endTimer(id); if (useTryCatch) { let result; try { result = callback(...args); } catch (error) { this[PropertySymbol.dispatchError](error); } if (result instanceof Promise) { result.catch((error) => this[PropertySymbol.dispatchError](error)); } } else { callback(...args); } }, settings?.timer.maxTimeout !== -1 && delay && delay > settings?.timer.maxTimeout ? settings?.timer.maxTimeout : delay); this.#browserFrame[PropertySymbol.asyncTaskManager].startTimer(id); return id; } /** * Cancels a timeout previously established by calling setTimeout(). * * @param id ID of the timeout. */ clearTimeout(id) { if (id && id instanceof Timeout) { const zeroDelayTimeout = this.#zeroDelayTimeout; if (!zeroDelayTimeout.timeouts) { return; } const index = zeroDelayTimeout.timeouts.indexOf(id); if (index !== -1) { zeroDelayTimeout.timeouts.splice(index, 1); } return; } // We need to make sure that the ID is a Timeout object, otherwise Node.js might throw an error. // This is only necessary if we are in a Node.js environment. if (IS_NODE_JS_TIMEOUT_ENVIRONMENT && (!id || id.constructor.name !== 'Timeout')) { return; } TIMER.clearTimeout(id); this.#browserFrame[PropertySymbol.asyncTaskManager].endTimer(id); } /**