aseprite/src/doc/image_impl.h
David Capello da1358c5dc Unify all render code in one library
Changes:
* Create render library (move util/render.cpp to render/render.cpp)
* Move app::Zoom class to render::Zoom
* Remove doc::Image::merge() member function
* Add gfx::Clip helper class (to clip dst/src rectangles before a blit)
* Move doc::composite_image() to render::composite_image()
* Remove doc::Sprite::render()
* Replace Sprite::getPixel() with render::get_sprite_pixel()
* Remove doc::layer_render() function
* Convert DitheringMethod to a enum class
* Add AppRender to configure a render::Render with the app configuration
* Move checked background preferences as document-specific configuration
* Add doc::Sprite::layer() and palette() member functions
* Add doc::Layer::cel() member function
* Add doc::Palette::entry() member function()
* Add doc::frame_t type
* Move create_palette_from_rgb/convert_pixel_format to render library
* ExportSpriteSheet doesn't need a temporary image now that we can specify
  the source rectangle in the render routine
2014-12-28 11:06:11 -03:00

277 lines
7.3 KiB
C++

// Aseprite Document Library
// Copyright (c) 2001-2014 David Capello
//
// This file is released under the terms of the MIT license.
// Read LICENSE.txt for more information.
#ifndef DOC_IMAGE_IMPL_H_INCLUDED
#define DOC_IMAGE_IMPL_H_INCLUDED
#pragma once
#include "doc/blend.h"
#include "doc/image.h"
#include "doc/image_bits.h"
#include "doc/image_iterator.h"
#include "doc/palette.h"
namespace doc {
template<class Traits>
class ImageImpl : public Image {
private:
typedef typename Traits::address_t address_t;
typedef typename Traits::const_address_t const_address_t;
ImageBufferPtr m_buffer;
address_t m_bits;
address_t* m_rows;
inline address_t getBitsAddress() {
return m_bits;
}
inline const_address_t getBitsAddress() const {
return m_bits;
}
inline address_t getLineAddress(int y) {
ASSERT(y >= 0 && y < height());
return m_rows[y];
}
inline const_address_t getLineAddress(int y) const {
ASSERT(y >= 0 && y < height());
return m_rows[y];
}
public:
inline address_t address(int x, int y) const {
return (address_t)(m_rows[y] + x / (Traits::pixels_per_byte == 0 ? 1 : Traits::pixels_per_byte));
}
ImageImpl(int width, int height,
const ImageBufferPtr& buffer)
: Image(static_cast<PixelFormat>(Traits::pixel_format), width, height)
, m_buffer(buffer)
{
size_t for_rows = sizeof(address_t) * height;
size_t rowstride_bytes = Traits::getRowStrideBytes(width);
size_t required_size = for_rows + rowstride_bytes*height;
if (!m_buffer)
m_buffer.reset(new ImageBuffer(required_size));
else
m_buffer->resizeIfNecessary(required_size);
m_rows = (address_t*)m_buffer->buffer();
m_bits = (address_t)(m_buffer->buffer() + for_rows);
address_t addr = m_bits;
for (int y=0; y<height; ++y) {
m_rows[y] = addr;
addr = (address_t)(((uint8_t*)addr) + rowstride_bytes);
}
}
uint8_t* getPixelAddress(int x, int y) const override {
ASSERT(x >= 0 && x < width());
ASSERT(y >= 0 && y < height());
return (uint8_t*)address(x, y);
}
color_t getPixel(int x, int y) const override {
ASSERT(x >= 0 && x < width());
ASSERT(y >= 0 && y < height());
return *address(x, y);
}
void putPixel(int x, int y, color_t color) override {
ASSERT(x >= 0 && x < width());
ASSERT(y >= 0 && y < height());
*address(x, y) = color;
}
void clear(color_t color) override {
LockImageBits<Traits> bits(this);
typename LockImageBits<Traits>::iterator it(bits.begin());
typename LockImageBits<Traits>::iterator end(bits.end());
for (; it != end; ++it)
*it = color;
}
void copy(const Image* _src, gfx::Clip area) override {
const ImageImpl<Traits>* src = (const ImageImpl<Traits>*)_src;
address_t src_address;
address_t dst_address;
int bytes;
if (!area.clip(width(), height(), src->width(), src->height()))
return;
// Copy process
bytes = Traits::getRowStrideBytes(area.size.w);
for (int end_y=area.dst.y+area.size.h;
area.dst.y<end_y;
++area.dst.y, ++area.src.y) {
src_address = src->address(area.src.x, area.src.y);
dst_address = address(area.dst.x, area.dst.y);
memcpy(dst_address, src_address, bytes);
}
}
void drawHLine(int x1, int y, int x2, color_t color) override {
LockImageBits<Traits> bits(this, gfx::Rect(x1, y, x2 - x1 + 1, 1));
typename LockImageBits<Traits>::iterator it(bits.begin());
typename LockImageBits<Traits>::iterator end(bits.end());
for (; it != end; ++it)
*it = color;
}
void fillRect(int x1, int y1, int x2, int y2, color_t color) override {
for (int y=y1; y<=y2; ++y)
ImageImpl<Traits>::drawHLine(x1, y, x2, color);
}
void blendRect(int x1, int y1, int x2, int y2, color_t color, int opacity) override {
fillRect(x1, y1, x2, y2, color);
}
private:
bool clip_rects(const Image* src, int& dst_x, int& dst_y, int& src_x, int& src_y, int& w, int& h) const {
// Clip with destionation image
if (dst_x < 0) {
w += dst_x;
src_x -= dst_x;
dst_x = 0;
}
if (dst_y < 0) {
h += dst_y;
src_y -= dst_y;
dst_y = 0;
}
if (dst_x+w > width()) {
w = width() - dst_x;
}
if (dst_y+h > height()) {
h = height() - dst_y;
}
// Clip with source image
if (src_x < 0) {
w += src_x;
dst_x -= src_x;
src_x = 0;
}
if (src_y < 0) {
h += src_y;
dst_y -= src_y;
src_y = 0;
}
if (src_x+w > src->width()) {
w = src->width() - src_x;
}
if (src_y+h > src->height()) {
h = src->height() - src_y;
}
// Empty cases
if (w < 1 || h < 1)
return false;
if ((src_x+w <= 0) || (src_x >= src->width()) ||
(src_y+h <= 0) || (src_y >= src->height()))
return false;
if ((dst_x+w <= 0) || (dst_x >= width()) ||
(dst_y+h <= 0) || (dst_y >= height()))
return false;
// Check this function is working correctly
ASSERT(src->bounds().contains(gfx::Rect(src_x, src_y, w, h)));
ASSERT(bounds().contains(gfx::Rect(dst_x, dst_y, w, h)));
return true;
}
};
//////////////////////////////////////////////////////////////////////
// Specializations
template<>
inline void ImageImpl<IndexedTraits>::clear(color_t color) {
memset(m_bits, color, width()*height());
}
template<>
inline void ImageImpl<BitmapTraits>::clear(color_t color) {
memset(m_bits, (color ? 0xff: 0x00),
BitmapTraits::getRowStrideBytes(width()) * height());
}
template<>
inline color_t ImageImpl<BitmapTraits>::getPixel(int x, int y) const {
ASSERT(x >= 0 && x < width());
ASSERT(y >= 0 && y < height());
div_t d = div(x, 8);
return ((*(m_rows[y] + d.quot)) & (1<<d.rem)) ? 1: 0;
}
template<>
inline void ImageImpl<BitmapTraits>::putPixel(int x, int y, color_t color) {
ASSERT(x >= 0 && x < width());
ASSERT(y >= 0 && y < height());
div_t d = div(x, 8);
if (color)
(*(m_rows[y] + d.quot)) |= (1 << d.rem);
else
(*(m_rows[y] + d.quot)) &= ~(1 << d.rem);
}
template<>
inline void ImageImpl<RgbTraits>::blendRect(int x1, int y1, int x2, int y2, color_t color, int opacity) {
address_t addr;
int x, y;
for (y=y1; y<=y2; ++y) {
addr = (address_t)getPixelAddress(x1, y);
for (x=x1; x<=x2; ++x) {
*addr = rgba_blend_normal(*addr, color, opacity);
++addr;
}
}
}
template<>
inline void ImageImpl<BitmapTraits>::copy(const Image* src, gfx::Clip area) {
if (!area.clip(width(), height(), src->width(), src->height()))
return;
// Copy process
ImageConstIterator<BitmapTraits> src_it(src, area.srcBounds(), area.src.x, area.src.y);
ImageIterator<BitmapTraits> dst_it(this, area.dstBounds(), area.dst.x, area.dst.y);
int end_x = area.dst.x+area.size.w;
for (int end_y=area.dst.y+area.size.h;
area.dst.y<end_y;
++area.dst.y, ++area.src.y) {
for (int x=area.dst.x; x<end_x; ++x) {
*dst_it = *src_it;
++src_it;
++dst_it;
}
}
}
} // namespace doc
#endif