dxx-rebirth/common/2d/bitmap.cpp

263 lines
6.9 KiB
C++
Raw Normal View History

2006-03-20 17:12:09 +00:00
/*
2014-06-01 17:55:23 +00:00
* Portions of this file are copyright Rebirth contributors and licensed as
* described in COPYING.txt.
* Portions of this file are copyright Parallax Software and licensed
* according to the Parallax license below.
* See COPYING.txt for license details.
2006-03-20 17:12:09 +00:00
THE COMPUTER CODE CONTAINED HEREIN IS THE SOLE PROPERTY OF PARALLAX
SOFTWARE CORPORATION ("PARALLAX"). PARALLAX, IN DISTRIBUTING THE CODE TO
END-USERS, AND SUBJECT TO ALL OF THE TERMS AND CONDITIONS HEREIN, GRANTS A
ROYALTY-FREE, PERPETUAL LICENSE TO SUCH END-USERS FOR USE BY SUCH END-USERS
IN USING, DISPLAYING, AND CREATING DERIVATIVE WORKS THEREOF, SO LONG AS
SUCH USE, DISPLAY OR CREATION IS FOR NON-COMMERCIAL, ROYALTY OR REVENUE
FREE PURPOSES. IN NO EVENT SHALL THE END-USER USE THE COMPUTER CODE
CONTAINED HEREIN FOR REVENUE-BEARING PURPOSES. THE END-USER UNDERSTANDS
AND AGREES TO THE TERMS HEREIN AND ACCEPTS THE SAME BY USE OF THIS FILE.
COPYRIGHT 1993-1998 PARALLAX SOFTWARE CORPORATION. ALL RIGHTS RESERVED.
*/
/*
*
* Graphical routines for manipulating grs_bitmaps.
*
*/
2014-11-12 03:43:44 +00:00
#include <stdexcept>
2006-03-20 17:12:09 +00:00
#include <stdlib.h>
#include <stdio.h>
#include "u_mem.h"
#include "gr.h"
#include "grdef.h"
#include "dxxerror.h"
2006-03-20 17:12:09 +00:00
#ifdef OGL
#include "ogl_init.h"
#endif
2012-11-02 17:37:54 +00:00
#include "bitmap.h"
2006-03-20 17:12:09 +00:00
2014-11-15 03:31:41 +00:00
// Allocated a bitmap and makes its data be raw_data that is already somewhere.
static grs_bitmap_ptr gr_create_bitmap_raw(uint16_t w, uint16_t h, unsigned char * raw_data);
void gr_set_bitmap_data (grs_bitmap &bm, unsigned char *data)
2006-03-20 17:12:09 +00:00
{
#ifdef OGL
ogl_freebmtexture(bm);
2006-03-20 17:12:09 +00:00
#endif
bm.bm_data = data;
2006-03-20 17:12:09 +00:00
}
2014-11-15 03:31:41 +00:00
grs_bitmap_ptr gr_create_bitmap(uint16_t w, uint16_t h )
2006-03-20 17:12:09 +00:00
{
2012-11-24 16:11:04 +00:00
unsigned char *d;
MALLOC(d, unsigned char, MAX_BMP_SIZE(w, h));
return gr_create_bitmap_raw (w, h, d);
2006-03-20 17:12:09 +00:00
}
2014-11-15 03:31:41 +00:00
grs_bitmap_ptr gr_create_bitmap_raw(uint16_t w, uint16_t h, unsigned char * raw_data )
2006-03-20 17:12:09 +00:00
{
2014-07-22 02:14:56 +00:00
grs_bitmap_ptr n(new grs_bitmap);
2014-11-30 22:09:20 +00:00
gr_init_bitmap(*n.get(), 0, 0, 0, w, h, w, raw_data);
return n;
2006-03-20 17:12:09 +00:00
}
2014-11-30 22:09:20 +00:00
void gr_init_bitmap(grs_bitmap &bm, uint8_t mode, uint16_t x, uint16_t y, uint16_t w, uint16_t h, uint16_t bytesperline, unsigned char * data ) // TODO: virtualize
2006-03-20 17:12:09 +00:00
{
2014-11-30 22:09:20 +00:00
bm.bm_x = x;
bm.bm_y = y;
bm.bm_w = w;
bm.bm_h = h;
bm.bm_flags = 0;
bm.bm_type = mode;
bm.bm_rowsize = bytesperline;
2006-03-20 17:12:09 +00:00
2014-11-30 22:09:20 +00:00
bm.bm_data = nullptr;
2006-03-20 17:12:09 +00:00
#ifdef OGL
2014-11-30 22:09:20 +00:00
bm.bm_parent = nullptr;
bm.gltexture = nullptr;
2006-03-20 17:12:09 +00:00
#endif
2014-11-30 22:09:20 +00:00
gr_set_bitmap_data(bm, data);
2006-03-20 17:12:09 +00:00
}
2014-11-30 22:09:21 +00:00
void gr_init_bitmap_alloc(grs_bitmap &bm, uint8_t mode, uint16_t x, uint16_t y, uint16_t w, uint16_t h, uint16_t bytesperline)
2006-03-20 17:12:09 +00:00
{
2012-11-24 16:11:04 +00:00
unsigned char *d;
MALLOC(d, unsigned char, MAX_BMP_SIZE(w, h));
2014-11-30 22:09:21 +00:00
gr_init_bitmap(bm, mode, x, y, w, h, bytesperline, d);
2006-03-20 17:12:09 +00:00
}
2014-11-30 22:09:20 +00:00
void gr_init_bitmap_data (grs_bitmap &bm) // TODO: virtulize
2006-03-20 17:12:09 +00:00
{
2014-11-30 22:09:20 +00:00
bm.bm_data = nullptr;
bm.bm_parent = nullptr;
2006-03-20 17:12:09 +00:00
#ifdef OGL
2014-11-30 22:09:20 +00:00
bm.gltexture = nullptr;
2006-03-20 17:12:09 +00:00
#endif
}
2014-11-30 22:09:21 +00:00
grs_subbitmap_ptr gr_create_sub_bitmap(grs_bitmap &bm, uint16_t x, uint16_t y, uint16_t w, uint16_t h)
2006-03-20 17:12:09 +00:00
{
2014-07-22 02:50:01 +00:00
grs_subbitmap_ptr n(new grs_bitmap);
2014-11-30 22:09:21 +00:00
gr_init_sub_bitmap(*n.get(), bm, x, y, w, h);
return n;
2006-03-20 17:12:09 +00:00
}
2014-07-22 02:14:56 +00:00
void gr_free_bitmap(std::unique_ptr<grs_bitmap> bm)
2006-03-20 17:12:09 +00:00
{
2014-11-30 22:09:20 +00:00
gr_free_bitmap_data(*bm.get());
2006-03-20 17:12:09 +00:00
}
2014-11-30 22:09:20 +00:00
void gr_free_bitmap_data (grs_bitmap &bm) // TODO: virtulize
2006-03-20 17:12:09 +00:00
{
#ifdef OGL
2014-11-30 22:09:20 +00:00
ogl_freebmtexture(bm);
2006-03-20 17:12:09 +00:00
#endif
2014-11-30 22:09:20 +00:00
if (bm.bm_data != NULL)
d_free (bm.bm_data);
2006-03-20 17:12:09 +00:00
}
2014-11-30 22:09:21 +00:00
void gr_init_sub_bitmap (grs_bitmap &bm, grs_bitmap &bmParent, uint16_t x, uint16_t y, uint16_t w, uint16_t h ) // TODO: virtualize
2006-03-20 17:12:09 +00:00
{
2014-11-30 22:09:21 +00:00
uint32_t subx = x + bmParent.bm_x;
uint32_t suby = y + bmParent.bm_y;
if (subx != (bm.bm_x = static_cast<uint16_t>(subx)) ||
suby != (bm.bm_y = static_cast<uint16_t>(suby)))
2014-11-12 03:43:44 +00:00
throw std::overflow_error("offset overflow");
2014-11-30 22:09:21 +00:00
bm.bm_w = w;
bm.bm_h = h;
bm.bm_flags = bmParent.bm_flags;
bm.bm_type = bmParent.bm_type;
bm.bm_rowsize = bmParent.bm_rowsize;
2006-03-20 17:12:09 +00:00
#ifdef OGL
2014-11-30 22:09:21 +00:00
bm.gltexture = bmParent.gltexture;
2006-03-20 17:12:09 +00:00
#endif
2014-11-30 22:09:21 +00:00
bm.bm_parent = &bmParent;
bm.bm_data = &bmParent.bm_data[(unsigned int)((y*bmParent.bm_rowsize)+x)];
2006-03-20 17:12:09 +00:00
}
void decode_data(ubyte *data, int num_pixels, ubyte *colormap, int *count)
2006-03-20 17:12:09 +00:00
{
ubyte mapped;
for (int i = 0; i < num_pixels; i++) {
2006-03-20 17:12:09 +00:00
count[*data]++;
mapped = *data;
*data = colormap[mapped];
data++;
}
}
2014-11-15 03:31:41 +00:00
void gr_set_bitmap_flags (grs_bitmap *pbm, uint8_t flags)
2006-03-20 17:12:09 +00:00
{
pbm->bm_flags = flags;
}
void gr_set_transparent (grs_bitmap *pbm, int bTransparent)
{
if (bTransparent)
{
gr_set_bitmap_flags (pbm, pbm->bm_flags | BM_FLAG_TRANSPARENT);
}
else
{
gr_set_bitmap_flags (pbm, pbm->bm_flags & ~BM_FLAG_TRANSPARENT);
}
}
void gr_set_super_transparent (grs_bitmap *pbm, int bTransparent)
{
if (bTransparent)
{
gr_set_bitmap_flags (pbm, pbm->bm_flags & ~BM_FLAG_SUPER_TRANSPARENT);
}
else
{
gr_set_bitmap_flags (pbm, pbm->bm_flags | BM_FLAG_SUPER_TRANSPARENT);
}
}
2014-08-08 02:07:47 +00:00
void build_colormap_good( palette_array_t &palette, color_t * colormap, int * freq )
2006-03-20 17:12:09 +00:00
{
int r, g, b;
2006-03-20 17:12:09 +00:00
for (int i=0; i < 256; i++ ) {
2013-01-06 21:11:53 +00:00
r = palette[i].r;
g = palette[i].g;
b = palette[i].b;
2006-03-20 17:12:09 +00:00
*colormap++ = gr_find_closest_color( r, g, b );
*freq++ = 0;
}
}
void gr_remap_bitmap( grs_bitmap * bmp, palette_array_t &palette, int transparent_color, int super_transparent_color )
2006-03-20 17:12:09 +00:00
{
ubyte colormap[256];
int freq[256];
if (bmp->bm_type != BM_LINEAR)
return; //can't do it
// This should be build_colormap_asm, but we're not using invert table, so...
build_colormap_good( palette, colormap, freq );
if ( (super_transparent_color>=0) && (super_transparent_color<=255))
colormap[super_transparent_color] = 254;
if ( (transparent_color>=0) && (transparent_color<=255))
colormap[transparent_color] = TRANSPARENCY_COLOR;
decode_data(bmp->bm_data, bmp->bm_w * bmp->bm_h, colormap, freq );
2006-03-20 17:12:09 +00:00
if ( (transparent_color>=0) && (transparent_color<=255) && (freq[transparent_color]>0) )
gr_set_transparent (bmp, 1);
if ( (super_transparent_color>=0) && (super_transparent_color<=255) && (freq[super_transparent_color]>0) )
gr_set_super_transparent (bmp, 0);
}
void gr_remap_bitmap_good( grs_bitmap * bmp, palette_array_t &palette, int transparent_color, int super_transparent_color )
2006-03-20 17:12:09 +00:00
{
ubyte colormap[256];
int freq[256];
build_colormap_good( palette, colormap, freq );
if ( (super_transparent_color>=0) && (super_transparent_color<=255))
colormap[super_transparent_color] = 254;
if ( (transparent_color>=0) && (transparent_color<=255))
colormap[transparent_color] = TRANSPARENCY_COLOR;
if (bmp->bm_w == bmp->bm_rowsize)
decode_data(bmp->bm_data, bmp->bm_w * bmp->bm_h, colormap, freq );
2006-03-20 17:12:09 +00:00
else {
ubyte *p = bmp->bm_data;
for (int y=0;y<bmp->bm_h;y++,p+=bmp->bm_rowsize)
decode_data(p, bmp->bm_w, colormap, freq );
2006-03-20 17:12:09 +00:00
}
if ( (transparent_color>=0) && (transparent_color<=255) && (freq[transparent_color]>0) )
gr_set_transparent (bmp, 1);
if ( (super_transparent_color>=0) && (super_transparent_color<=255) && (freq[super_transparent_color]>0) )
gr_set_super_transparent (bmp, 1);
}
void gr_bitmap_check_transparency( grs_bitmap * bmp )
{
ubyte * data;
data = bmp->bm_data;
for (int y=0; y<bmp->bm_h; y++ ) {
for (int x=0; x<bmp->bm_w; x++ ) {
2006-03-20 17:12:09 +00:00
if (*data++ == TRANSPARENCY_COLOR ) {
gr_set_transparent (bmp, 1);
return;
}
}
data += bmp->bm_rowsize - bmp->bm_w;
}
bmp->bm_flags = 0;
}