
This replaces the previous Web::ImageDecoding::Decoder interface. While we're doing this, also move the SerenityOS implementation of this interface from LibWebView to WebContent. That means we no longer have to link with LibImageDecoderClient in applications that use a web view.
110 lines
2.8 KiB
C++
110 lines
2.8 KiB
C++
/*
|
|
* Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/Function.h>
|
|
#include <LibGfx/Bitmap.h>
|
|
#include <LibWeb/Loader/ImageResource.h>
|
|
#include <LibWeb/Platform/ImageCodecPlugin.h>
|
|
|
|
namespace Web {
|
|
|
|
NonnullRefPtr<ImageResource> ImageResource::convert_from_resource(Resource& resource)
|
|
{
|
|
return adopt_ref(*new ImageResource(resource));
|
|
}
|
|
|
|
ImageResource::ImageResource(LoadRequest const& request)
|
|
: Resource(Type::Image, request)
|
|
{
|
|
}
|
|
|
|
ImageResource::ImageResource(Resource& resource)
|
|
: Resource(Type::Image, resource)
|
|
{
|
|
}
|
|
|
|
ImageResource::~ImageResource() = default;
|
|
|
|
int ImageResource::frame_duration(size_t frame_index) const
|
|
{
|
|
decode_if_needed();
|
|
if (frame_index >= m_decoded_frames.size())
|
|
return 0;
|
|
return m_decoded_frames[frame_index].duration;
|
|
}
|
|
|
|
void ImageResource::decode_if_needed() const
|
|
{
|
|
if (!has_encoded_data())
|
|
return;
|
|
|
|
if (m_has_attempted_decode)
|
|
return;
|
|
|
|
if (!m_decoded_frames.is_empty())
|
|
return;
|
|
|
|
auto image = Platform::ImageCodecPlugin::the().decode_image(encoded_data());
|
|
|
|
if (image.has_value()) {
|
|
m_loop_count = image.value().loop_count;
|
|
m_animated = image.value().is_animated;
|
|
m_decoded_frames.resize(image.value().frames.size());
|
|
for (size_t i = 0; i < m_decoded_frames.size(); ++i) {
|
|
auto& frame = m_decoded_frames[i];
|
|
frame.bitmap = image.value().frames[i].bitmap;
|
|
frame.duration = image.value().frames[i].duration;
|
|
}
|
|
}
|
|
|
|
m_has_attempted_decode = true;
|
|
}
|
|
|
|
Gfx::Bitmap const* ImageResource::bitmap(size_t frame_index) const
|
|
{
|
|
decode_if_needed();
|
|
if (frame_index >= m_decoded_frames.size())
|
|
return nullptr;
|
|
return m_decoded_frames[frame_index].bitmap;
|
|
}
|
|
|
|
void ImageResource::update_volatility()
|
|
{
|
|
bool visible_in_viewport = false;
|
|
for_each_client([&](auto& client) {
|
|
if (static_cast<const ImageResourceClient&>(client).is_visible_in_viewport())
|
|
visible_in_viewport = true;
|
|
});
|
|
|
|
if (!visible_in_viewport) {
|
|
for (auto& frame : m_decoded_frames) {
|
|
if (frame.bitmap)
|
|
frame.bitmap->set_volatile();
|
|
}
|
|
return;
|
|
}
|
|
|
|
bool still_has_decoded_image = true;
|
|
for (auto& frame : m_decoded_frames) {
|
|
if (!frame.bitmap) {
|
|
still_has_decoded_image = false;
|
|
} else {
|
|
bool was_purged = false;
|
|
bool bitmap_has_memory = frame.bitmap->set_nonvolatile(was_purged);
|
|
if (!bitmap_has_memory || was_purged)
|
|
still_has_decoded_image = false;
|
|
}
|
|
}
|
|
if (still_has_decoded_image)
|
|
return;
|
|
|
|
m_decoded_frames.clear();
|
|
m_has_attempted_decode = false;
|
|
}
|
|
|
|
ImageResourceClient::~ImageResourceClient() = default;
|
|
|
|
}
|