背景和头像缺失
This commit is contained in:
338
cocos2d/cocos/base/TGAlib.cpp
Normal file
338
cocos2d/cocos/base/TGAlib.cpp
Normal file
@@ -0,0 +1,338 @@
|
||||
/****************************************************************************
|
||||
Copyright (c) 2010-2012 cocos2d-x.org
|
||||
Copyright (c) 2013-2014 Chukong Technologies Inc.
|
||||
|
||||
http://www.cocos2d-x.org
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in
|
||||
all copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
THE SOFTWARE.
|
||||
****************************************************************************/
|
||||
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "TGAlib.h"
|
||||
#include "base/CCData.h"
|
||||
#include "platform/CCFileUtils.h"
|
||||
|
||||
NS_CC_BEGIN
|
||||
|
||||
static bool tgaLoadRLEImageData(unsigned char* Buffer, unsigned long bufSize, tImageTGA *info);
|
||||
void tgaFlipImage( tImageTGA *info );
|
||||
|
||||
// load the image header field from stream
|
||||
bool tgaLoadHeader(unsigned char* buffer, unsigned long bufSize, tImageTGA *info)
|
||||
{
|
||||
bool ret = false;
|
||||
|
||||
do
|
||||
{
|
||||
size_t step = sizeof(unsigned char) * 2;
|
||||
CC_BREAK_IF((step + sizeof(unsigned char)) > bufSize);
|
||||
memcpy(&info->type, buffer + step, sizeof(unsigned char));
|
||||
|
||||
step += sizeof(unsigned char) * 2;
|
||||
step += sizeof(signed short) * 4;
|
||||
CC_BREAK_IF((step + sizeof(signed short) * 2 + sizeof(unsigned char)) > bufSize);
|
||||
memcpy(&info->width, buffer + step, sizeof(signed short));
|
||||
memcpy(&info->height, buffer + step + sizeof(signed short), sizeof(signed short));
|
||||
memcpy(&info->pixelDepth, buffer + step + sizeof(signed short) * 2, sizeof(unsigned char));
|
||||
|
||||
step += sizeof(unsigned char);
|
||||
step += sizeof(signed short) * 2;
|
||||
CC_BREAK_IF((step + sizeof(unsigned char)) > bufSize);
|
||||
unsigned char cGarbage;
|
||||
memcpy(&cGarbage, buffer + step, sizeof(unsigned char));
|
||||
|
||||
info->flipped = 0;
|
||||
if ( cGarbage & 0x20 )
|
||||
{
|
||||
info->flipped = 1;
|
||||
}
|
||||
ret = true;
|
||||
} while (0);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
bool tgaLoadImageData(unsigned char *Buffer, unsigned long bufSize, tImageTGA *info)
|
||||
{
|
||||
bool ret = false;
|
||||
|
||||
do
|
||||
{
|
||||
int mode,total,i;
|
||||
unsigned char aux;
|
||||
size_t step = (sizeof(unsigned char) + sizeof(signed short)) * 6;
|
||||
|
||||
// mode equal the number of components for each pixel
|
||||
mode = info->pixelDepth / 8;
|
||||
// total is the number of unsigned chars we'll have to read
|
||||
total = info->height * info->width * mode;
|
||||
|
||||
size_t dataSize = sizeof(unsigned char) * total;
|
||||
CC_BREAK_IF((step + dataSize) > bufSize);
|
||||
memcpy(info->imageData, Buffer + step, dataSize);
|
||||
|
||||
// mode=3 or 4 implies that the image is RGB(A). However TGA
|
||||
// stores it as BGR(A) so we'll have to swap R and B.
|
||||
if (mode >= 3)
|
||||
{
|
||||
for (i=0; i < total; i+= mode)
|
||||
{
|
||||
aux = info->imageData[i];
|
||||
info->imageData[i] = info->imageData[i+2];
|
||||
info->imageData[i+2] = aux;
|
||||
}
|
||||
}
|
||||
|
||||
ret = true;
|
||||
} while (0);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static bool tgaLoadRLEImageData(unsigned char* buffer, unsigned long bufSize, tImageTGA *info)
|
||||
{
|
||||
unsigned int mode,total,i, index = 0;
|
||||
unsigned char aux[4], runlength = 0;
|
||||
unsigned int skip = 0, flag = 0;
|
||||
size_t step = (sizeof(unsigned char) + sizeof(signed short)) * 6;
|
||||
|
||||
// mode equal the number of components for each pixel
|
||||
mode = info->pixelDepth / 8;
|
||||
// total is the number of unsigned chars we'll have to read
|
||||
total = info->height * info->width;
|
||||
|
||||
for( i = 0; i < total; i++ )
|
||||
{
|
||||
// if we have a run length pending, run it
|
||||
if ( runlength != 0 )
|
||||
{
|
||||
// we do, update the run length count
|
||||
runlength--;
|
||||
skip = (flag != 0);
|
||||
}
|
||||
else
|
||||
{
|
||||
// otherwise, read in the run length token
|
||||
CC_BREAK_IF((step + sizeof(unsigned char)) > bufSize);
|
||||
memcpy(&runlength, buffer + step, sizeof(unsigned char));
|
||||
step += sizeof(unsigned char);
|
||||
|
||||
// see if it's a RLE encoded sequence
|
||||
flag = runlength & 0x80;
|
||||
if ( flag )
|
||||
{
|
||||
runlength -= 128;
|
||||
}
|
||||
skip = 0;
|
||||
}
|
||||
|
||||
// do we need to skip reading this pixel?
|
||||
if ( !skip )
|
||||
{
|
||||
// no, read in the pixel data
|
||||
CC_BREAK_IF((step + sizeof(unsigned char) * mode) > bufSize);
|
||||
|
||||
memcpy(aux, buffer + step, sizeof(unsigned char) * mode);
|
||||
step += sizeof(unsigned char) * mode;
|
||||
|
||||
// mode=3 or 4 implies that the image is RGB(A). However TGA
|
||||
// stores it as BGR(A) so we'll have to swap R and B.
|
||||
if ( mode >= 3 )
|
||||
{
|
||||
unsigned char tmp;
|
||||
|
||||
tmp = aux[0];
|
||||
aux[0] = aux[2];
|
||||
aux[2] = tmp;
|
||||
}
|
||||
}
|
||||
|
||||
// add the pixel to our image
|
||||
memcpy(&info->imageData[index], aux, mode);
|
||||
index += mode;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
void tgaFlipImage( tImageTGA *info )
|
||||
{
|
||||
// mode equal the number of components for each pixel
|
||||
int mode = info->pixelDepth / 8;
|
||||
int rowbytes = info->width*mode;
|
||||
unsigned char *row = (unsigned char *)malloc(rowbytes);
|
||||
int y;
|
||||
|
||||
if (row == nullptr) return;
|
||||
|
||||
for( y = 0; y < (info->height/2); y++ )
|
||||
{
|
||||
memcpy(row, &info->imageData[y*rowbytes],rowbytes);
|
||||
memcpy(&info->imageData[y*rowbytes], &info->imageData[(info->height-(y+1))*rowbytes], rowbytes);
|
||||
memcpy(&info->imageData[(info->height-(y+1))*rowbytes], row, rowbytes);
|
||||
}
|
||||
|
||||
free(row);
|
||||
info->flipped = 0;
|
||||
}
|
||||
|
||||
tImageTGA* tgaLoadBuffer(unsigned char* buffer, long size)
|
||||
{
|
||||
int mode,total;
|
||||
tImageTGA *info = nullptr;
|
||||
|
||||
do
|
||||
{
|
||||
CC_BREAK_IF(! buffer);
|
||||
info = (tImageTGA *)malloc(sizeof(tImageTGA));
|
||||
|
||||
// get the file header info
|
||||
if (! tgaLoadHeader(buffer, size, info))
|
||||
{
|
||||
info->status = TGA_ERROR_MEMORY;
|
||||
break;
|
||||
}
|
||||
|
||||
// check if the image is color indexed
|
||||
if (info->type == 1)
|
||||
{
|
||||
info->status = TGA_ERROR_INDEXED_COLOR;
|
||||
break;
|
||||
}
|
||||
|
||||
// check for other types (compressed images)
|
||||
if ((info->type != 2) && (info->type !=3) && (info->type !=10) )
|
||||
{
|
||||
info->status = TGA_ERROR_COMPRESSED_FILE;
|
||||
break;
|
||||
}
|
||||
|
||||
// mode equals the number of image components
|
||||
mode = info->pixelDepth / 8;
|
||||
// total is the number of unsigned chars to read
|
||||
total = info->height * info->width * mode;
|
||||
// allocate memory for image pixels
|
||||
info->imageData = (unsigned char *)malloc(sizeof(unsigned char) * total);
|
||||
|
||||
// check to make sure we have the memory required
|
||||
if (info->imageData == nullptr)
|
||||
{
|
||||
info->status = TGA_ERROR_MEMORY;
|
||||
break;
|
||||
}
|
||||
|
||||
bool bLoadImage = false;
|
||||
// finally load the image pixels
|
||||
if ( info->type == 10 )
|
||||
{
|
||||
bLoadImage = tgaLoadRLEImageData(buffer, size, info);
|
||||
}
|
||||
else
|
||||
{
|
||||
bLoadImage = tgaLoadImageData(buffer, size, info);
|
||||
}
|
||||
|
||||
// check for errors when reading the pixels
|
||||
if (! bLoadImage)
|
||||
{
|
||||
info->status = TGA_ERROR_READING_FILE;
|
||||
break;
|
||||
}
|
||||
info->status = TGA_OK;
|
||||
|
||||
if ( info->flipped )
|
||||
{
|
||||
tgaFlipImage( info );
|
||||
if ( info->flipped )
|
||||
{
|
||||
info->status = TGA_ERROR_MEMORY;
|
||||
}
|
||||
}
|
||||
} while(0);
|
||||
|
||||
return info;
|
||||
}
|
||||
|
||||
// this is the function to call when we want to load an image
|
||||
tImageTGA * tgaLoad(const char *filename)
|
||||
{
|
||||
Data data = FileUtils::getInstance()->getDataFromFile(filename);
|
||||
|
||||
if (!data.isNull())
|
||||
{
|
||||
return tgaLoadBuffer(data.getBytes(), data.getSize());
|
||||
}
|
||||
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
// converts RGB to grayscale
|
||||
void tgaRGBtogreyscale(tImageTGA *info) {
|
||||
|
||||
int mode,i,j;
|
||||
|
||||
unsigned char *newImageData;
|
||||
|
||||
// if the image is already grayscale do nothing
|
||||
if (info->pixelDepth == 8)
|
||||
return;
|
||||
|
||||
// compute the number of actual components
|
||||
mode = info->pixelDepth / 8;
|
||||
|
||||
// allocate an array for the new image data
|
||||
newImageData = (unsigned char *)malloc(sizeof(unsigned char) *
|
||||
info->height * info->width);
|
||||
if (newImageData == nullptr) {
|
||||
return;
|
||||
}
|
||||
|
||||
// convert pixels: grayscale = o.30 * R + 0.59 * G + 0.11 * B
|
||||
for (i = 0,j = 0; j < info->width * info->height; i +=mode, j++)
|
||||
newImageData[j] =
|
||||
(unsigned char)(0.30 * info->imageData[i] +
|
||||
0.59 * info->imageData[i+1] +
|
||||
0.11 * info->imageData[i+2]);
|
||||
|
||||
|
||||
//free old image data
|
||||
free(info->imageData);
|
||||
|
||||
// reassign pixelDepth and type according to the new image type
|
||||
info->pixelDepth = 8;
|
||||
info->type = 3;
|
||||
// reassigning imageData to the new array.
|
||||
info->imageData = newImageData;
|
||||
}
|
||||
|
||||
// releases the memory used for the image
|
||||
void tgaDestroy(tImageTGA *info) {
|
||||
|
||||
if (info != nullptr) {
|
||||
if (info->imageData != nullptr)
|
||||
{
|
||||
free(info->imageData);
|
||||
}
|
||||
|
||||
free(info);
|
||||
}
|
||||
}
|
||||
NS_CC_END
|
||||
Reference in New Issue
Block a user