blob: 81467b4d357ec8abffc3b809f68dbf16c6227f85 [file] [log] [blame]
// Copyright 2015 The Cobalt Authors. All Rights Reserved.
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// See the License for the specific language governing permissions and
// limitations under the License.
#include <memory>
#include <string>
#include <vector>
#include "base/callback.h"
#include "base/containers/hash_tables.h"
#include "base/memory/ref_counted.h"
#include "base/memory/weak_ptr.h"
#include "base/synchronization/waitable_event.h"
#include "base/timer/timer.h"
#include "cobalt/base/application_state.h"
#include "cobalt/base/clock.h"
#include "cobalt/cssom/css_parser.h"
#include "cobalt/cssom/css_style_declaration.h"
#include "cobalt/cssom/viewport_size.h"
#include "cobalt/dom/animation_frame_request_callback_list.h"
#include "cobalt/dom/application_lifecycle_state.h"
#include "cobalt/dom/captions/system_caption_settings.h"
#include "cobalt/dom/crypto.h"
#include "cobalt/dom/csp_delegate_type.h"
#include "cobalt/dom/dom_stat_tracker.h"
#include "cobalt/dom/event_target.h"
#include "cobalt/dom/html_element_context.h"
#include "cobalt/dom/location.h"
#include "cobalt/dom/media_query_list.h"
#include "cobalt/dom/on_screen_keyboard.h"
#include "cobalt/dom/on_screen_keyboard_bridge.h"
#include "cobalt/dom/parser.h"
#include "cobalt/dom/screenshot_manager.h"
#include "cobalt/dom/test_runner.h"
#include "cobalt/dom/url_registry.h"
#include "cobalt/dom/user_agent_platform_info.h"
#include "cobalt/dom/window_timers.h"
#include "cobalt/input/camera_3d.h"
#include "cobalt/loader/cors_preflight_cache.h"
#include "cobalt/loader/decoder.h"
#include "cobalt/loader/fetcher_factory.h"
#include "cobalt/loader/font/remote_typeface_cache.h"
#include "cobalt/loader/image/animated_image_tracker.h"
#include "cobalt/loader/image/image.h"
#include "cobalt/loader/image/image_cache.h"
#include "cobalt/loader/loader.h"
#include "cobalt/loader/loader_factory.h"
#include "cobalt/loader/mesh/mesh_cache.h"
#include "cobalt/media/can_play_type_handler.h"
#include "cobalt/media/web_media_player_factory.h"
#include "cobalt/network_bridge/cookie_jar.h"
#include "cobalt/network_bridge/net_poster.h"
#include "cobalt/script/callback_function.h"
#include "cobalt/script/environment_settings.h"
#include "cobalt/script/error_report.h"
#include "cobalt/script/execution_state.h"
#include "cobalt/script/script_runner.h"
#include "cobalt/script/script_value_factory.h"
#include "cobalt/ui_navigation/nav_item.h"
#include "starboard/window.h"
#include "url/gurl.h"
namespace cobalt {
namespace media_session {
class MediaSession;
} // namespace media_session
namespace speech {
class SpeechSynthesis;
} // namespace speech
} // namespace cobalt
namespace cobalt {
namespace dom {
class Camera3D;
class Document;
class Element;
class Event;
class History;
class LocalStorageDatabase;
class Location;
class MediaSource;
class Navigator;
class OnScreenKeyboard;
class Performance;
class Screen;
class Storage;
class WindowTimers;
// The window object represents a window containing a DOM document.
// TODO: Properly handle viewport resolution change event.
class Window : public EventTarget, public ApplicationLifecycleState::Observer {
typedef AnimationFrameRequestCallbackList::FrameRequestCallback
typedef WindowTimers::TimerCallback TimerCallback;
typedef base::Callback<void(const scoped_refptr<dom::Event>& event)>
typedef base::Callback<void(const scoped_refptr<dom::Event>& event)>
// Callback that will be called when window.close() is called. The
// base::TimeDelta parameter will contain the document's timeline time when
// close() was called.
typedef base::Callback<void(base::TimeDelta)> CloseCallback;
typedef UrlRegistry<MediaSource> MediaSourceRegistry;
typedef base::Callback<void(const std::string&,
const base::Optional<std::string>&)>
enum ClockType {
script::EnvironmentSettings* settings,
const cssom::ViewportSize& view_size,
base::ApplicationState initial_application_state,
cssom::CSSParser* css_parser, Parser* dom_parser,
loader::FetcherFactory* fetcher_factory,
loader::LoaderFactory* loader_factory,
render_tree::ResourceProvider** resource_provider,
loader::image::AnimatedImageTracker* animated_image_tracker,
loader::image::ImageCache* image_cache,
loader::font::RemoteTypefaceCache* remote_typeface_cache,
loader::mesh::MeshCache* mesh_cache,
LocalStorageDatabase* local_storage_database,
media::CanPlayTypeHandler* can_play_type_handler,
media::WebMediaPlayerFactory* web_media_player_factory,
script::ExecutionState* execution_state,
script::ScriptRunner* script_runner,
script::ScriptValueFactory* script_value_factory,
MediaSourceRegistry* media_source_registry,
DomStatTracker* dom_stat_tracker, const GURL& url,
const std::string& user_agent, UserAgentPlatformInfo* platform_info,
const std::string& language, const std::string& font_language_script,
const base::Callback<void(const GURL&)> navigation_callback,
const loader::Decoder::OnCompleteFunction& load_complete_callback,
network_bridge::CookieJar* cookie_jar,
const network_bridge::PostSender& post_sender,
csp::CSPHeaderPolicy require_csp,
dom::CspEnforcementType csp_enforcement_mode,
const base::Closure& csp_policy_changed_callback,
const base::Closure& ran_animation_frame_callbacks_callback,
const CloseCallback& window_close_callback,
const base::Closure& window_minimize_callback,
OnScreenKeyboardBridge* on_screen_keyboard_bridge,
const scoped_refptr<input::Camera3D>& camera_3d,
const OnStartDispatchEventCallback&
const OnStopDispatchEventCallback& stop_tracking_dispatch_event_callback,
const ScreenshotManager::ProvideScreenshotFunctionCallback&
base::WaitableEvent* synchronous_loader_interrupt,
bool enable_inline_script_warnings = false,
const scoped_refptr<ui_navigation::NavItem>& ui_nav_root = nullptr,
bool enable_map_to_mesh = true, int csp_insecure_allowed_token = 0,
int dom_max_element_depth = 0, float video_playback_rate_multiplier = 1.f,
ClockType clock_type = kClockTypeSystemTime,
const CacheCallback& splash_screen_cache_callback = CacheCallback(),
const scoped_refptr<captions::SystemCaptionSettings>& captions = nullptr,
bool log_tts = false);
// Web API: Window
scoped_refptr<Window> window() { return this; }
scoped_refptr<Window> self() { return this; }
const scoped_refptr<Document>& document() const;
const scoped_refptr<Location>& location() const;
const scoped_refptr<History>& history() const;
void Close();
void Minimize();
scoped_refptr<Window> frames() { return this; }
unsigned int length() { return 0; }
scoped_refptr<Window> top() { return this; }
scoped_refptr<Window> opener() { return this; }
scoped_refptr<Window> parent() { return this; }
scoped_refptr<Window> AnonymousIndexedGetter(unsigned int index) {
return NULL;
const scoped_refptr<Navigator>& navigator() const;
script::Handle<ScreenshotManager::InterfacePromise> Screenshot();
// Web API: CSSOM (partial interface)
// Returns the computed style of the given element, as described in
scoped_refptr<cssom::CSSStyleDeclaration> GetComputedStyle(
const scoped_refptr<Element>& elt);
scoped_refptr<cssom::CSSStyleDeclaration> GetComputedStyle(
const scoped_refptr<Element>& elt, const std::string& pseudo_elt);
// Web API: Timing control for script-based animations (partial interface)
int32 RequestAnimationFrame(
const AnimationFrameRequestCallbackList::FrameRequestCallbackArg&);
void CancelAnimationFrame(int32 handle);
float viewport_diagonal_inches() const {
return viewport_size_.diagonal_inches();
// Web API: CSSOM View Module (partial interface)
// Parses a media query.
scoped_refptr<MediaQueryList> MatchMedia(const std::string& query);
// As its name suggests, the Screen interface represents information about the
// screen of the output device.
const scoped_refptr<Screen>& screen();
// The innerWidth attribute must return the viewport width including the size
// of a rendered scroll bar (if any), or zero if there is no viewport.
float inner_width() const {
return static_cast<float>(viewport_size_.width());
// The innerHeight attribute must return the viewport height including the
// size of a rendered scroll bar (if any), or zero if there is no viewport.
float inner_height() const {
return static_cast<float>(viewport_size_.height());
// The screenX attribute must return the x-coordinate, relative to the origin
// of the screen of the output device, of the left of the client window as
// number of pixels, or zero if there is no such thing.
float screen_x() const { return 0.0f; }
// The screenY attribute must return the y-coordinate, relative to the origin
// of the screen of the output device, of the top of the client window as
// number of pixels, or zero if there is no such thing.
float screen_y() const { return 0.0f; }
// The outerWidth attribute must return the width of the client window.
float outer_width() const { return inner_width(); }
// The outerHeight attribute must return the height of the client window.
float outer_height() const { return inner_height(); }
// The devicePixelRatio attribute returns the ratio of CSS pixels per device
// pixel.
float device_pixel_ratio() const {
return viewport_size_.device_pixel_ratio();
// Web API: GlobalCrypto (implements)
scoped_refptr<Crypto> crypto() const;
// base64 encoding and decoding
std::string Btoa(const std::string& string_to_encode,
script::ExceptionState* exception_state);
std::vector<uint8_t> Atob(const std::string& encoded_string,
script::ExceptionState* exception_state);
// Web API: WindowTimers (implements)
int SetTimeout(const WindowTimers::TimerCallbackArg& handler) {
return SetTimeout(handler, 0);
int SetTimeout(const WindowTimers::TimerCallbackArg& handler, int timeout);
void ClearTimeout(int handle);
int SetInterval(const WindowTimers::TimerCallbackArg& handler) {
return SetInterval(handler, 0);
int SetInterval(const WindowTimers::TimerCallbackArg& handler, int timeout);
void ClearInterval(int handle);
void DestroyTimers();
// Web API: Storage (implements)
scoped_refptr<Storage> local_storage() const;
scoped_refptr<Storage> session_storage() const;
// Access to the Performance API (partial interface)
const scoped_refptr<Performance>& performance() const;
// Web API: SpeechSynthesisGetter (implements)
scoped_refptr<speech::SpeechSynthesis> speech_synthesis() const;
const scoped_refptr<Camera3D>& camera_3d() const;
const scoped_refptr<TestRunner>& test_runner() const;
void Gc(script::EnvironmentSettings* settings);
HTMLElementContext* html_element_context() const;
// Will fire the animation frame callbacks and reset the animation frame
// request callback list.
void RunAnimationFrameCallbacks();
bool HasPendingAnimationFrameCallbacks() const;
// Call this to inject an event into the window which will ultimately make
// its way to the appropriate object in DOM.
void InjectEvent(const scoped_refptr<Event>& event);
scoped_refptr<Window> opener() const { return NULL; }
// Sets the function to call to trigger a synchronous layout.
using SynchronousLayoutAndProduceRenderTreeCallback =
void SetSynchronousLayoutCallback(
const base::Closure& synchronous_layout_callback);
void SetSynchronousLayoutAndProduceRenderTreeCallback(
const SynchronousLayoutAndProduceRenderTreeCallback&
void SetSize(cssom::ViewportSize size);
void SetCamera3D(const scoped_refptr<input::Camera3D>& camera_3d);
void set_web_media_player_factory(
media::WebMediaPlayerFactory* web_media_player_factory) {
// Sets the current application state, forwarding on to the
// ApplicationLifecycleState associated with it and its document, causing
// precipitate events to be dispatched.
void SetApplicationState(base::ApplicationState state,
SbTimeMonotonic timestamp);
// Performs the steps specified for runtime script errors:
// Returns whether or not the script was handled.
bool ReportScriptError(const script::ErrorReport& error_report);
// ApplicationLifecycleState::Observer implementation.
void OnWindowFocusChanged(bool has_focus) override;
void OnVisibilityStateChanged(VisibilityState visibility_state) override;
void OnFrozennessChanged(bool is_frozen) override;
// Called when the document's root element has its offset dimensions requested
// and is unable to provide them.
void OnDocumentRootElementUnableToProvideOffsetDimensions();
void OnWindowOnOnlineEvent();
void OnWindowOnOfflineEvent();
// Cache the passed in splash screen content for the window.location URL.
void CacheSplashScreen(const std::string& content,
const base::Optional<std::string>& topic);
const scoped_refptr<loader::CORSPreflightCache> get_preflight_cache() {
return preflight_cache_;
// Custom on screen keyboard.
const scoped_refptr<OnScreenKeyboard>& on_screen_keyboard() const;
void ReleaseOnScreenKeyboard();
void OnStartDispatchEvent(const scoped_refptr<dom::Event>& event);
void OnStopDispatchEvent(const scoped_refptr<dom::Event>& event);
// |PointerState| will in general create reference cycles back to us, which is
// ok, as they are cleared over time. During shutdown however, since no
// more queue progress can possibly be made, we must forcibly clear the
// queue.
void ClearPointerStateForShutdown();
void TraceMembers(script::Tracer* tracer) override;
const scoped_refptr<ui_navigation::NavItem>& GetUiNavRoot() const {
return ui_nav_root_;
bool enable_map_to_mesh() { return enable_map_to_mesh_; }
const scoped_refptr<media_session::MediaSession> media_session() const;
void StartDocumentLoad(
loader::FetcherFactory* fetcher_factory, const GURL& url,
Parser* dom_parser,
const loader::Decoder::OnCompleteFunction& load_complete_callback);
scoped_refptr<base::BasicClock> MakePerformanceClock(
Window::ClockType clock_type);
class RelayLoadEvent;
~Window() override;
// From EventTarget.
std::string GetDebugName() override { return "Window"; }
void FireHashChangeEvent();
cssom::ViewportSize viewport_size_;
// A resize event can be pending if a resize occurs and the current visibility
// state is not visible. In this case, the resize event will run when the
// visibility state changes to visible.
bool is_resize_event_pending_;
// Whether or not the window is currently reporting a script error. This is
// used to prevent infinite recursion, because reporting the error causes an
// event to be dispatched, which can generate a new script error.
bool is_reporting_script_error_;
scoped_refptr<TestRunner> test_runner_;
scoped_refptr<Performance> performance_;
const std::unique_ptr<HTMLElementContext> html_element_context_;
scoped_refptr<Document> document_;
std::unique_ptr<loader::Loader> document_loader_;
scoped_refptr<History> history_;
scoped_refptr<Navigator> navigator_;
std::unique_ptr<RelayLoadEvent> relay_on_load_event_;
scoped_refptr<Camera3D> camera_3d_;
std::unique_ptr<WindowTimers> window_timers_;
scoped_refptr<Crypto> crypto_;
scoped_refptr<speech::SpeechSynthesis> speech_synthesis_;
scoped_refptr<Storage> local_storage_;
scoped_refptr<Storage> session_storage_;
scoped_refptr<Screen> screen_;
// Global preflight cache.
scoped_refptr<loader::CORSPreflightCache> preflight_cache_;
const base::Closure ran_animation_frame_callbacks_callback_;
const CloseCallback window_close_callback_;
const base::Closure window_minimize_callback_;
scoped_refptr<OnScreenKeyboard> on_screen_keyboard_;
CacheCallback splash_screen_cache_callback_;
OnStartDispatchEventCallback on_start_dispatch_event_callback_;
OnStopDispatchEventCallback on_stop_dispatch_event_callback_;
ScreenshotManager screenshot_manager_;
// This UI navigation root container should contain all active UI navigation
// items for this window.
scoped_refptr<ui_navigation::NavItem> ui_nav_root_;
bool enable_map_to_mesh_;
} // namespace dom
} // namespace cobalt