2019-01-09 04:06:04 +03:00
|
|
|
#pragma once
|
|
|
|
|
2019-01-10 07:36:32 +03:00
|
|
|
#include "Color.h"
|
2019-01-25 01:40:12 +03:00
|
|
|
#include "Rect.h"
|
2019-01-09 04:06:04 +03:00
|
|
|
#include "Size.h"
|
|
|
|
#include <AK/Retainable.h>
|
|
|
|
#include <AK/RetainPtr.h>
|
2019-02-08 01:13:47 +03:00
|
|
|
#include <AK/AKString.h>
|
2019-04-03 15:32:45 +03:00
|
|
|
#include <AK/MappedFile.h>
|
2019-03-08 14:22:55 +03:00
|
|
|
#include <SharedBuffer.h>
|
2019-01-14 22:00:42 +03:00
|
|
|
|
2019-01-09 04:06:04 +03:00
|
|
|
class GraphicsBitmap : public Retainable<GraphicsBitmap> {
|
|
|
|
public:
|
2019-05-06 20:32:56 +03:00
|
|
|
enum class Format { Invalid, RGB32, RGBA32, Indexed8 };
|
2019-02-19 03:42:53 +03:00
|
|
|
|
2019-02-25 14:43:52 +03:00
|
|
|
static Retained<GraphicsBitmap> create(Format, const Size&);
|
|
|
|
static Retained<GraphicsBitmap> create_wrapper(Format, const Size&, RGBA32*);
|
2019-03-22 02:19:53 +03:00
|
|
|
static RetainPtr<GraphicsBitmap> load_from_file(const String& path);
|
2019-02-19 03:42:53 +03:00
|
|
|
static RetainPtr<GraphicsBitmap> load_from_file(Format, const String& path, const Size&);
|
2019-03-17 06:23:54 +03:00
|
|
|
static Retained<GraphicsBitmap> create_with_shared_buffer(Format, Retained<SharedBuffer>&&, const Size&);
|
2019-01-09 04:06:04 +03:00
|
|
|
~GraphicsBitmap();
|
|
|
|
|
2019-01-10 07:36:32 +03:00
|
|
|
RGBA32* scanline(int y);
|
2019-01-12 08:39:34 +03:00
|
|
|
const RGBA32* scanline(int y) const;
|
2019-01-09 04:06:04 +03:00
|
|
|
|
2019-05-06 21:29:52 +03:00
|
|
|
byte* bits(int y);
|
2019-05-06 20:32:56 +03:00
|
|
|
const byte* bits(int y) const;
|
|
|
|
|
2019-01-25 01:40:12 +03:00
|
|
|
Rect rect() const { return { {}, m_size }; }
|
2019-01-09 04:06:04 +03:00
|
|
|
Size size() const { return m_size; }
|
|
|
|
int width() const { return m_size.width(); }
|
|
|
|
int height() const { return m_size.height(); }
|
2019-01-12 23:45:45 +03:00
|
|
|
size_t pitch() const { return m_pitch; }
|
2019-03-08 14:22:55 +03:00
|
|
|
int shared_buffer_id() const { return m_shared_buffer ? m_shared_buffer->shared_buffer_id() : -1; }
|
2019-02-16 14:13:43 +03:00
|
|
|
|
2019-02-19 03:42:53 +03:00
|
|
|
bool has_alpha_channel() const { return m_format == Format::RGBA32; }
|
2019-03-23 14:37:33 +03:00
|
|
|
Format format() const { return m_format; }
|
2019-02-19 03:42:53 +03:00
|
|
|
|
2019-04-30 14:46:03 +03:00
|
|
|
void set_mmap_name(const String&);
|
|
|
|
|
2019-05-08 04:28:13 +03:00
|
|
|
size_t size_in_bytes() const { return m_pitch * m_size.height(); }
|
2019-05-06 15:04:54 +03:00
|
|
|
|
2019-05-06 20:32:56 +03:00
|
|
|
Color palette_color(byte index) const { return Color::from_rgba(m_palette[index]); }
|
|
|
|
void set_palette_color(byte index, Color color) { m_palette[index] = color.value(); }
|
|
|
|
|
|
|
|
Color get_pixel(int x, int y) const
|
|
|
|
{
|
|
|
|
switch (m_format) {
|
|
|
|
case Format::RGB32:
|
|
|
|
return Color::from_rgb(scanline(y)[x]);
|
|
|
|
case Format::RGBA32:
|
|
|
|
return Color::from_rgba(scanline(y)[x]);
|
|
|
|
case Format::Indexed8:
|
|
|
|
return Color::from_rgba(m_palette[bits(y)[x]]);
|
|
|
|
default:
|
|
|
|
ASSERT_NOT_REACHED();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-09 04:06:04 +03:00
|
|
|
private:
|
2019-02-19 03:42:53 +03:00
|
|
|
GraphicsBitmap(Format, const Size&);
|
|
|
|
GraphicsBitmap(Format, const Size&, RGBA32*);
|
2019-04-03 15:32:45 +03:00
|
|
|
GraphicsBitmap(Format, const Size&, MappedFile&&);
|
2019-03-08 14:22:55 +03:00
|
|
|
GraphicsBitmap(Format, Retained<SharedBuffer>&&, const Size&);
|
2019-01-09 04:06:04 +03:00
|
|
|
|
|
|
|
Size m_size;
|
2019-01-10 07:36:32 +03:00
|
|
|
RGBA32* m_data { nullptr };
|
2019-05-06 20:32:56 +03:00
|
|
|
RGBA32* m_palette { nullptr };
|
2019-01-12 23:29:05 +03:00
|
|
|
size_t m_pitch { 0 };
|
2019-02-19 03:42:53 +03:00
|
|
|
Format m_format { Format::Invalid };
|
2019-04-26 19:25:05 +03:00
|
|
|
bool m_needs_munmap { false };
|
2019-04-03 15:32:45 +03:00
|
|
|
MappedFile m_mapped_file;
|
2019-03-08 14:22:55 +03:00
|
|
|
RetainPtr<SharedBuffer> m_shared_buffer;
|
2019-01-09 04:06:04 +03:00
|
|
|
};
|
2019-01-16 21:43:01 +03:00
|
|
|
|
|
|
|
inline RGBA32* GraphicsBitmap::scanline(int y)
|
|
|
|
{
|
|
|
|
return reinterpret_cast<RGBA32*>((((byte*)m_data) + (y * m_pitch)));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline const RGBA32* GraphicsBitmap::scanline(int y) const
|
|
|
|
{
|
|
|
|
return reinterpret_cast<const RGBA32*>((((const byte*)m_data) + (y * m_pitch)));
|
|
|
|
}
|
2019-05-06 20:32:56 +03:00
|
|
|
|
|
|
|
inline const byte* GraphicsBitmap::bits(int y) const
|
|
|
|
{
|
|
|
|
return reinterpret_cast<const byte*>(scanline(y));
|
|
|
|
}
|
2019-05-06 21:29:52 +03:00
|
|
|
|
|
|
|
inline byte* GraphicsBitmap::bits(int y)
|
|
|
|
{
|
|
|
|
return reinterpret_cast<byte*>(scanline(y));
|
|
|
|
}
|