Mercurial > fife-parpg
view engine/core/video/image.cpp @ 378:64738befdf3b
bringing in the changes from the build_system_rework branch in preparation for the 0.3.0 release. This commit will require the Jan2010 devkit. Clients will also need to be modified to the new way to import fife.
author | vtchill@33b003aa-7bff-0310-803a-e67f0ece8222 |
---|---|
date | Mon, 11 Jan 2010 23:34:52 +0000 |
parents | 3804348fe3fb |
children | d242e6ce6f9f |
line wrap: on
line source
/*************************************************************************** * Copyright (C) 2005-2008 by the FIFE team * * http://www.fifengine.de * * This file is part of FIFE. * * * * FIFE is free software; you can redistribute it and/or * * modify it under the terms of the GNU Lesser General Public * * License as published by the Free Software Foundation; either * * version 2.1 of the License, or (at your option) any later version. * * * * This library is distributed in the hope that it will be useful, * * but WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * * Lesser General Public License for more details. * * * * You should have received a copy of the GNU Lesser General Public * * License along with this library; if not, write to the * * Free Software Foundation, Inc., * * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * ***************************************************************************/ // Standard C++ library includes #include <cassert> #include <iostream> // 3rd party library includes #include <SDL.h> // FIFE includes // These includes are split up in two parts, separated by one empty line // First block: files included from the FIFE root src directory // Second block: files included from the same folder #include "image.h" namespace FIFE { Image::Image(SDL_Surface* surface): m_surface(NULL) { reset(surface); } Image::Image(const uint8_t* data, unsigned int width, unsigned int height): m_surface(NULL) { SDL_Surface* surface = SDL_CreateRGBSurface(SDL_SWSURFACE | SDL_SRCALPHA, width,height, 32, RMASK, GMASK, BMASK ,AMASK); SDL_LockSurface(surface); unsigned int size = width * height * 4; uint8_t* pixeldata = static_cast<uint8_t*>(surface->pixels); std::copy(data, data + size, pixeldata); SDL_UnlockSurface(surface); reset(surface); } void Image::reset(SDL_Surface* surface) { if( m_surface ) { SDL_FreeSurface(m_surface); } m_surface = surface; m_xshift = 0; m_yshift = 0; while (!m_clipstack.empty()) { m_clipstack.pop(); } m_area.x = m_area.y = m_area.w = m_area.h = 0; m_surface = surface; } Image::~Image() { //assert(m_refcount == 0); reset(NULL); } SDL_Surface* Image::detachSurface() { SDL_Surface* srf = m_surface; m_surface = NULL; return srf; } unsigned int Image::getWidth() const { if (!m_surface) { return 0; } return m_surface->w; } unsigned int Image::getHeight() const { if (!m_surface) { return 0; } return m_surface->h; } const Rect& Image::getArea() { m_area.w = getWidth(); m_area.h = getHeight(); return m_area; } void Image::setXShift(int xshift) { m_xshift = xshift; } void Image::setYShift(int yshift) { m_yshift = yshift; } void Image::getPixelRGBA(int x, int y, uint8_t* r, uint8_t* g, uint8_t* b, uint8_t* a) { if ((x < 0) || (x >= m_surface->w) || (y < 0) || (y >= m_surface->h)) { r = 0; g = 0; b = 0; a = 0; return; } int bpp = m_surface->format->BytesPerPixel; Uint8 *p = (Uint8*)m_surface->pixels + y * m_surface->pitch + x * bpp; uint32_t pixel; switch(bpp) { case 1: pixel = *p; case 2: pixel = *(Uint16 *)p; case 3: if (SDL_BYTEORDER == SDL_BIG_ENDIAN) { pixel = p[0] << 16 | p[1] << 8 | p[2]; } else { pixel = p[0] | p[1] << 8 | p[2] << 16; } case 4: pixel = *(Uint32 *)p; } SDL_GetRGBA(pixel, m_surface->format, r, g, b, a); } void Image::render(const Rect& rect, unsigned char alpha) { render(rect, SDL_GetVideoSurface(), alpha); } void Image::pushClipArea(const Rect& cliparea, bool clear) { ClipInfo ci; ci.r = cliparea; ci.clearing = clear; m_clipstack.push(ci); setClipArea(cliparea, clear); } void Image::popClipArea() { assert(!m_clipstack.empty()); m_clipstack.pop(); if (m_clipstack.empty()) { clearClipArea(); } else { ClipInfo ci = m_clipstack.top(); setClipArea(ci.r, ci.clearing); } } const Rect& Image::getClipArea() const { if (m_clipstack.empty()) { return m_clipstack.top().r; } else { return m_area; } } void Image::clearClipArea() { setClipArea(m_area, true); } void Image::saveAsPng(const std::string& filename, SDL_Surface& surface) { FILE *fp; png_structp pngptr; png_infop infoptr; int colortype; png_bytep *rowpointers = NULL; Uint32 rmask, gmask, bmask, amask; //get endian #if SDL_BYTEORDER == SDL_BIG_ENDIAN rmask = 0xff000000; gmask = 0x00ff0000; bmask = 0x0000ff00; amask = 0x000000ff; #else rmask = 0x000000ff; gmask = 0x0000ff00; bmask = 0x00ff0000; amask = 0xff000000; #endif fp = fopen(filename.c_str(), "wb"); if (fp == NULL) { return; } //create the png file pngptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL); if (pngptr == NULL) { fclose(fp); return; } //create information struct infoptr = png_create_info_struct(pngptr); if (infoptr == NULL) { fclose(fp); png_destroy_write_struct(&pngptr, (png_infopp)NULL); return; } if (setjmp(png_jmpbuf(pngptr))) { png_destroy_write_struct(&pngptr, &infoptr); fclose(fp); return; } //initialize io png_init_io(pngptr, fp); //lock the surface for access SDL_LockSurface(&surface); colortype = PNG_COLOR_TYPE_RGB; if(m_surface->format->palette){ colortype |= PNG_COLOR_TYPE_PALETTE; } else if (m_surface->format->Amask){ colortype |= PNG_COLOR_TYPE_RGB_ALPHA; } else{} png_set_IHDR(pngptr, infoptr, surface.w, surface.h, 8, colortype, PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_DEFAULT, PNG_FILTER_TYPE_DEFAULT); png_write_info(pngptr, infoptr); png_set_packing(pngptr); rowpointers = new png_bytep[surface.h]; for (int i = 0; i < surface.h; i++) { rowpointers[i] = (png_bytep)(Uint8 *)surface.pixels + i*surface.pitch; } //write the image png_write_image(pngptr, rowpointers); png_write_end(pngptr, infoptr); SDL_UnlockSurface(&surface); delete [] rowpointers; png_destroy_write_struct(&pngptr, &infoptr); fclose(fp); } }