#include <boost/bind.hpp> #include <boost/algorithm/string/trim.hpp> #include <SDL_image.h> #include "../lib/CLodHandler.h" #include "CBitmapHandler.h" #include "Graphics.h" #include "CAnimation.h" #include "SDL_Extensions.h" /* * CAnimation.cpp, part of VCMI engine * * Authors: listed in file AUTHORS in main folder * * License: GNU General Public License v2.0 or later * Full text of license available in license.txt file, in main folder * */ extern DLL_EXPORT CLodHandler *spriteh; extern DLL_EXPORT CLodHandler *bitmaph; typedef std::map <size_t, std::vector <std::string> > source_map; typedef std::map<size_t, IImage* > image_map; typedef std::map<size_t, image_map > group_map; class SDLImageLoader { SDLImage * image; ui8 * lineStart; ui8 * position; public: //load size raw pixels from data inline void Load(size_t size, const ui8 * data); //set size pixels to color inline void Load(size_t size, ui8 color=0); inline void EndLine(); //init image with these sizes and palette inline void init(Point SpriteSize, Point Margins, Point FullSize, SDL_Color *pal); SDLImageLoader(SDLImage * Img); ~SDLImageLoader(); }; class CompImageLoader { CompImage * image; ui8 *position; ui8 *entry; unsigned int currentLine; inline ui8 typeOf(ui8 color); inline void NewEntry(ui8 color, size_t size); inline void NewEntry(const ui8 * &data, size_t size); public: //load size raw pixels from data inline void Load(size_t size, const ui8 * data); //set size pixels to color inline void Load(size_t size, ui8 color=0); inline void EndLine(); //init image with these sizes and palette inline void init(Point SpriteSize, Point Margins, Point FullSize, SDL_Color *pal); CompImageLoader(CompImage * Img); ~CompImageLoader(); }; //Small internal class for parsing texts class TextParser { int type; size_t position; std::string text; public: TextParser(const std::string &name); int getType() const; void parseFile(std::map<size_t, std::vector <std::string> > &result); }; /************************************************************************* * DefFile, class used for def loading * *************************************************************************/ size_t getPaletteSize(int type) { switch (type) { case 66: return 32; case 71: return 4; default: return 20; } } CDefFile::CDefFile(std::string Name): data(NULL), palette(NULL) { //First 8 colors in def palette used for transparency static SDL_Color H3Palette[8] = { { 0, 0, 0, 0},// 100% - transparency { 0, 0, 0, 192},// 75% - shadow border, { 0, 0, 0, 128},// TODO: find exact value { 0, 0, 0, 128},// TODO: for transparency { 0, 0, 0, 128},// 50% - shadow body { 0, 0, 0, 0},// 100% - selection highlight { 0, 0, 0, 128},// 50% - shadow body below selection { 0, 0, 0, 192} // 75% - shadow border below selection }; data = spriteh->giveFile(Name, FILE_ANIMATION); palette = new SDL_Color[256]; int it = 0; unsigned int type = readNormalNr(data, it); it+=4; //int width = readNormalNr(data, it); it+=4;//not used //int height = readNormalNr(data, it); it+=4; it+=8; unsigned int totalBlocks = readNormalNr(data, it); it+=4; for (unsigned int i= 0; i<256; i++) { palette[i].r = data[it++]; palette[i].g = data[it++]; palette[i].b = data[it++]; palette[i].unused = 255; } memcpy(palette, H3Palette, getPaletteSize(type));//initialize shadow\selection colors for (unsigned int i=0; i<totalBlocks; i++) { size_t blockID = readNormalNr(data, it); it+=4; size_t totalEntries = readNormalNr(data, it); it+=12; //8 unknown bytes - skipping //13 bytes for name of every frame in this block - not used, skipping it+= 13 * totalEntries; for (unsigned int j=0; j<totalEntries; j++) { size_t currOffset = readNormalNr(data, it); offset[blockID].push_back(currOffset); it += 4; } } } template<class ImageLoader> void CDefFile::loadFrame(size_t frame, size_t group, ImageLoader &loader) const { std::map<size_t, std::vector <size_t> >::const_iterator it; it = offset.find(group); assert (it != offset.end()); const ui8 * FDef = data+it->second[frame]; const SSpriteDef sd = * reinterpret_cast<const SSpriteDef *>(FDef); SSpriteDef sprite; //sprite.size = SDL_SwapLE32(sd.size);//unused sprite.format = SDL_SwapLE32(sd.format); sprite.fullWidth = SDL_SwapLE32(sd.fullWidth); sprite.fullHeight = SDL_SwapLE32(sd.fullHeight); sprite.width = SDL_SwapLE32(sd.width); sprite.height = SDL_SwapLE32(sd.height); sprite.leftMargin = SDL_SwapLE32(sd.leftMargin); sprite.topMargin = SDL_SwapLE32(sd.topMargin); unsigned int currentOffset = sizeof(SSpriteDef); unsigned int BaseOffset = sizeof(SSpriteDef); loader.init(Point(sprite.width, sprite.height), Point(sprite.leftMargin, sprite.topMargin), Point(sprite.fullWidth, sprite.fullHeight), palette); switch (sprite.format) { case 0: { //pixel data is not compressed, copy data to surface for (unsigned int i=0; i<sprite.height; i++) { loader.Load(sprite.width, FDef[currentOffset]); currentOffset += sprite.width; loader.EndLine(); } break; } case 1: { //for each line we have offset of pixel data const ui32 * RWEntriesLoc = reinterpret_cast<const ui32 *>(FDef+currentOffset); currentOffset += sizeof(ui32) * sprite.height; for (unsigned int i=0; i<sprite.height; i++) { //get position of the line currentOffset=BaseOffset + SDL_SwapLE32(read_unaligned_u32(RWEntriesLoc + i)); unsigned int TotalRowLength = 0; while (TotalRowLength<sprite.width) { unsigned char type=FDef[currentOffset++]; unsigned int length=FDef[currentOffset++] + 1; if (type==0xFF)//Raw data { loader.Load(length, FDef + currentOffset); currentOffset+=length; } else// RLE { loader.Load(length, type); } TotalRowLength += length; } loader.EndLine(); } break; } case 2: { currentOffset = BaseOffset + SDL_SwapLE16(read_unaligned_u16(FDef + BaseOffset)); for (unsigned int i=0; i<sprite.height; i++) { unsigned int TotalRowLength=0; while (TotalRowLength<sprite.width) { unsigned char SegmentType=FDef[currentOffset++]; unsigned char code = SegmentType / 32; unsigned char length = (SegmentType & 31) + 1; if (code==7)//Raw data { loader.Load(length, FDef[currentOffset]); currentOffset += length; } else//RLE { loader.Load(length, code); } TotalRowLength+=length; } loader.EndLine(); } break; } case 3: { for (unsigned int i=0; i<sprite.height; i++) { currentOffset = BaseOffset + SDL_SwapLE16(read_unaligned_u16(FDef + BaseOffset+i*2*(sprite.width/32))); unsigned int TotalRowLength=0; while (TotalRowLength<sprite.width) { unsigned char segment = FDef[currentOffset++]; unsigned char code = segment / 32; unsigned char length = (segment & 31) + 1; if (code==7)//Raw data { loader.Load(length, FDef + currentOffset); currentOffset += length; } else//RLE { loader.Load(length, code); } TotalRowLength += length; } loader.EndLine(); } break; } default: tlog0<<"Error: unsupported format of def file:"<<sprite.format<<"\n"; break; } }; CDefFile::~CDefFile() { delete[] data; delete[] palette; } const std::map<size_t, size_t > CDefFile::getEntries() const { std::map<size_t, size_t > ret; for (std::map<size_t, std::vector <size_t> >::const_iterator mapIt = offset.begin(); mapIt!=offset.end(); ++mapIt) ret[mapIt->first] = mapIt->second.size(); return ret; } /************************************************************************* * Classes for image loaders - helpers for loading from def files * *************************************************************************/ SDLImageLoader::SDLImageLoader(SDLImage * Img): image(Img), lineStart(NULL), position(NULL) { } void SDLImageLoader::init(Point SpriteSize, Point Margins, Point FullSize, SDL_Color *pal) { //Init image image->surf = SDL_CreateRGBSurface(SDL_SWSURFACE, SpriteSize.x, SpriteSize.y, 8, 0, 0, 0, 0); image->margins = Margins; image->fullSize = FullSize; //Prepare surface SDL_SetColors(image->surf, pal, 0, 256); SDL_LockSurface(image->surf); lineStart = position = (ui8*)image->surf->pixels; } inline void SDLImageLoader::Load(size_t size, const ui8 * data) { if (size) { memcpy((void *)position, data, size); position += size; } } inline void SDLImageLoader::Load(size_t size, ui8 color) { if (size) { memset((void *)position, color, size); position += size; } } inline void SDLImageLoader::EndLine() { lineStart += image->surf->pitch; position = lineStart; } SDLImageLoader::~SDLImageLoader() { SDL_UnlockSurface(image->surf); SDL_SetColorKey(image->surf, SDL_SRCCOLORKEY, 0); //TODO: RLE if compressed and bpp>1 } //////////////////////////////////////////////////////////////////////////////// CompImageLoader::CompImageLoader(CompImage * Img): image(Img), position(NULL), entry(NULL), currentLine(0) { } void CompImageLoader::init(Point SpriteSize, Point Margins, Point FullSize, SDL_Color *pal) { image->sprite = Rect(Margins, SpriteSize); image->fullSize = FullSize; if (SpriteSize.x && SpriteSize.y) { image->palette = new SDL_Color[256]; memcpy((void*)image->palette, (void*)pal, 256*sizeof(SDL_Color)); //Allocate enought space for worst possible case, c-style malloc used due to resizing after load image->surf = (ui8*)malloc(SpriteSize.x*SpriteSize.y*3); image->line = new unsigned int[SpriteSize.y+1]; image->line[0] = 0; position = image->surf; } } inline void CompImageLoader::NewEntry(ui8 color, size_t size) { assert(color != 0xff); assert(size && size<256); entry = position; entry[0] = color; entry[1] = size; position +=2; } inline void CompImageLoader::NewEntry(const ui8 * &data, size_t size) { assert(size && size<256); entry = position; entry[0] = 0xff; entry[1] = size; position +=2; memcpy(position, data, size); position+=size; data+=size; } inline ui8 CompImageLoader::typeOf(ui8 color) { if (color == 0) return 0; if (image->palette[color].unused != 255) return 1; return 2; } inline void CompImageLoader::Load(size_t size, const ui8 * data) { while (size) { //Try to compress data while(true) { ui8 color = data[0]; if (color != 0xff) { size_t runLength = 1; while (runLength < size && color == data[runLength]) runLength++; if (runLength > 1)//Row of one color found - use RLE { Load(runLength, color); data += runLength; size -= runLength; if (!size) return; } else break; } else break; } //Select length for new raw entry size_t runLength = 1; ui8 color = data[0]; ui8 type = typeOf(color); ui8 color2; ui8 type2; if (size > 1) { do { color2 = data[runLength]; type2 = typeOf(color2); runLength++; } //While we have data of this type and different colors while ((runLength < size) && (type == type2) && ( (color2 != 0xff) || (color2 != color))); } size -= runLength; //add data to last entry if (entry && entry[0] == 0xff && type == typeOf(entry[2])) { size_t toCopy = std::min<size_t>(runLength, 255 - entry[1]); runLength -= toCopy; entry[1] += toCopy; memcpy(position, data, toCopy); data+=toCopy; position+=toCopy; } //Create new entries while (runLength > 255) { NewEntry(data, 255); runLength -= 255; } if (runLength) NewEntry(data, runLength); } } inline void CompImageLoader::Load(size_t size, ui8 color) { if (!size) return; if (color==0xff) { ui8* tmpbuf = new ui8[size]; memset((void*)tmpbuf, color, size); Load(size, tmpbuf); delete [] tmpbuf; return; } //Current entry is RLE with same color as new block if (entry && entry[0] == color) { size_t toCopy = std::min<size_t>(size, 255 - entry[1]); entry[1] = 255; size -= toCopy; entry[1] += toCopy; } //Create new entries while (size > 255) { NewEntry(color, 255); size -= 255; } if (size) NewEntry(color, size); } void CompImageLoader::EndLine() { currentLine++; image->line[currentLine] = position - image->surf; entry = NULL; } CompImageLoader::~CompImageLoader() { if (!image->surf) return; ui8* newPtr = (ui8*)realloc((void*)image->surf, position - image->surf); if (newPtr) image->surf = newPtr; } /************************************************************************* * Classes for images, support loading from file and drawing on surface * *************************************************************************/ IImage::IImage(): refCount(1) { } bool IImage::decreaseRef() { refCount--; return refCount <= 0; } void IImage::increaseRef() { refCount++; } SDLImage::SDLImage(CDefFile *data, size_t frame, size_t group, bool compressed): surf(NULL) { SDLImageLoader loader(this); data->loadFrame(frame, group, loader); } SDLImage::SDLImage(SDL_Surface * from, bool extraRef): margins(0,0) { surf = from; if (extraRef) surf->refcount++; fullSize.x = surf->w; fullSize.y = surf->h; } SDLImage::SDLImage(std::string filename, bool compressed): margins(0,0) { if (spriteh->haveFile(filename, FILE_GRAPHICS)) { int size; unsigned char * pic=NULL; pic = spriteh->giveFile(filename, FILE_GRAPHICS, &size); surf = IMG_Load_RW( SDL_RWFromMem((void*)pic, size), 1); delete [] pic; } else if(bitmaph->haveFile(filename, FILE_GRAPHICS)) { surf = BitmapHandler::loadBitmap(filename); } else { tlog0<<"Error: file not found: "<<filename<<"\n"; surf = NULL; return; } fullSize.x = surf->w; fullSize.y = surf->h; } void SDLImage::draw(SDL_Surface *where, int posX, int posY, Rect *src, unsigned char rotation) const { if (!surf) return; Rect sourceRect(margins.x, margins.y, surf->w, surf->h); //TODO: rotation and scaling if (src) { sourceRect = sourceRect & *src; } Rect destRect(posX, posY, surf->w, surf->h); destRect += sourceRect.topLeft(); sourceRect -= margins; CSDL_Ext::blitSurface(surf, &sourceRect, where, &destRect); } void SDLImage::playerColored(int player) { graphics->blueToPlayersAdv(surf, player); } int SDLImage::width() const { return fullSize.x; } int SDLImage::height() const { return fullSize.y; } SDLImage::~SDLImage() { SDL_FreeSurface(surf); } CompImage::CompImage(const CDefFile *data, size_t frame, size_t group): surf(NULL), line(NULL), palette(NULL) { CompImageLoader loader(this); data->loadFrame(frame, group, loader); } CompImage::CompImage(SDL_Surface * surf) { //TODO assert(0); } void CompImage::draw(SDL_Surface *where, int posX, int posY, Rect *src, ui8 alpha) const { int rotation = 0; //TODO //rotation & 2 = horizontal rotation //rotation & 4 = vertical rotation if (!surf) return; Rect sourceRect(sprite); //TODO: rotation and scaling if (src) sourceRect = sourceRect & *src; //Limit source rect to sizes of surface sourceRect = sourceRect & Rect(0, 0, where->w, where->h); //Starting point on SDL surface Point dest(posX+sourceRect.x, posY+sourceRect.y); if (rotation & 2) dest.y += sourceRect.h; if (rotation & 4) dest.x += sourceRect.w; sourceRect -= sprite.topLeft(); for (int currY = 0; currY <sourceRect.h; currY++) { ui8* data = surf + line[currY+sourceRect.y]; ui8 type = *(data++); ui8 size = *(data++); int currX = sourceRect.x; //Skip blocks until starting position reached while ( currX > size ) { currX -= size; if (type == 0xff) data += size; type = *(data++); size = *(data++); } //This block will be shown partially - calculate size\position size -= currX; if (type == 0xff) data += currX; currX = 0; ui8 bpp = where->format->BytesPerPixel; //Calculate position for blitting: pixels + Y + X ui8* blitPos = (ui8*) where->pixels; if (rotation & 4) blitPos += (dest.y - currY) * where->pitch; else blitPos += (dest.y + currY) * where->pitch; blitPos += dest.x * bpp; //Blit blocks that must be fully visible while (currX + size < sourceRect.w) { //blit block, pointers will be modified if needed BlitBlockWithBpp(bpp, type, size, data, blitPos, alpha, rotation & 2); currX += size; type = *(data++); size = *(data++); } //Blit last, semi-visible block size = sourceRect.w - currX; BlitBlockWithBpp(bpp, type, size, data, blitPos, alpha, rotation & 2); } } #define CASEBPP(x,y) case x: BlitBlock<x,y>(type, size, data, dest, alpha); break //FIXME: better way to get blitter void CompImage::BlitBlockWithBpp(ui8 bpp, ui8 type, ui8 size, ui8 *&data, ui8 *&dest, ui8 alpha, bool rotated) const { assert(bpp>1 && bpp<5); if (rotated) switch (bpp) { CASEBPP(2,1); CASEBPP(3,1); CASEBPP(4,1); } else switch (bpp) { CASEBPP(2,1); CASEBPP(3,1); CASEBPP(4,1); } } #undef CASEBPP //Blit one block from RLE-d surface template<int bpp, int dir> void CompImage::BlitBlock(ui8 type, ui8 size, ui8 *&data, ui8 *&dest, ui8 alpha) const { //Raw data if (type == 0xff) { ui8 color = *data; if (alpha != 255)//Per-surface alpha is set { for (size_t i=0; i<size; i++) { SDL_Color col = palette[*(data++)]; col.unused = (unsigned int)col.unused*(255-alpha)/255; ColorPutter<bpp, 1>::PutColorAlpha(dest, col); } return; } if (palette[color].unused == 255) { //Put row of RGB data for (size_t i=0; i<size; i++) ColorPutter<bpp, 1>::PutColor(dest, palette[*(data++)]); } else { //Put row of RGBA data for (size_t i=0; i<size; i++) ColorPutter<bpp, 1>::PutColorAlpha(dest, palette[*(data++)]); } } //RLE-d sequence else { if (alpha != 255 && palette[type].unused !=0)//Per-surface alpha is set { SDL_Color col = palette[type]; col.unused = (int)col.unused*(255-alpha)/255; for (size_t i=0; i<size; i++) ColorPutter<bpp, 1>::PutColorAlpha(dest, col); return; } switch (palette[type].unused) { case 0: { //Skip row dest += size*bpp; break; } case 255: { //Put RGB row ColorPutter<bpp, 1>::PutColorRow(dest, palette[type], size); break; } default: { //Put RGBA row for (size_t i=0; i<size; i++) ColorPutter<bpp, 1>::PutColorAlpha(dest, palette[type]); break; } } } } void CompImage::playerColored(int player) { SDL_Color *pal = NULL; if(player < PLAYER_LIMIT && player >= 0) { pal = graphics->playerColorPalette + 32*player; } else if(player == 255 || player == -1) { pal = graphics->neutralColorPalette; } else assert(0); for(int i=0; i<32; ++i) { palette[224+i].r = pal[i].r; palette[224+i].g = pal[i].g; palette[224+i].b = pal[i].b; palette[224+i].unused = pal[i].unused; } } int CompImage::width() const { return fullSize.x; } int CompImage::height() const { return fullSize.y; } CompImage::~CompImage() { free(surf); delete [] line; delete [] palette; } /************************************************************************* * CAnimation for animations handling, can load part of file if needed * *************************************************************************/ TextParser::TextParser(const std::string &name): type(0), position(std::string::npos) { if (!spriteh->haveFile(name, FILE_TEXT)) return; text = spriteh->getTextFile(name); position = text.find('\n'); std::string typeStr = text.substr(0, position); boost::algorithm::trim(typeStr); if (typeStr == "Replace") type = 1; else if (typeStr == "Append") type = 2; } int TextParser::getType() const { return type; } void TextParser::parseFile(std::map<size_t, std::vector <std::string> > &result) { while (position != std::string::npos) { size_t lineStart = position+1; position = text.find('\n', lineStart); std::string buf = text.substr(lineStart, position-lineStart); size_t currentBlock = atoi(buf.c_str()); while (position != std::string::npos) { lineStart = position+1; position = text.find('\n', lineStart); std::string res = text.substr(lineStart, position-lineStart); boost::algorithm::trim(res); if (res.empty()) break; result[currentBlock].push_back(res); } } } IImage * CAnimation::getFromExtraDef(std::string filename) { size_t pos = filename.find(':'); if (pos == -1) return NULL; CAnimation anim(filename.substr(0, pos)); pos++; size_t frame = atoi(filename.c_str()+pos); size_t group = 0; pos = filename.find(':', pos); if (pos != -1) { group = frame; frame = atoi(filename.c_str()+pos); } anim.load(frame ,group); IImage * ret = anim.images[group][frame]; anim.images.clear(); return ret; } bool CAnimation::loadFrame(CDefFile * file, size_t frame, size_t group) { if (size(group) <= frame) { printError(frame, group, "LoadFrame"); return false; } IImage *image = getImage(frame, group, false); if (image) { image->increaseRef(); return true; } //try to get image from def if (source[group][frame].empty()) { if (compressed) images[group][frame] = new CompImage(file, frame, group); else images[group][frame] = new SDLImage(file, frame, group); } else //load from separate file { IImage * img = getFromExtraDef(source[group][frame].c_str()); if (!img)//TODO: load compressed image img = new SDLImage(source[group][frame].c_str()); images[group][frame] = img; return true; } return false; } bool CAnimation::unloadFrame(size_t frame, size_t group) { IImage *image = getImage(frame, group, false); if (image) { //decrease ref count for image and delete if needed if (image->decreaseRef()) { delete image; images[group].erase(frame); } if (images[group].empty()) images.erase(group); return true; } return false; } void CAnimation::init(CDefFile * file) { TextParser txtFile(name); if ( file && txtFile.getType() != 1 ) { const std::map<size_t, size_t> defEntries = file->getEntries(); for (std::map<size_t, size_t>::const_iterator mapIt = defEntries.begin(); mapIt!=defEntries.end(); ++mapIt) source[mapIt->first].resize(mapIt->second); } txtFile.parseFile(source); } CDefFile * CAnimation::getFile() const { if (spriteh->haveFile(name, FILE_ANIMATION)) return new CDefFile(name); return NULL; } void CAnimation::printError(size_t frame, size_t group, std::string type) const { tlog0 << type <<" error: Request for frame not present in CAnimation!\n" <<"\tFile name: "<<name<<" Group: "<<group<<" Frame: "<<frame<<"\n"; } CAnimation::CAnimation(std::string Name, bool Compressed): name(Name), compressed(Compressed) { size_t dotPos = name.find_last_of('.'); if ( dotPos!=-1 ) name.erase(dotPos); std::transform(name.begin(), name.end(), name.begin(), toupper); CDefFile * file = getFile(); init(file); delete file; loadedAnims.insert(this); } CAnimation::CAnimation(): name(""), compressed(false) { init(NULL); loadedAnims.insert(this); } CAnimation::~CAnimation() { if (!images.empty()) { tlog2<<"Warning: not all frames were unloaded from "<<name<<"\n"; for (group_map::iterator group = images.begin(); group != images.end(); ++group ) for (image_map::iterator image = group->second.begin(); image != group->second.end(); ++image ) delete image->second; } loadedAnims.erase(this); } void CAnimation::setCustom(std::string filename, size_t frame, size_t group) { if (source[group].size() <= frame) source[group].resize(frame+1); source[group][frame] = filename; //FIXME: update image if already loaded } IImage * CAnimation::getImage(size_t frame, size_t group, bool verbose) const { group_map::const_iterator groupIter = images.find(group); if (groupIter != images.end()) { image_map::const_iterator imageIter = groupIter->second.find(frame); if (imageIter != groupIter->second.end()) return imageIter->second; } if (verbose) printError(frame, group, "GetImage"); return NULL; } void CAnimation::load() { CDefFile * file = getFile(); for (source_map::iterator group = source.begin(); group != source.end(); ++group ) for (size_t image=0; image < group->second.size(); image++) loadFrame(file, image, group->first); delete file; } void CAnimation::unload() { for (source_map::iterator group = source.begin(); group != source.end(); ++group ) for (size_t image=0; image < group->second.size(); image++) unloadFrame(image, group->first); } void CAnimation::loadGroup(size_t group) { CDefFile * file = getFile(); if (vstd::contains(source, group)) for (size_t image=0; image < source[group].size(); image++) loadFrame(file, image, group); delete file; } void CAnimation::unloadGroup(size_t group) { if (vstd::contains(source, group)) for (size_t image=0; image < source[group].size(); image++) unloadFrame(image, group); } void CAnimation::load(size_t frame, size_t group) { CDefFile * file = getFile(); loadFrame(file, frame, group); delete file; } void CAnimation::unload(size_t frame, size_t group) { unloadFrame(frame, group); } size_t CAnimation::size(size_t group) const { source_map::const_iterator iter = source.find(group); if (iter != source.end()) return iter->second.size(); return 0; } std::set<CAnimation*> CAnimation::loadedAnims; void CAnimation::getAnimInfo() { tlog1<<"Animation stats: Loaded "<<loadedAnims.size()<<" total\n"; for (std::set<CAnimation*>::iterator it = loadedAnims.begin(); it != loadedAnims.end(); it++) { CAnimation * anim = *it; tlog1<<"Name: "<<anim->name<<" Groups: "<<anim->images.size(); if (!anim->images.empty()) tlog1<<", "<<anim->images.begin()->second.size()<<" image loaded in group "<< anim->images.begin()->first; tlog1<<"\n"; } } CAnimImage::CAnimImage(std::string name, size_t Frame, size_t Group, int x, int y, unsigned char Flags): frame(Frame), group(Group), player(-1), flags(Flags) { pos.x += x; pos.y += y; anim = new CAnimation(name); init(); } CAnimImage::CAnimImage(CAnimation *Anim, size_t Frame, size_t Group, int x, int y, unsigned char Flags): anim(Anim), frame(Frame), group(Group), player(-1), flags(Flags) { pos.x += x; pos.y += y; init(); } void CAnimImage::init() { anim->load(frame, group); if (flags & CShowableAnim::BASE) anim->load(0,group); IImage *img = anim->getImage(frame, group); pos.w = img->width(); pos.h = img->height(); } CAnimImage::~CAnimImage() { anim->unload(frame, group); if (flags & CShowableAnim::BASE) anim->unload(0,group); delete anim; } void CAnimImage::showAll(SDL_Surface *to) { anim->getImage(frame, group)->draw(to, pos.x, pos.y); } void CAnimImage::setFrame(size_t Frame, size_t Group) { if (frame == Frame && group==Group) return; if (anim->size(Group) > Frame) { anim->load(Frame, Group); anim->unload(frame, group); frame = Frame; group = Group; if (flags & CShowableAnim::PLAYER_COLORED) anim->getImage(frame, group)->playerColored(player); } } void CAnimImage::playerColored(int currPlayer) { player = currPlayer; flags |= CShowableAnim::PLAYER_COLORED; anim->getImage(frame, group)->playerColored(player); if (flags & CShowableAnim::BASE) anim->getImage(0, group)->playerColored(player); } CShowableAnim::CShowableAnim(int x, int y, std::string name, unsigned char Flags, unsigned int Delay, size_t Group): anim(name, Flags & USE_RLE), group(Group), frame(0), first(0), frameDelay(Delay), value(0), flags(Flags), xOffset(0), yOffset(0), alpha(255) { anim.loadGroup(group); last = anim.size(group); pos.w = anim.getImage(0, group)->width(); pos.h = anim.getImage(0, group)->height(); pos.x+= x; pos.y+= y; } CShowableAnim::~CShowableAnim() { anim.unloadGroup(group); } void CShowableAnim::setAlpha(unsigned int alphaValue) { alpha = std::min<unsigned int>(alphaValue, 255); } bool CShowableAnim::set(size_t Group, size_t from, size_t to) { size_t max = anim.size(Group); if (max>to) max = to; if (max < from || max == 0) return false; anim.load(Group); anim.unload(group); group = Group; frame = first = from; last = max; value = 0; return true; } bool CShowableAnim::set(size_t Group) { if (anim.size(Group)== 0) return false; if (group != Group) { anim.loadGroup(Group); anim.unloadGroup(group); first = 0; group = Group; last = anim.size(Group); } frame = value = 0; return true; } void CShowableAnim::reset() { value = 0; frame = first; if (callback) callback(); } void CShowableAnim::clipRect(int posX, int posY, int width, int height) { xOffset = posX; yOffset = posY; pos.w = width; pos.h = height; } void CShowableAnim::show(SDL_Surface *to) { if ( flags & BASE && frame != first) blitImage(first, group, to); blitImage(frame, group, to); if ( ++value == frameDelay ) { value = 0; if ( ++frame == last) reset(); } } void CShowableAnim::showAll(SDL_Surface *to) { if ( flags & BASE && frame != first) blitImage(first, group, to); blitImage(frame, group, to); } void CShowableAnim::blitImage(size_t frame, size_t group, SDL_Surface *to) { assert(to); Rect src( xOffset, yOffset, pos.w, pos.h); IImage * img = anim.getImage(frame, group); img->draw(to, pos.x-xOffset, pos.y-yOffset, &src, alpha); } void CShowableAnim::rotate(bool on, bool vertical) { unsigned char flag = vertical? VERTICAL_FLIP:HORIZONTAL_FLIP; if (on) flags |= flag; else flags &= ~flag; } CCreatureAnim::CCreatureAnim(int x, int y, std::string name, Rect picPos, unsigned char flags, EAnimType type): CShowableAnim(x,y,name,flags,3,type) { xOffset = picPos.x; yOffset = picPos.y; if (picPos.w) pos.w = picPos.w; if (picPos.h) pos.h = picPos.h; }; void CCreatureAnim::loopPreview() { std::vector<EAnimType> available; static const EAnimType previewList[] = {HOLDING, HITTED, DEFENCE, ATTACK_FRONT, CAST_FRONT}; for (size_t i=0; i<5; i++) if (anim.size(previewList[i])) available.push_back(previewList[i]); size_t rnd = rand()%(available.size()*2); if (rnd >= available.size()) { if ( anim.size(MOVING) == 0 )//no moving animation present addLast( HOLDING ); else addLast( MOVING ) ; } else addLast(available[rnd]); } void CCreatureAnim::addLast(EAnimType newType) { if (type != MOVING && newType == MOVING)//starting moving - play init sequence { queue.push( MOVE_START ); } else if (type == MOVING && newType != MOVING )//previous anim was moving - finish it { queue.push( MOVE_END ); } if (newType == TURN_L || newType == TURN_R) queue.push(newType); queue.push(newType); } void CCreatureAnim::reset() { //if we are in the middle of rotation - set flag if (type == TURN_L && !queue.empty() && queue.front() == TURN_L) rotate(true); if (type == TURN_R && !queue.empty() && queue.front() == TURN_R) rotate(false); while (!queue.empty()) { EAnimType at = queue.front(); queue.pop(); if (set(at)) return; } if (callback) callback(); while (!queue.empty()) { EAnimType at = queue.front(); queue.pop(); if (set(at)) return; } set(HOLDING); } void CCreatureAnim::startPreview() { callback = boost::bind(&CCreatureAnim::loopPreview,this); } void CCreatureAnim::clearAndSet(EAnimType type) { while (!queue.empty()) queue.pop(); set(type); }