/* ScummVM - Graphic Adventure Engine * * ScummVM is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * This program 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 General Public License for more details. * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * $URL$ * $Id$ * */ /* * This code is based on Broken Sword 2.5 engine * * Copyright (c) Malte Thiesen, Daniel Queteschiner and Michael Elsdoerfer * * Licensed under GNU GPL v2 * */ // ----------------------------------------------------------------------------- // INCLUDES // ----------------------------------------------------------------------------- #include "sword25/package/packagemanager.h" #include "sword25/gfx/image/imageloader.h" #include "sword25/gfx/opengl/openglgfx.h" #include "sword25/gfx/opengl/glimage.h" #include "common/system.h" namespace Sword25 { #define BS_LOG_PREFIX "GLIMAGE" // ----------------------------------------------------------------------------- // CONSTRUCTION / DESTRUCTION // ----------------------------------------------------------------------------- GLImage::GLImage(const Common::String &Filename, bool &Result) : _data(0), m_Width(0), m_Height(0) { Result = false; PackageManager *pPackage = static_cast(Kernel::GetInstance()->GetService("package")); BS_ASSERT(pPackage); _backSurface = (static_cast(Kernel::GetInstance()->GetService("gfx")))->getSurface(); // Datei laden char *pFileData; unsigned int FileSize; if (!(pFileData = (char *) pPackage->GetFile(Filename, &FileSize))) { BS_LOG_ERRORLN("File \"%s\" could not be loaded.", Filename.c_str()); return; } // Bildeigenschaften bestimmen GraphicEngine::COLOR_FORMATS ColorFormat; int Pitch; if (!ImageLoader::ExtractImageProperties(pFileData, FileSize, ColorFormat, m_Width, m_Height)) { BS_LOG_ERRORLN("Could not read image properties."); return; } // Das Bild dekomprimieren if (!ImageLoader::LoadImage(pFileData, FileSize, GraphicEngine::CF_ABGR32, _data, m_Width, m_Height, Pitch)) { BS_LOG_ERRORLN("Could not decode image."); return; } // Dateidaten freigeben delete[] pFileData; Result = true; return; } // ----------------------------------------------------------------------------- GLImage::GLImage(unsigned int Width, unsigned int Height, bool &Result) : m_Width(Width), m_Height(Height) { Result = false; _data = new byte[Width * Height * 4]; Result = true; return; } // ----------------------------------------------------------------------------- GLImage::~GLImage() { delete[] _data; } // ----------------------------------------------------------------------------- bool GLImage::Fill(const Common::Rect *pFillRect, unsigned int Color) { BS_LOG_ERRORLN("Fill() is not supported."); return false; } // ----------------------------------------------------------------------------- bool GLImage::SetContent(const byte *Pixeldata, uint size, unsigned int Offset, unsigned int Stride) { // Überprüfen, ob PixelData ausreichend viele Pixel enthält um ein Bild der Größe Width * Height zu erzeugen if (size < static_cast(m_Width * m_Height * 4)) { BS_LOG_ERRORLN("PixelData vector is too small to define a 32 bit %dx%d image.", m_Width, m_Height); return false; } const byte *in = &Pixeldata[Offset]; byte *out = _data; for (int i = 0; i < m_Height; i++) { memcpy(out, in, m_Width * 4); out += m_Width * 4; in += Stride; } return true; } // ----------------------------------------------------------------------------- unsigned int GLImage::GetPixel(int X, int Y) { BS_LOG_ERRORLN("GetPixel() is not supported. Returning black."); return 0; } // ----------------------------------------------------------------------------- bool GLImage::Blit(int PosX, int PosY, int Flipping, Common::Rect *pPartRect, unsigned int Color, int Width, int Height) { int x1 = 0, y1 = 0; int w = m_Width, h = m_Height; if (pPartRect) { x1 = pPartRect->left; y1 = pPartRect->top; w = pPartRect->right - pPartRect->left; h = pPartRect->bottom - pPartRect->top; } debug(0, "Blit(%d, %d, %d, [%d, %d, %d, %d], %d, %d, %d)", PosX, PosY, Flipping, x1, y1, w, h, Color, Width, Height); // Skalierungen berechnen float ScaleX, ScaleY; if (Width == -1) Width = m_Width; ScaleX = (float) Width / (float) m_Width; if (Height == -1) Height = m_Height; ScaleY = (float) Height / (float) m_Height; if (Color != 0xffffffff) { warning("STUB: Image bg color: %x", Color); } if (ScaleX != 1.0 || ScaleY != 1.0) { warning("STUB: Sprite scaling (%f x %f)", ScaleX, ScaleY); } if (Flipping & (Image::FLIP_V | Image::FLIP_H)) { warning("STUB: Sprite flipping"); } if (PosX < 0) { w -= PosX; x1 = -PosX; PosX = 0; } if (PosY < 0) { h -= PosY; y1 = -PosY; PosY = 0; } w = CLIP(w, 0, (int)_backSurface->w - PosX - 1); h = CLIP(h, 0, (int)_backSurface->h - PosY - 1); if (w == 0 || h == 0) return true; // Rendern // TODO: // Die Bedeutung von FLIP_V und FLIP_H ist vertauscht. Allerdings glaubt der Rest der Engine auch daran, daher war es einfacher diesen Fehler // weiterzuführen. Bei Gelegenheit ist dieses aber zu ändern. // TODO: scaling // TODO: Flipping byte *ino = &_data[y1 * m_Width * 4 + x1 * 4]; byte *outo = (byte *)_backSurface->getBasePtr(PosX, PosY); byte *in, *out; for (int i = 0; i < h; i++) { out = outo; in = ino; for (int j = 0; j < w; j++) { int r = *in++; int g = *in++; int b = *in++; int a = *in++; switch (a) { case 0: // Full transparency out += 4; break; case 255: // Full opacity *out++ = r; *out++ = g; *out++ = b; *out++ = a; break; default: // alpha blending *out = (byte)((((int)(*out - r) * a + r) >> 8) & 0xff); out++; *out = (byte)((((int)(*out - g) * a + g) >> 8) & 0xff); out++; *out = (byte)((((int)(*out - b) * a + b) >> 8) & 0xff); out++; *out = 255; out++; } } outo += _backSurface->pitch; ino += m_Width * 4; } g_system->copyRectToScreen((byte *)_backSurface->getBasePtr(PosX, PosY), _backSurface->pitch, PosX, PosY, w, h); return true; } } // End of namespace Sword25