| /* |
| * Copyright (C) 2006, 2007, 2009, 2010 Apple Inc. All rights reserved. |
| * Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies) |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR |
| * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
| * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
| * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
| * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #ifndef THIRD_PARTY_BLINK_RENDERER_CORE_FRAME_LOCAL_DOM_WINDOW_H_ |
| #define THIRD_PARTY_BLINK_RENDERER_CORE_FRAME_LOCAL_DOM_WINDOW_H_ |
| |
| #include "services/metrics/public/cpp/ukm_recorder.h" |
| #include "services/metrics/public/cpp/ukm_source_id.h" |
| #include "third_party/blink/public/common/tokens/tokens.h" |
| #include "third_party/blink/renderer/bindings/core/v8/script_value.h" |
| #include "third_party/blink/renderer/core/core_export.h" |
| #include "third_party/blink/renderer/core/dom/document.h" |
| #include "third_party/blink/renderer/core/dom/events/event_target.h" |
| #include "third_party/blink/renderer/core/events/page_transition_event.h" |
| #include "third_party/blink/renderer/core/execution_context/execution_context.h" |
| #include "third_party/blink/renderer/core/frame/dom_window.h" |
| #include "third_party/blink/renderer/core/frame/local_frame.h" |
| #include "third_party/blink/renderer/core/geometry/dom_rect.h" |
| #include "third_party/blink/renderer/core/scroll/scrollable_area.h" |
| #include "third_party/blink/renderer/platform/heap/handle.h" |
| #include "third_party/blink/renderer/platform/supplementable.h" |
| #include "third_party/blink/renderer/platform/wtf/assertions.h" |
| #include "third_party/blink/renderer/platform/wtf/casting.h" |
| #include "third_party/blink/renderer/platform/wtf/forward.h" |
| |
| #include <memory> |
| |
| namespace blink { |
| |
| class ApplicationCache; |
| class BarProp; |
| class CSSStyleDeclaration; |
| class CustomElementRegistry; |
| class Document; |
| class DocumentInit; |
| class DOMSelection; |
| class DOMVisualViewport; |
| class Element; |
| class ExceptionState; |
| class External; |
| class FrameConsole; |
| class History; |
| class IdleRequestOptions; |
| class ImpressionParams; |
| class MediaQueryList; |
| class MessageEvent; |
| class Modulator; |
| class Navigator; |
| class Screen; |
| class ScriptController; |
| class ScriptPromise; |
| class ScriptState; |
| class ScrollToOptions; |
| class SecurityOrigin; |
| class SerializedScriptValue; |
| class SourceLocation; |
| class StyleMedia; |
| class TrustedTypePolicyFactory; |
| class V8FrameRequestCallback; |
| class V8IdleRequestCallback; |
| class V8VoidFunction; |
| class WindowAgent; |
| |
| enum PageTransitionEventPersistence { |
| kPageTransitionEventNotPersisted = 0, |
| kPageTransitionEventPersisted = 1 |
| }; |
| |
| // Note: if you're thinking of returning something DOM-related by reference, |
| // please ping dcheng@chromium.org first. You probably don't want to do that. |
| class CORE_EXPORT LocalDOMWindow final : public DOMWindow, |
| public ExecutionContext, |
| public Supplementable<LocalDOMWindow> { |
| USING_PRE_FINALIZER(LocalDOMWindow, Dispose); |
| |
| public: |
| class CORE_EXPORT EventListenerObserver : public GarbageCollectedMixin { |
| public: |
| virtual void DidAddEventListener(LocalDOMWindow*, const AtomicString&) = 0; |
| virtual void DidRemoveEventListener(LocalDOMWindow*, |
| const AtomicString&) = 0; |
| virtual void DidRemoveAllEventListeners(LocalDOMWindow*) = 0; |
| }; |
| |
| static LocalDOMWindow* From(const ScriptState*); |
| |
| LocalDOMWindow(LocalFrame&, WindowAgent*); |
| ~LocalDOMWindow() override; |
| |
| // Returns the token identifying the frame that this ExecutionContext was |
| // associated with at the moment of its creation. This remains valid even |
| // after the frame has been destroyed and the ExecutionContext is detached. |
| // This is used as a stable and persistent identifier for attributing detached |
| // context memory usage. |
| const LocalFrameToken& GetLocalFrameToken() const { return token_; } |
| ExecutionContextToken GetExecutionContextToken() const final { |
| return token_; |
| } |
| |
| LocalFrame* GetFrame() const { return To<LocalFrame>(DOMWindow::GetFrame()); } |
| |
| ScriptController& GetScriptController() const { return *script_controller_; } |
| |
| void Initialize(); |
| void ClearForReuse() { document_ = nullptr; } |
| |
| void ResetWindowAgent(WindowAgent*); |
| |
| mojom::blink::V8CacheOptions GetV8CacheOptions() const override; |
| |
| // Bind Content Security Policy to this window. This will cause the |
| // CSP to resolve the 'self' attribute and all policies will then be |
| // applied to this document. |
| void BindContentSecurityPolicy(); |
| |
| void Trace(Visitor*) const override; |
| |
| // ExecutionContext overrides: |
| bool IsWindow() const final { return true; } |
| bool IsContextThread() const final; |
| bool ShouldInstallV8Extensions() const final; |
| ContentSecurityPolicy* GetContentSecurityPolicyForWorld( |
| const DOMWrapperWorld* world) final; |
| const KURL& Url() const final; |
| const KURL& BaseURL() const final; |
| KURL CompleteURL(const String&) const final; |
| void DisableEval(const String& error_message) final; |
| String UserAgent() const final; |
| UserAgentMetadata GetUserAgentMetadata() const final; |
| HttpsState GetHttpsState() const final; |
| ResourceFetcher* Fetcher() const final; |
| bool CanExecuteScripts(ReasonForCallingCanExecuteScripts) final; |
| void ExceptionThrown(ErrorEvent*) final; |
| void AddInspectorIssue(mojom::blink::InspectorIssueInfoPtr) final; |
| EventTarget* ErrorEventTarget() final { return this; } |
| String OutgoingReferrer() const final; |
| network::mojom::ReferrerPolicy GetReferrerPolicy() const final; |
| CoreProbeSink* GetProbeSink() final; |
| const BrowserInterfaceBrokerProxy& GetBrowserInterfaceBroker() const final; |
| FrameOrWorkerScheduler* GetScheduler() final; |
| scoped_refptr<base::SingleThreadTaskRunner> GetTaskRunner(TaskType) final; |
| TrustedTypePolicyFactory* GetTrustedTypes() const final { |
| return GetTrustedTypesForWorld(*GetCurrentWorld()); |
| } |
| ScriptWrappable* ToScriptWrappable() final { return this; } |
| void CountPotentialFeaturePolicyViolation( |
| mojom::blink::FeaturePolicyFeature) const final; |
| void ReportFeaturePolicyViolation( |
| mojom::blink::FeaturePolicyFeature, |
| mojom::blink::PolicyDisposition, |
| const String& message = g_empty_string) const final; |
| void ReportDocumentPolicyViolation( |
| mojom::blink::DocumentPolicyFeature, |
| mojom::blink::PolicyDisposition, |
| const String& message = g_empty_string, |
| // If source_file is set to empty string, |
| // current JS file would be used as source_file instead. |
| const String& source_file = g_empty_string) const final; |
| |
| void AddConsoleMessageImpl(ConsoleMessage*, bool discard_duplicates) final; |
| |
| // UseCounter orverrides: |
| void CountUse(mojom::WebFeature feature) final; |
| |
| // Count |feature| only when this window is associated with a cross-origin |
| // iframe. |
| void CountUseOnlyInCrossOriginIframe(mojom::blink::WebFeature feature); |
| |
| Document* InstallNewDocument(const DocumentInit&); |
| |
| // EventTarget overrides: |
| ExecutionContext* GetExecutionContext() const override; |
| const LocalDOMWindow* ToLocalDOMWindow() const override; |
| LocalDOMWindow* ToLocalDOMWindow() override; |
| |
| // Same-origin DOM Level 0 |
| Screen* screen(); |
| History* history(); |
| BarProp* locationbar(); |
| BarProp* menubar(); |
| BarProp* personalbar(); |
| BarProp* scrollbars(); |
| BarProp* statusbar(); |
| BarProp* toolbar(); |
| Navigator* navigator(); |
| Navigator* clientInformation() { return navigator(); } |
| |
| bool offscreenBuffering() const; |
| |
| int outerHeight() const; |
| int outerWidth() const; |
| int innerHeight() const; |
| int innerWidth() const; |
| int screenX() const; |
| int screenY() const; |
| int screenLeft() const { return screenX(); } |
| int screenTop() const { return screenY(); } |
| double scrollX() const; |
| double scrollY() const; |
| double pageXOffset() const { return scrollX(); } |
| double pageYOffset() const { return scrollY(); } |
| |
| DOMVisualViewport* visualViewport(); |
| |
| HeapVector<Member<DOMRect>> getWindowSegments() const; |
| |
| const AtomicString& name() const; |
| void setName(const AtomicString&); |
| |
| String status() const; |
| void setStatus(const String&); |
| String defaultStatus() const; |
| void setDefaultStatus(const String&); |
| String origin() const; |
| |
| // DOM Level 2 AbstractView Interface |
| Document* document() const; |
| |
| // CSSOM View Module |
| StyleMedia* styleMedia(); |
| |
| // WebKit extensions |
| double devicePixelRatio() const; |
| |
| ApplicationCache* applicationCache(); |
| |
| // This is the interface orientation in degrees. Some examples are: |
| // 0 is straight up; -90 is when the device is rotated 90 clockwise; |
| // 90 is when rotated counter clockwise. |
| int orientation() const; |
| |
| DOMSelection* getSelection(); |
| |
| void blur() override; |
| void print(ScriptState*); |
| void stop(); |
| |
| void alert(ScriptState*, const String& message = String()); |
| bool confirm(ScriptState*, const String& message); |
| String prompt(ScriptState*, |
| const String& message, |
| const String& default_value); |
| |
| bool find(const String&, |
| bool case_sensitive, |
| bool backwards, |
| bool wrap, |
| bool whole_word, |
| bool search_in_frames, |
| bool show_dialog) const; |
| |
| // FIXME: ScrollBehaviorSmooth is currently unsupported in VisualViewport. |
| // crbug.com/434497 |
| void scrollBy(double x, double y) const; |
| void scrollBy(const ScrollToOptions*) const; |
| void scrollTo(double x, double y) const; |
| void scrollTo(const ScrollToOptions*) const; |
| void scroll(double x, double y) const { scrollTo(x, y); } |
| void scroll(const ScrollToOptions* scroll_to_options) const { |
| scrollTo(scroll_to_options); |
| } |
| void moveBy(int x, int y) const; |
| void moveTo(int x, int y) const; |
| |
| void resizeBy(int x, int y) const; |
| void resizeTo(int width, int height) const; |
| |
| MediaQueryList* matchMedia(const String&); |
| |
| // DOM Level 2 Style Interface |
| CSSStyleDeclaration* getComputedStyle( |
| Element*, |
| const String& pseudo_elt = String()) const; |
| |
| // Acessibility Object Model |
| ScriptPromise getComputedAccessibleNode(ScriptState*, Element*); |
| |
| // WebKit animation extensions |
| int requestAnimationFrame(V8FrameRequestCallback*); |
| int webkitRequestAnimationFrame(V8FrameRequestCallback*); |
| void cancelAnimationFrame(int id); |
| |
| // https://html.spec.whatwg.org/C/#windoworworkerglobalscope-mixin |
| void queueMicrotask(V8VoidFunction*); |
| |
| // https://wicg.github.io/origin-policy/#monkeypatch-html-windoworworkerglobalscope |
| const Vector<String>& originPolicyIds() const; |
| void SetOriginPolicyIds(const Vector<String>&); |
| |
| // https://html.spec.whatwg.org/C/#dom-originagentcluster |
| bool originAgentCluster() const; |
| |
| // Idle callback extensions |
| int requestIdleCallback(V8IdleRequestCallback*, const IdleRequestOptions*); |
| void cancelIdleCallback(int id); |
| |
| // Custom elements |
| CustomElementRegistry* customElements(ScriptState*) const; |
| CustomElementRegistry* customElements() const; |
| CustomElementRegistry* MaybeCustomElements() const; |
| |
| void SetModulator(Modulator*); |
| |
| // Obsolete APIs |
| void captureEvents() {} |
| void releaseEvents() {} |
| External* external(); |
| |
| bool isSecureContext() const; |
| |
| DEFINE_ATTRIBUTE_EVENT_LISTENER(search, kSearch) |
| |
| DEFINE_ATTRIBUTE_EVENT_LISTENER(orientationchange, kOrientationchange) |
| |
| void RegisterEventListenerObserver(EventListenerObserver*); |
| |
| void FrameDestroyed(); |
| void Reset(); |
| |
| Element* frameElement() const; |
| |
| DOMWindow* open(v8::Isolate*, |
| const String& url_string, |
| const AtomicString& target, |
| const String& features, |
| ExceptionState&); |
| |
| DOMWindow* open(v8::Isolate*, |
| const String& url_string, |
| const AtomicString& target, |
| const String& features, |
| bool unused, |
| ExceptionState&); |
| |
| DOMWindow* open(v8::Isolate*, |
| const String& url_string, |
| const AtomicString& target, |
| const String& features, |
| const ImpressionParams* impression_params, |
| ExceptionState&); |
| |
| FrameConsole* GetFrameConsole() const; |
| |
| void PrintErrorMessage(const String&) const; |
| |
| void DispatchPostMessage( |
| MessageEvent* event, |
| scoped_refptr<const SecurityOrigin> intended_target_origin, |
| std::unique_ptr<SourceLocation> location, |
| const base::UnguessableToken& source_agent_cluster_id); |
| |
| void DispatchMessageEventWithOriginCheck( |
| const SecurityOrigin* intended_target_origin, |
| MessageEvent*, |
| std::unique_ptr<SourceLocation>, |
| const base::UnguessableToken& source_agent_cluster_id); |
| |
| // Events |
| // EventTarget API |
| void RemoveAllEventListeners() override; |
| |
| using EventTarget::DispatchEvent; |
| DispatchEventResult DispatchEvent(Event&, EventTarget*); |
| |
| void FinishedLoading(FrameLoader::NavigationFinishState); |
| |
| // Dispatch the (deprecated) orientationchange event to this DOMWindow and |
| // recurse on its child frames. |
| void SendOrientationChangeEvent(); |
| |
| void EnqueueWindowEvent(Event&, TaskType); |
| void EnqueueDocumentEvent(Event&, TaskType); |
| void EnqueueNonPersistedPageshowEvent(); |
| void EnqueueHashchangeEvent(const String& old_url, const String& new_url); |
| void EnqueuePopstateEvent(scoped_refptr<SerializedScriptValue>); |
| void DispatchWindowLoadEvent(); |
| void DocumentWasClosed(); |
| void StatePopped(scoped_refptr<SerializedScriptValue>); |
| |
| void AcceptLanguagesChanged(); |
| |
| // https://dom.spec.whatwg.org/#dom-window-event |
| ScriptValue event(ScriptState*); |
| Event* CurrentEvent() const; |
| void SetCurrentEvent(Event*); |
| |
| TrustedTypePolicyFactory* trustedTypes(ScriptState*) const; |
| TrustedTypePolicyFactory* GetTrustedTypesForWorld( |
| const DOMWrapperWorld&) const; |
| |
| // Returns true if this window is cross-site to the main frame. Defaults to |
| // false in a detached window. |
| bool IsCrossSiteSubframe() const; |
| |
| void DispatchPersistedPageshowEvent(base::TimeTicks navigation_start); |
| |
| void DispatchPagehideEvent(PageTransitionEventPersistence persistence); |
| |
| InputMethodController& GetInputMethodController() const { |
| return *input_method_controller_; |
| } |
| TextSuggestionController& GetTextSuggestionController() const { |
| return *text_suggestion_controller_; |
| } |
| SpellChecker& GetSpellChecker() const { return *spell_checker_; } |
| |
| void ClearIsolatedWorldCSPForTesting(int32_t world_id); |
| |
| bool CrossOriginIsolatedCapability() const override; |
| |
| // These delegate to the document_. |
| ukm::UkmRecorder* UkmRecorder() override; |
| ukm::SourceId UkmSourceID() const override; |
| |
| protected: |
| // EventTarget overrides. |
| void AddedEventListener(const AtomicString& event_type, |
| RegisteredEventListener&) override; |
| void RemovedEventListener(const AtomicString& event_type, |
| const RegisteredEventListener&) override; |
| |
| // Protected DOMWindow overrides. |
| void SchedulePostMessage(MessageEvent*, |
| scoped_refptr<const SecurityOrigin> target, |
| LocalDOMWindow* source) override; |
| |
| private: |
| // Intentionally private to prevent redundant checks when the type is |
| // already LocalDOMWindow. |
| bool IsLocalDOMWindow() const override { return true; } |
| bool IsRemoteDOMWindow() const override { return false; } |
| |
| bool HasInsecureContextInAncestors() override; |
| |
| void Dispose(); |
| |
| void DispatchLoadEvent(); |
| |
| // Return the viewport size including scrollbars. |
| IntSize GetViewportSize() const; |
| |
| Member<ScriptController> script_controller_; |
| |
| Member<Document> document_; |
| Member<DOMVisualViewport> visualViewport_; |
| |
| bool should_print_when_finished_loading_; |
| |
| mutable Member<Screen> screen_; |
| mutable Member<History> history_; |
| mutable Member<BarProp> locationbar_; |
| mutable Member<BarProp> menubar_; |
| mutable Member<BarProp> personalbar_; |
| mutable Member<BarProp> scrollbars_; |
| mutable Member<BarProp> statusbar_; |
| mutable Member<BarProp> toolbar_; |
| mutable Member<Navigator> navigator_; |
| mutable Member<StyleMedia> media_; |
| mutable Member<CustomElementRegistry> custom_elements_; |
| // We store reference to Modulator here to have it TraceWrapper-ed. |
| // This is wrong, as Modulator is per-context, where as LocalDOMWindow is |
| // shared among context. However, this *works* as Modulator is currently only |
| // enabled in the main world, |
| Member<Modulator> modulator_; |
| Member<External> external_; |
| |
| String status_; |
| String default_status_; |
| |
| Vector<String> origin_policy_ids_; |
| |
| mutable Member<ApplicationCache> application_cache_; |
| |
| scoped_refptr<SerializedScriptValue> pending_state_object_; |
| |
| HeapHashSet<WeakMember<EventListenerObserver>> event_listener_observers_; |
| |
| // https://dom.spec.whatwg.org/#window-current-event |
| // We represent the "undefined" value as nullptr. |
| Member<Event> current_event_; |
| |
| // Store TrustedTypesPolicyFactory, per DOMWrapperWorld. |
| mutable HeapHashMap<scoped_refptr<const DOMWrapperWorld>, |
| Member<TrustedTypePolicyFactory>> |
| trusted_types_map_; |
| |
| // A dummy scheduler to return when the window is detached. |
| // All operations on it result in no-op, but due to this it's safe to |
| // use the returned value of GetScheduler() without additional checks. |
| // A task posted to a task runner obtained from one of its task runners |
| // will be forwarded to the default task runner. |
| // TODO(altimin): We should be able to remove it after we complete |
| // frame:document lifetime refactoring. |
| std::unique_ptr<FrameOrWorkerScheduler> detached_scheduler_; |
| |
| Member<InputMethodController> input_method_controller_; |
| Member<SpellChecker> spell_checker_; |
| Member<TextSuggestionController> text_suggestion_controller_; |
| |
| // Map from isolated world IDs to their ContentSecurityPolicy instances. |
| Member<HeapHashMap<int, Member<ContentSecurityPolicy>>> |
| isolated_world_csp_map_; |
| |
| // Tracks which features have already been potentially violated in this |
| // document. This helps to count them only once per page load. |
| // We don't use std::bitset to avoid to include feature_policy.mojom-blink.h. |
| mutable Vector<bool> potentially_violated_features_; |
| |
| // Token identifying the LocalFrame that this window was associated with at |
| // creation. Remains valid even after the frame is destroyed and the context |
| // is detached. |
| const LocalFrameToken token_; |
| |
| // Tracks which document policy violation reports have already been sent in |
| // this document, to avoid reporting duplicates. The value stored comes |
| // from |DocumentPolicyViolationReport::MatchId()|. |
| mutable HashSet<unsigned> document_policy_violation_reports_sent_; |
| |
| // A list of the most recently recorded source frame UKM source IDs for the |
| // PostMessage.Incoming.Frame UKM event, in order to partially deduplicate |
| // logged events. Its size is limited to 20. See SchedulePostMessage() where |
| // this UKM is logged. |
| // TODO(crbug.com/1112491): Remove when no longer needed. |
| Deque<ukm::SourceId> post_message_ukm_recorded_source_ids_; |
| }; |
| |
| template <> |
| struct DowncastTraits<LocalDOMWindow> { |
| static bool AllowFrom(const ExecutionContext& context) { |
| return context.IsWindow(); |
| } |
| static bool AllowFrom(const DOMWindow& window) { |
| return window.IsLocalDOMWindow(); |
| } |
| }; |
| |
| inline String LocalDOMWindow::status() const { |
| return status_; |
| } |
| |
| inline String LocalDOMWindow::defaultStatus() const { |
| return default_status_; |
| } |
| |
| } // namespace blink |
| |
| #endif // THIRD_PARTY_BLINK_RENDERER_CORE_FRAME_LOCAL_DOM_WINDOW_H_ |