/* * Copyright (c) 2018-2023, Andreas Kling * Copyright (c) 2021, Tobias Christiansen * Copyright (c) 2021-2023, Sam Atkins * Copyright (c) 2022-2023, MacDue * * SPDX-License-Identifier: BSD-2-Clause */ #include "ImageStyleValue.h" #include #include #include #include #include #include #include #include #include namespace Web::CSS { ImageStyleValue::ImageStyleValue(AK::URL const& url) : AbstractImageStyleValue(Type::Image) , m_url(url) { } void ImageStyleValue::load_any_resources(DOM::Document& document) { if (m_image_request) return; m_document = &document; m_image_request = HTML::SharedImageRequest::get_or_create(document.realm(), *document.page(), m_url); m_image_request->add_callbacks( [this, weak_this = make_weak_ptr()] { if (!weak_this) return; if (!m_document) return; // FIXME: Do less than a full repaint if possible? if (auto navigable = m_document->navigable()) navigable->set_needs_display(); auto image_data = m_image_request->image_data(); if (image_data->is_animated() && image_data->frame_count() > 1) { m_timer = Platform::Timer::create(); m_timer->set_interval(image_data->frame_duration(0)); m_timer->on_timeout = [this] { animate(); }; m_timer->start(); } }, nullptr); if (m_image_request->needs_fetching()) { auto request = HTML::create_potential_CORS_request(document.vm(), m_url, Fetch::Infrastructure::Request::Destination::Image, HTML::CORSSettingAttribute::NoCORS); request->set_client(&document.relevant_settings_object()); m_image_request->fetch_image(document.realm(), request); } } void ImageStyleValue::animate() { if (!m_image_request) return; auto image_data = m_image_request->image_data(); if (!image_data) return; m_current_frame_index = (m_current_frame_index + 1) % image_data->frame_count(); auto current_frame_duration = image_data->frame_duration(m_current_frame_index); if (current_frame_duration != m_timer->interval()) m_timer->restart(current_frame_duration); if (m_current_frame_index == image_data->frame_count() - 1) { ++m_loops_completed; if (m_loops_completed > 0 && m_loops_completed == image_data->loop_count()) m_timer->stop(); } if (on_animate) on_animate(); } bool ImageStyleValue::is_paintable() const { return image_data(); } Gfx::ImmutableBitmap const* ImageStyleValue::bitmap(size_t frame_index, Gfx::IntSize size) const { if (auto image_data = this->image_data()) return image_data->bitmap(frame_index, size); return nullptr; } String ImageStyleValue::to_string() const { return serialize_a_url(MUST(m_url.to_string())); } bool ImageStyleValue::equals(StyleValue const& other) const { if (type() != other.type()) return false; return m_url == other.as_image().m_url; } Optional ImageStyleValue::natural_width() const { if (auto image_data = this->image_data()) return image_data->intrinsic_width(); return {}; } Optional ImageStyleValue::natural_height() const { if (auto image_data = this->image_data()) return image_data->intrinsic_height(); return {}; } void ImageStyleValue::paint(PaintContext& context, DevicePixelRect const& dest_rect, CSS::ImageRendering image_rendering) const { if (auto const* b = bitmap(m_current_frame_index, dest_rect.size().to_type()); b != nullptr) { auto scaling_mode = to_gfx_scaling_mode(image_rendering, b->rect(), dest_rect.to_type()); context.recording_painter().draw_scaled_immutable_bitmap(dest_rect.to_type(), *b, b->rect(), scaling_mode); } } RefPtr ImageStyleValue::image_data() const { if (!m_image_request) return nullptr; return m_image_request->image_data(); } }