/* * Copyright (c) 2023-2024, Aliaksandr Kalenik * * SPDX-License-Identifier: BSD-2-Clause */ #include #include namespace Web::Painting { DisplayListRecorder::DisplayListRecorder(DisplayList& command_list) : m_command_list(command_list) { } DisplayListRecorder::~DisplayListRecorder() = default; void DisplayListRecorder::append(Command&& command) { Optional scroll_frame_id; if (!m_scroll_frame_id_stack.is_empty()) scroll_frame_id = m_scroll_frame_id_stack.last(); m_command_list.append(move(command), scroll_frame_id); } void DisplayListRecorder::paint_nested_display_list(RefPtr display_list, Gfx::IntRect rect) { append(PaintNestedDisplayList { move(display_list), rect }); } void DisplayListRecorder::add_rounded_rect_clip(CornerRadii corner_radii, Gfx::IntRect border_rect, CornerClip corner_clip) { append(AddRoundedRectClip { corner_radii, border_rect, corner_clip }); } void DisplayListRecorder::add_mask(RefPtr display_list, Gfx::IntRect rect) { append(AddMask { move(display_list), rect }); } void DisplayListRecorder::fill_rect(Gfx::IntRect const& rect, Color color) { if (rect.is_empty()) return; append(FillRect { rect, color }); } void DisplayListRecorder::fill_path(FillPathUsingColorParams params) { auto aa_translation = params.translation.value_or(Gfx::FloatPoint {}); auto path_bounding_rect = params.path.bounding_box().translated(aa_translation).to_type(); if (path_bounding_rect.is_empty()) return; append(FillPathUsingColor { .path_bounding_rect = path_bounding_rect, .path = move(params.path), .color = params.color, .winding_rule = params.winding_rule, .aa_translation = aa_translation, }); } void DisplayListRecorder::fill_path(FillPathUsingPaintStyleParams params) { auto aa_translation = params.translation.value_or(Gfx::FloatPoint {}); auto path_bounding_rect = params.path.bounding_box().translated(aa_translation).to_type(); if (path_bounding_rect.is_empty()) return; append(FillPathUsingPaintStyle { .path_bounding_rect = path_bounding_rect, .path = move(params.path), .paint_style = params.paint_style, .winding_rule = params.winding_rule, .opacity = params.opacity, .aa_translation = aa_translation, }); } void DisplayListRecorder::stroke_path(StrokePathUsingColorParams params) { auto aa_translation = params.translation.value_or(Gfx::FloatPoint {}); auto path_bounding_rect = params.path.bounding_box().translated(aa_translation).to_type(); // Increase path bounding box by `thickness` to account for stroke. path_bounding_rect.inflate(params.thickness, params.thickness); if (path_bounding_rect.is_empty()) return; append(StrokePathUsingColor { .cap_style = params.cap_style, .join_style = params.join_style, .miter_limit = params.miter_limit, .dash_array = move(params.dash_array), .dash_offset = params.dash_offset, .path_bounding_rect = path_bounding_rect, .path = move(params.path), .color = params.color, .thickness = params.thickness, .aa_translation = aa_translation, }); } void DisplayListRecorder::stroke_path(StrokePathUsingPaintStyleParams params) { auto aa_translation = params.translation.value_or(Gfx::FloatPoint {}); auto path_bounding_rect = params.path.bounding_box().translated(aa_translation).to_type(); // Increase path bounding box by `thickness` to account for stroke. path_bounding_rect.inflate(params.thickness, params.thickness); if (path_bounding_rect.is_empty()) return; append(StrokePathUsingPaintStyle { .cap_style = params.cap_style, .join_style = params.join_style, .miter_limit = params.miter_limit, .dash_array = move(params.dash_array), .dash_offset = params.dash_offset, .path_bounding_rect = path_bounding_rect, .path = move(params.path), .paint_style = params.paint_style, .thickness = params.thickness, .opacity = params.opacity, .aa_translation = aa_translation, }); } void DisplayListRecorder::draw_ellipse(Gfx::IntRect const& a_rect, Color color, int thickness) { if (a_rect.is_empty()) return; append(DrawEllipse { .rect = a_rect, .color = color, .thickness = thickness, }); } void DisplayListRecorder::fill_ellipse(Gfx::IntRect const& a_rect, Color color) { if (a_rect.is_empty()) return; append(FillEllipse { a_rect, color }); } void DisplayListRecorder::fill_rect_with_linear_gradient(Gfx::IntRect const& gradient_rect, LinearGradientData const& data) { if (gradient_rect.is_empty()) return; append(PaintLinearGradient { gradient_rect, data }); } void DisplayListRecorder::fill_rect_with_conic_gradient(Gfx::IntRect const& rect, ConicGradientData const& data, Gfx::IntPoint const& position) { if (rect.is_empty()) return; append(PaintConicGradient { .rect = rect, .conic_gradient_data = data, .position = position }); } void DisplayListRecorder::fill_rect_with_radial_gradient(Gfx::IntRect const& rect, RadialGradientData const& data, Gfx::IntPoint center, Gfx::IntSize size) { if (rect.is_empty()) return; append(PaintRadialGradient { .rect = rect, .radial_gradient_data = data, .center = center, .size = size }); } void DisplayListRecorder::draw_rect(Gfx::IntRect const& rect, Color color, bool rough) { if (rect.is_empty()) return; append(DrawRect { .rect = rect, .color = color, .rough = rough }); } void DisplayListRecorder::draw_painting_surface(Gfx::IntRect const& dst_rect, NonnullRefPtr surface, Gfx::IntRect const& src_rect, Gfx::ScalingMode scaling_mode) { if (dst_rect.is_empty()) return; append(DrawPaintingSurface { .dst_rect = dst_rect, .surface = surface, .src_rect = src_rect, .scaling_mode = scaling_mode, }); } void DisplayListRecorder::draw_scaled_immutable_bitmap(Gfx::IntRect const& dst_rect, Gfx::ImmutableBitmap const& bitmap, Gfx::IntRect const& src_rect, Gfx::ScalingMode scaling_mode) { if (dst_rect.is_empty()) return; append(DrawScaledImmutableBitmap { .dst_rect = dst_rect, .bitmap = bitmap, .src_rect = src_rect, .scaling_mode = scaling_mode, }); } void DisplayListRecorder::draw_repeated_immutable_bitmap(Gfx::IntRect dst_rect, Gfx::IntRect clip_rect, NonnullRefPtr bitmap, Gfx::ScalingMode scaling_mode, DrawRepeatedImmutableBitmap::Repeat repeat) { append(DrawRepeatedImmutableBitmap { .dst_rect = dst_rect, .clip_rect = clip_rect, .bitmap = move(bitmap), .scaling_mode = scaling_mode, .repeat = repeat, }); } void DisplayListRecorder::draw_line(Gfx::IntPoint from, Gfx::IntPoint to, Color color, int thickness, Gfx::LineStyle style, Color alternate_color) { append(DrawLine { .color = color, .from = from, .to = to, .thickness = thickness, .style = style, .alternate_color = alternate_color, }); } void DisplayListRecorder::draw_text(Gfx::IntRect const& rect, String raw_text, Gfx::Font const& font, Gfx::TextAlignment alignment, Color color) { if (rect.is_empty()) return; auto glyph_run = Gfx::shape_text({}, 0, raw_text.code_points(), font, Gfx::GlyphRun::TextType::Ltr); float baseline_x = 0; if (alignment == Gfx::TextAlignment::CenterLeft) { baseline_x = rect.x(); } else if (alignment == Gfx::TextAlignment::Center) { baseline_x = static_cast(rect.x()) + (static_cast(rect.width()) - glyph_run->width()) / 2.0f; } else if (alignment == Gfx::TextAlignment::CenterRight) { baseline_x = static_cast(rect.right()) - glyph_run->width(); } else { // Unimplemented alignment. TODO(); } auto metrics = font.pixel_metrics(); float baseline_y = static_cast(rect.y()) + metrics.ascent + (static_cast(rect.height()) - (metrics.ascent + metrics.descent)) / 2.0f; draw_text_run(Gfx::IntPoint(roundf(baseline_x), roundf(baseline_y)), *glyph_run, color, rect, 1.0, Orientation::Horizontal); } void DisplayListRecorder::draw_text_run(Gfx::IntPoint baseline_start, Gfx::GlyphRun const& glyph_run, Color color, Gfx::IntRect const& rect, double scale, Orientation orientation) { if (rect.is_empty()) return; append(DrawGlyphRun { .glyph_run = glyph_run, .scale = scale, .rect = rect, .translation = baseline_start.to_type(), .color = color, .orientation = orientation, }); } void DisplayListRecorder::add_clip_rect(Gfx::IntRect const& rect) { append(AddClipRect { rect }); } void DisplayListRecorder::translate(Gfx::IntPoint delta) { append(Translate { delta }); } void DisplayListRecorder::save() { append(Save {}); } void DisplayListRecorder::restore() { append(Restore {}); } void DisplayListRecorder::push_scroll_frame_id(Optional id) { m_scroll_frame_id_stack.append(id); } void DisplayListRecorder::pop_scroll_frame_id() { (void)m_scroll_frame_id_stack.take_last(); } void DisplayListRecorder::push_stacking_context(PushStackingContextParams params) { append(PushStackingContext { .opacity = params.opacity, .source_paintable_rect = params.source_paintable_rect, .transform = { .origin = params.transform.origin, .matrix = params.transform.matrix, }, .clip_path = params.clip_path }); m_scroll_frame_id_stack.append({}); } void DisplayListRecorder::pop_stacking_context() { (void)m_scroll_frame_id_stack.take_last(); append(PopStackingContext {}); } void DisplayListRecorder::apply_backdrop_filter(Gfx::IntRect const& backdrop_region, BorderRadiiData const& border_radii_data, CSS::ResolvedFilter const& backdrop_filter) { if (backdrop_region.is_empty()) return; append(ApplyBackdropFilter { .backdrop_region = backdrop_region, .border_radii_data = border_radii_data, .backdrop_filter = backdrop_filter, }); } void DisplayListRecorder::paint_outer_box_shadow_params(PaintBoxShadowParams params) { append(PaintOuterBoxShadow { .box_shadow_params = params }); } void DisplayListRecorder::paint_inner_box_shadow_params(PaintBoxShadowParams params) { append(PaintInnerBoxShadow { .box_shadow_params = params }); } void DisplayListRecorder::paint_text_shadow(int blur_radius, Gfx::IntRect bounding_rect, Gfx::IntRect text_rect, Gfx::GlyphRun const& glyph_run, double glyph_run_scale, Color color, Gfx::IntPoint draw_location) { append(PaintTextShadow { .glyph_run = glyph_run, .glyph_run_scale = glyph_run_scale, .shadow_bounding_rect = bounding_rect, .text_rect = text_rect, .draw_location = draw_location, .blur_radius = blur_radius, .color = color }); } void DisplayListRecorder::fill_rect_with_rounded_corners(Gfx::IntRect const& rect, Color color, CornerRadius top_left_radius, CornerRadius top_right_radius, CornerRadius bottom_right_radius, CornerRadius bottom_left_radius) { if (rect.is_empty()) return; if (!top_left_radius && !top_right_radius && !bottom_right_radius && !bottom_left_radius) { fill_rect(rect, color); return; } append(FillRectWithRoundedCorners { .rect = rect, .color = color, .corner_radii = { .top_left = top_left_radius, .top_right = top_right_radius, .bottom_right = bottom_right_radius, .bottom_left = bottom_left_radius, }, }); } void DisplayListRecorder::fill_rect_with_rounded_corners(Gfx::IntRect const& a_rect, Color color, int radius) { if (a_rect.is_empty()) return; fill_rect_with_rounded_corners(a_rect, color, radius, radius, radius, radius); } void DisplayListRecorder::fill_rect_with_rounded_corners(Gfx::IntRect const& a_rect, Color color, int top_left_radius, int top_right_radius, int bottom_right_radius, int bottom_left_radius) { if (a_rect.is_empty()) return; fill_rect_with_rounded_corners(a_rect, color, { top_left_radius, top_left_radius }, { top_right_radius, top_right_radius }, { bottom_right_radius, bottom_right_radius }, { bottom_left_radius, bottom_left_radius }); } void DisplayListRecorder::draw_triangle_wave(Gfx::IntPoint a_p1, Gfx::IntPoint a_p2, Color color, int amplitude, int thickness = 1) { append(DrawTriangleWave { .p1 = a_p1, .p2 = a_p2, .color = color, .amplitude = amplitude, .thickness = thickness }); } void DisplayListRecorder::paint_scrollbar(int scroll_frame_id, Gfx::IntRect rect, CSSPixelFraction scroll_size, bool vertical) { append(PaintScrollBar { .scroll_frame_id = scroll_frame_id, .rect = rect, .scroll_size = scroll_size, .vertical = vertical }); } void DisplayListRecorder::apply_opacity(float opacity) { append(ApplyOpacity { .opacity = opacity }); } void DisplayListRecorder::apply_filters(CSS::ResolvedFilter filter) { append(ApplyFilters { .filter = filter }); } void DisplayListRecorder::apply_transform(Gfx::FloatPoint origin, Gfx::FloatMatrix4x4 matrix) { append(ApplyTransform { .origin = origin, .matrix = matrix, }); } void DisplayListRecorder::apply_mask_bitmap(Gfx::IntPoint origin, Gfx::ImmutableBitmap const& bitmap, Gfx::Bitmap::MaskKind kind) { append(ApplyMaskBitmap { .origin = origin, .bitmap = bitmap, .kind = kind, }); } }