#pragma once #include "Color.h" #include "Rect.h" #include "Size.h" #include #include #include #include #include #include class GraphicsBitmap : public RefCounted { public: enum class Format { Invalid, RGB32, RGBA32, Indexed8 }; static NonnullRefPtr create(Format, const Size&); static NonnullRefPtr create_wrapper(Format, const Size&, RGBA32*); static RefPtr load_from_file(const StringView& path); static RefPtr load_from_file(Format, const StringView& path, const Size&); static NonnullRefPtr create_with_shared_buffer(Format, NonnullRefPtr&&, const Size&); ~GraphicsBitmap(); RGBA32* scanline(int y); const RGBA32* scanline(int y) const; u8* bits(int y); const u8* bits(int y) const; Rect rect() const { return { {}, m_size }; } Size size() const { return m_size; } int width() const { return m_size.width(); } int height() const { return m_size.height(); } size_t pitch() const { return m_pitch; } int shared_buffer_id() const { return m_shared_buffer ? m_shared_buffer->shared_buffer_id() : -1; } unsigned bpp() const { switch (m_format) { case Format::Indexed8: return 8; case Format::RGB32: case Format::RGBA32: return 32; case Format::Invalid: return 0; default: ASSERT_NOT_REACHED(); } } void fill(Color); bool has_alpha_channel() const { return m_format == Format::RGBA32; } Format format() const { return m_format; } void set_mmap_name(const StringView&); size_t size_in_bytes() const { return m_pitch * m_size.height(); } Color palette_color(u8 index) const { return Color::from_rgba(m_palette[index]); } void set_palette_color(u8 index, Color color) { m_palette[index] = color.value(); } template Color get_pixel(int x, int y) const { ASSERT_NOT_REACHED(); } Color get_pixel(int x, int y) const; Color get_pixel(const Point& position) const { return get_pixel(position.x(), position.y()); } template void set_pixel(int x, int y, Color) { ASSERT_NOT_REACHED(); } void set_pixel(int x, int y, Color); void set_pixel(const Point& position, Color color) { set_pixel(position.x(), position.y(), color); } private: GraphicsBitmap(Format, const Size&); GraphicsBitmap(Format, const Size&, RGBA32*); GraphicsBitmap(Format, const Size&, MappedFile&&); GraphicsBitmap(Format, NonnullRefPtr&&, const Size&); Size m_size; RGBA32* m_data { nullptr }; RGBA32* m_palette { nullptr }; size_t m_pitch { 0 }; Format m_format { Format::Invalid }; bool m_needs_munmap { false }; MappedFile m_mapped_file; RefPtr m_shared_buffer; }; inline RGBA32* GraphicsBitmap::scanline(int y) { return reinterpret_cast((((u8*)m_data) + (y * m_pitch))); } inline const RGBA32* GraphicsBitmap::scanline(int y) const { return reinterpret_cast((((const u8*)m_data) + (y * m_pitch))); } inline const u8* GraphicsBitmap::bits(int y) const { return reinterpret_cast(scanline(y)); } inline u8* GraphicsBitmap::bits(int y) { return reinterpret_cast(scanline(y)); } template<> inline Color GraphicsBitmap::get_pixel(int x, int y) const { return Color::from_rgb(scanline(y)[x]); } template<> inline Color GraphicsBitmap::get_pixel(int x, int y) const { return Color::from_rgba(scanline(y)[x]); } template<> inline Color GraphicsBitmap::get_pixel(int x, int y) const { return Color::from_rgba(m_palette[bits(y)[x]]); } inline Color GraphicsBitmap::get_pixel(int x, int y) const { switch (m_format) { case Format::RGB32: return get_pixel(x, y); case Format::RGBA32: return get_pixel(x, y); case Format::Indexed8: return get_pixel(x, y); default: ASSERT_NOT_REACHED(); } } template<> inline void GraphicsBitmap::set_pixel(int x, int y, Color color) { scanline(y)[x] = color.value(); } template<> inline void GraphicsBitmap::set_pixel(int x, int y, Color color) { scanline(y)[x] = color.value(); } inline void GraphicsBitmap::set_pixel(int x, int y, Color color) { switch (m_format) { case Format::RGB32: set_pixel(x, y, color); break; case Format::RGBA32: set_pixel(x, y, color); break; case Format::Indexed8: ASSERT_NOT_REACHED(); default: ASSERT_NOT_REACHED(); } }