/* * Copyright (c) 2024, Aliaksandr Kalenik * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #ifdef AK_OS_MACOS # include #endif namespace Gfx { struct PaintingSurface::Impl { IntSize size; sk_sp surface; RefPtr bitmap; RefPtr context; }; NonnullRefPtr PaintingSurface::create_with_size(RefPtr context, Gfx::IntSize size, Gfx::BitmapFormat color_type, Gfx::AlphaType alpha_type) { auto sk_color_type = to_skia_color_type(color_type); auto sk_alpha_type = alpha_type == Gfx::AlphaType::Premultiplied ? kPremul_SkAlphaType : kUnpremul_SkAlphaType; auto image_info = SkImageInfo::Make(size.width(), size.height(), sk_color_type, sk_alpha_type, SkColorSpace::MakeSRGB()); if (!context) { auto bitmap = Gfx::Bitmap::create(color_type, alpha_type, size).value(); auto surface = SkSurfaces::WrapPixels(image_info, bitmap->begin(), bitmap->pitch()); VERIFY(surface); return adopt_ref(*new PaintingSurface(make(size, surface, bitmap, context))); } auto surface = SkSurfaces::RenderTarget(context->sk_context(), skgpu::Budgeted::kNo, image_info); VERIFY(surface); return adopt_ref(*new PaintingSurface(make(size, surface, nullptr, context))); } NonnullRefPtr PaintingSurface::wrap_bitmap(Bitmap& bitmap) { auto color_type = to_skia_color_type(bitmap.format()); auto alpha_type = bitmap.alpha_type() == Gfx::AlphaType::Premultiplied ? kPremul_SkAlphaType : kUnpremul_SkAlphaType; auto size = bitmap.size(); auto image_info = SkImageInfo::Make(bitmap.width(), bitmap.height(), color_type, alpha_type, SkColorSpace::MakeSRGB()); auto surface = SkSurfaces::WrapPixels(image_info, bitmap.begin(), bitmap.pitch()); return adopt_ref(*new PaintingSurface(make(size, surface, bitmap, nullptr))); } #ifdef AK_OS_MACOS NonnullRefPtr PaintingSurface::wrap_iosurface(Core::IOSurfaceHandle const& iosurface_handle, RefPtr context, Origin origin) { auto metal_texture = context->metal_context().create_texture_from_iosurface(iosurface_handle); IntSize const size { metal_texture->width(), metal_texture->height() }; auto image_info = SkImageInfo::Make(size.width(), size.height(), kBGRA_8888_SkColorType, kPremul_SkAlphaType, SkColorSpace::MakeSRGB()); GrMtlTextureInfo mtl_info; mtl_info.fTexture = sk_ret_cfp(metal_texture->texture()); auto backend_render_target = GrBackendRenderTargets::MakeMtl(metal_texture->width(), metal_texture->height(), mtl_info); GrSurfaceOrigin sk_origin; switch (origin) { case Origin::TopLeft: sk_origin = kTopLeft_GrSurfaceOrigin; break; case Origin::BottomLeft: sk_origin = kBottomLeft_GrSurfaceOrigin; break; default: VERIFY_NOT_REACHED(); } auto surface = SkSurfaces::WrapBackendRenderTarget(context->sk_context(), backend_render_target, sk_origin, kBGRA_8888_SkColorType, nullptr, nullptr); return adopt_ref(*new PaintingSurface(make(size, surface, nullptr, context))); } #endif PaintingSurface::PaintingSurface(NonnullOwnPtr&& impl) : m_impl(move(impl)) { } PaintingSurface::~PaintingSurface() = default; void PaintingSurface::read_into_bitmap(Gfx::Bitmap& bitmap) { auto color_type = to_skia_color_type(bitmap.format()); auto alpha_type = bitmap.alpha_type() == Gfx::AlphaType::Premultiplied ? kPremul_SkAlphaType : kUnpremul_SkAlphaType; auto image_info = SkImageInfo::Make(bitmap.width(), bitmap.height(), color_type, alpha_type, SkColorSpace::MakeSRGB()); SkPixmap const pixmap(image_info, bitmap.begin(), bitmap.pitch()); m_impl->surface->readPixels(pixmap, 0, 0); } void PaintingSurface::write_from_bitmap(Gfx::Bitmap const& bitmap) { auto color_type = to_skia_color_type(bitmap.format()); auto alpha_type = bitmap.alpha_type() == Gfx::AlphaType::Premultiplied ? kPremul_SkAlphaType : kUnpremul_SkAlphaType; auto image_info = SkImageInfo::Make(bitmap.width(), bitmap.height(), color_type, alpha_type, SkColorSpace::MakeSRGB()); SkPixmap const pixmap(image_info, bitmap.begin(), bitmap.pitch()); m_impl->surface->writePixels(pixmap, 0, 0); } IntSize PaintingSurface::size() const { return m_impl->size; } IntRect PaintingSurface::rect() const { return { {}, m_impl->size }; } SkCanvas& PaintingSurface::canvas() const { return *m_impl->surface->getCanvas(); } SkSurface& PaintingSurface::sk_surface() const { return *m_impl->surface; } void PaintingSurface::notify_content_will_change() { m_impl->surface->notifyContentWillChange(SkSurface::kDiscard_ContentChangeMode); } template<> sk_sp PaintingSurface::sk_image_snapshot() const { return m_impl->surface->makeImageSnapshot(); } void PaintingSurface::flush() const { if (auto context = m_impl->context) { context->flush_and_submit(m_impl->surface.get()); } } }