/* * Copyright (c) 2020, Andreas Kling * All rights reserved. * * 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 THE COPYRIGHT HOLDERS AND CONTRIBUTORS "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 THE COPYRIGHT HOLDER 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. */ #include #include #include #include #include #include #include #include #include #include #include namespace Web::DOM { NonnullRefPtr Window::create_with_document(Document& document) { return adopt(*new Window(document)); } Window::Window(Document& document) : EventTarget(static_cast(document)) , m_document(document) , m_performance(make(*this)) { } Window::~Window() { } void Window::set_wrapper(Badge, Bindings::WindowObject& wrapper) { m_wrapper = wrapper.make_weak_ptr(); } void Window::alert(const String& message) { if (auto* page = m_document.page()) page->client().page_did_request_alert(message); } bool Window::confirm(const String& message) { auto confirm_result = GUI::MessageBox::show(nullptr, message, "Confirm", GUI::MessageBox::Type::Warning, GUI::MessageBox::InputType::OKCancel); return confirm_result == GUI::Dialog::ExecResult::ExecOK; } i32 Window::set_interval(JS::Function& callback, i32 interval) { auto timer = Timer::create_interval(*this, interval, callback); m_timers.set(timer->id(), timer); return timer->id(); } i32 Window::set_timeout(JS::Function& callback, i32 interval) { auto timer = Timer::create_timeout(*this, interval, callback); m_timers.set(timer->id(), timer); return timer->id(); } void Window::timer_did_fire(Badge, Timer& timer) { // We should not be here if there's no JS wrapper for the Window object. ASSERT(wrapper()); auto& vm = wrapper()->vm(); // NOTE: This protector pointer keeps the timer alive until the end of this function no matter what. NonnullRefPtr protector(timer); if (timer.type() == Timer::Type::Timeout) { m_timers.remove(timer.id()); } [[maybe_unused]] auto rc = vm.call(timer.callback(), wrapper()); if (vm.exception()) vm.clear_exception(); } i32 Window::allocate_timer_id(Badge) { return m_timer_id_allocator.allocate(); } void Window::deallocate_timer_id(Badge, i32 id) { m_timer_id_allocator.deallocate(id); } void Window::clear_timeout(i32 timer_id) { m_timers.remove(timer_id); } void Window::clear_interval(i32 timer_id) { m_timers.remove(timer_id); } i32 Window::request_animation_frame(JS::Function& callback) { // FIXME: This is extremely fake! static double fake_timestamp = 0; i32 link_id = GUI::DisplayLink::register_callback([handle = make_handle(&callback)](i32 link_id) { auto& function = const_cast(static_cast(*handle.cell())); auto& vm = function.vm(); fake_timestamp += 10; [[maybe_unused]] auto rc = vm.call(function, {}, JS::Value(fake_timestamp)); if (vm.exception()) vm.clear_exception(); GUI::DisplayLink::unregister_callback(link_id); }); // FIXME: Don't hand out raw DisplayLink ID's to JavaScript! return link_id; } void Window::cancel_animation_frame(i32 id) { // FIXME: We should not be passing untrusted numbers to DisplayLink::unregister_callback()! GUI::DisplayLink::unregister_callback(id); } void Window::did_set_location_href(Badge, const URL& new_href) { auto* frame = document().frame(); if (!frame) return; frame->loader().load(new_href, FrameLoader::Type::Navigation); } void Window::did_call_location_reload(Badge) { auto* frame = document().frame(); if (!frame) return; frame->loader().load(document().url(), FrameLoader::Type::Reload); } bool Window::dispatch_event(NonnullRefPtr event) { return EventDispatcher::dispatch(*this, event, true); } JS::Object* Window::create_wrapper(JS::GlobalObject& global_object) { return &global_object; } }