/* * Copyright (c) 2022, Luke Wilde * Copyright (c) 2024, Andrew Kaster * Copyright (c) 2023, Aliaksandr Kalenik * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Web::WebGL { GC_DEFINE_ALLOCATOR(WebGL2RenderingContext); JS::ThrowCompletionOr> WebGL2RenderingContext::create(JS::Realm& realm, HTML::HTMLCanvasElement& canvas_element, JS::Value options) { // We should be coming here from getContext being called on a wrapped element. auto context_attributes = TRY(convert_value_to_context_attributes_dictionary(canvas_element.vm(), options)); auto skia_backend_context = canvas_element.navigable()->traversable_navigable()->skia_backend_context(); if (!skia_backend_context) { fire_webgl_context_creation_error(canvas_element); return GC::Ptr { nullptr }; } auto context = OpenGLContext::create(*skia_backend_context); if (!context) { fire_webgl_context_creation_error(canvas_element); return GC::Ptr { nullptr }; } context->set_size(canvas_element.bitmap_size_for_canvas(1, 1)); return realm.create(realm, canvas_element, context.release_nonnull(), context_attributes, context_attributes); } WebGL2RenderingContext::WebGL2RenderingContext(JS::Realm& realm, HTML::HTMLCanvasElement& canvas_element, NonnullOwnPtr context, WebGLContextAttributes context_creation_parameters, WebGLContextAttributes actual_context_parameters) : PlatformObject(realm) , WebGL2RenderingContextImpl(realm, move(context)) , m_canvas_element(canvas_element) , m_context_creation_parameters(context_creation_parameters) , m_actual_context_parameters(actual_context_parameters) { } WebGL2RenderingContext::~WebGL2RenderingContext() = default; void WebGL2RenderingContext::initialize(JS::Realm& realm) { Base::initialize(realm); WEB_SET_PROTOTYPE_FOR_INTERFACE(WebGL2RenderingContext); } void WebGL2RenderingContext::visit_edges(Cell::Visitor& visitor) { Base::visit_edges(visitor); WebGL2RenderingContextImpl::visit_edges(visitor); visitor.visit(m_canvas_element); } void WebGL2RenderingContext::present() { if (!m_should_present) return; m_should_present = false; // "Before the drawing buffer is presented for compositing the implementation shall ensure that all rendering operations have been flushed to the drawing buffer." glFlush(); // "By default, after compositing the contents of the drawing buffer shall be cleared to their default values, as shown in the table above. // This default behavior can be changed by setting the preserveDrawingBuffer attribute of the WebGLContextAttributes object. // If this flag is true, the contents of the drawing buffer shall be preserved until the author either clears or overwrites them." if (!m_context_creation_parameters.preserve_drawing_buffer) { context().clear_buffer_to_default_values(); } } GC::Ref WebGL2RenderingContext::canvas_for_binding() const { return *m_canvas_element; } void WebGL2RenderingContext::needs_to_present() { m_should_present = true; if (!m_canvas_element->paintable()) return; m_canvas_element->paintable()->set_needs_display(); } void WebGL2RenderingContext::set_error(GLenum error) { auto context_error = glGetError(); if (context_error != GL_NO_ERROR) m_error = context_error; else m_error = error; } bool WebGL2RenderingContext::is_context_lost() const { dbgln_if(WEBGL_CONTEXT_DEBUG, "WebGLRenderingContext::is_context_lost()"); return m_context_lost; } Optional WebGL2RenderingContext::get_context_attributes() { if (is_context_lost()) return {}; return m_actual_context_parameters; } void WebGL2RenderingContext::set_size(Gfx::IntSize const& size) { Gfx::IntSize final_size; final_size.set_width(max(size.width(), 1)); final_size.set_height(max(size.height(), 1)); context().set_size(final_size); } void WebGL2RenderingContext::reset_to_default_state() { } RefPtr WebGL2RenderingContext::surface() { return context().surface(); } void WebGL2RenderingContext::allocate_painting_surface_if_needed() { context().allocate_painting_surface_if_needed(); } Optional> WebGL2RenderingContext::get_supported_extensions() { return context().get_supported_extensions(); } JS::Object* WebGL2RenderingContext::get_extension(String const&) { return nullptr; } WebIDL::Long WebGL2RenderingContext::drawing_buffer_width() const { auto size = canvas_for_binding()->bitmap_size_for_canvas(); return size.width(); } WebIDL::Long WebGL2RenderingContext::drawing_buffer_height() const { auto size = canvas_for_binding()->bitmap_size_for_canvas(); return size.height(); } }