mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-11-04 07:10:57 +00:00 
			
		
		
		
	Before this change, whenever ImageStyleValue had a non-null `m_image_request`, it was always leaked along with everything related to the document to which this value belongs. The issue arises due to the use of `JS::Handle` for the image request, as it introduces a cyclic dependency where `ImageRequest` prevents the `CSSStyleSheet`, that owns `ImageStyleValue`, from being deallocated: - ImageRequest - FetchController - FetchParams - Window - HTMLDocument - HTMLHtmlElement - HTMLBodyElement - Text - HTMLHeadElement - Text - HTMLMetaElement - Text - HTMLTitleElement - Text - HTMLStyleElement - CSSStyleSheet This change solves this by visiting `m_image_request` from `visit_edges` instead of introducing new heap root by using `JS::Handle`.
		
			
				
	
	
		
			69 lines
		
	
	
	
		
			2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			69 lines
		
	
	
	
		
			2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
 | 
						|
 * Copyright (c) 2021, Tobias Christiansen <tobyase@serenityos.org>
 | 
						|
 * Copyright (c) 2021-2023, Sam Atkins <atkinssj@serenityos.org>
 | 
						|
 * Copyright (c) 2022-2023, MacDue <macdue@dueutil.tech>
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier: BSD-2-Clause
 | 
						|
 */
 | 
						|
 | 
						|
#pragma once
 | 
						|
 | 
						|
#include <AK/URL.h>
 | 
						|
#include <LibJS/Heap/Cell.h>
 | 
						|
#include <LibJS/Heap/Handle.h>
 | 
						|
#include <LibWeb/CSS/Enums.h>
 | 
						|
#include <LibWeb/CSS/StyleValues/AbstractImageStyleValue.h>
 | 
						|
#include <LibWeb/HTML/SharedImageRequest.h>
 | 
						|
 | 
						|
namespace Web::CSS {
 | 
						|
 | 
						|
class ImageStyleValue final
 | 
						|
    : public AbstractImageStyleValue
 | 
						|
    , public Weakable<ImageStyleValue> {
 | 
						|
public:
 | 
						|
    static ValueComparingNonnullRefPtr<ImageStyleValue> create(AK::URL const& url)
 | 
						|
    {
 | 
						|
        return adopt_ref(*new (nothrow) ImageStyleValue(url));
 | 
						|
    }
 | 
						|
    virtual ~ImageStyleValue() override = default;
 | 
						|
 | 
						|
    void visit_edges(JS::Cell::Visitor& visitor) const
 | 
						|
    {
 | 
						|
        // FIXME: visit_edges in non-GC allocated classes is confusing pattern.
 | 
						|
        //        Consider making StyleValue to be GC allocated instead.
 | 
						|
        visitor.visit(m_image_request);
 | 
						|
    }
 | 
						|
 | 
						|
    virtual String to_string() const override;
 | 
						|
    virtual bool equals(StyleValue const& other) const override;
 | 
						|
 | 
						|
    virtual void load_any_resources(DOM::Document&) override;
 | 
						|
 | 
						|
    Optional<CSSPixels> natural_width() const override;
 | 
						|
    Optional<CSSPixels> natural_height() const override;
 | 
						|
 | 
						|
    virtual bool is_paintable() const override;
 | 
						|
    void paint(PaintContext& context, DevicePixelRect const& dest_rect, CSS::ImageRendering image_rendering) const override;
 | 
						|
 | 
						|
    Function<void()> on_animate;
 | 
						|
 | 
						|
    RefPtr<HTML::DecodedImageData const> image_data() const;
 | 
						|
 | 
						|
private:
 | 
						|
    ImageStyleValue(AK::URL const&);
 | 
						|
 | 
						|
    JS::GCPtr<HTML::SharedImageRequest> m_image_request;
 | 
						|
 | 
						|
    void animate();
 | 
						|
    Gfx::Bitmap const* bitmap(size_t frame_index, Gfx::IntSize = {}) const;
 | 
						|
 | 
						|
    AK::URL m_url;
 | 
						|
    WeakPtr<DOM::Document> m_document;
 | 
						|
 | 
						|
    size_t m_current_frame_index { 0 };
 | 
						|
    size_t m_loops_completed { 0 };
 | 
						|
    RefPtr<Platform::Timer> m_timer;
 | 
						|
};
 | 
						|
 | 
						|
}
 |