/* 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 3 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, see . * */ #include "common/stream.h" #include "image/cicn.h" #include "image/pict.h" #include "graphics/pixelformat.h" #include "graphics/surface.h" namespace Image { CicnDecoder::CicnDecoder(): _surface(nullptr), _palette(0), _mask(nullptr) { } CicnDecoder::~CicnDecoder() { destroy(); } void CicnDecoder::destroy() { if (_surface) { _surface->free(); delete _surface; _surface = nullptr; } _palette.clear(); if (_mask) { _mask->free(); delete _mask; _mask = nullptr; } } bool CicnDecoder::loadStream(Common::SeekableReadStream &stream) { destroy(); Image::PICTDecoder::PixMap pixMap = Image::PICTDecoder::readPixMap(stream); // Mask header stream.skip(4); uint16 maskRowBytes = stream.readUint16BE(); stream.readUint16BE(); // top stream.readUint16BE(); // left uint16 maskHeight = stream.readUint16BE(); // bottom uint16 maskWidth = stream.readUint16BE(); // right // Bitmap header stream.skip(4); uint16 bitmapRowBytes = stream.readUint16BE(); stream.readUint16BE(); // top stream.readUint16BE(); // left uint16 bitmapHeight = stream.readUint16BE(); // bottom uint16 bitmapWidth = stream.readUint16BE(); // right // Mask and bitmap data stream.skip(4); if (maskRowBytes && maskHeight) { _mask = new Graphics::Surface(); _mask->create(maskWidth, maskHeight, Graphics::PixelFormat::createFormatCLUT8()); byte *mask = new byte[maskRowBytes * maskHeight]; stream.read(mask, maskRowBytes * maskHeight); for (uint y = 0; y < maskHeight; y++) { for (uint x = 0; x < maskWidth; x++) { if ((mask[y * maskRowBytes + x / 8] & (0x80 >> (x % 8))) == 0) _mask->setPixel(x, y, 0); else _mask->setPixel(x, y, 255); } } delete[] mask; } stream.skip(bitmapRowBytes * bitmapHeight); // Palette stream.skip(6); uint16 paletteColorCount = stream.readUint16BE() + 1; _palette.resize(paletteColorCount, false); for (uint i = 0; i < paletteColorCount; i++) { stream.skip(2); byte r = stream.readUint16BE() >> 8; byte g = stream.readUint16BE() >> 8; byte b = stream.readUint16BE() >> 8; _palette.set(i, r, g, b); } _surface = new Graphics::Surface(); _surface->create(pixMap.bounds.width(), pixMap.bounds.height(), Graphics::PixelFormat::createFormatCLUT8()); if (pixMap.pixelSize == 1) { byte *buf = new byte[pixMap.rowBytes]; for (int y = 0; y < pixMap.bounds.height(); y++) { stream.read(buf, pixMap.rowBytes); for (int x = 0; x < bitmapWidth; x += 8) { for (int i = 0; i < 8 && x + i < bitmapWidth; i++) { _surface->setPixel(x + i, y, (buf[x / 8] >> (7 - i)) & 0x01); } } } delete[] buf; } else if (pixMap.pixelSize == 2) { byte *buf = new byte[pixMap.rowBytes]; for (int y = 0; y < bitmapHeight; y++) { stream.read(buf, pixMap.rowBytes); for (int x = 0; x < bitmapWidth; x += 4) { for (int i = 0; i < 4 && x + i < bitmapWidth; i++) { _surface->setPixel(x + i, y, (buf[x / 4] >> (6 - 2 * i)) & 0x03); } } } delete[] buf; } else if (pixMap.pixelSize == 4) { byte *buf = new byte[pixMap.rowBytes]; for (int y = 0; y < bitmapHeight; y++) { stream.read(buf, pixMap.rowBytes); for (int x = 0; x < bitmapWidth; x += 2) { for (int i = 0; i < 2 && x + i < bitmapWidth; i++) { _surface->setPixel(x + i, y, (buf[x / 2] >> (4 - 4 * i)) & 0x0F); } } } delete[] buf; } else if (pixMap.pixelSize == 8) { byte *buf = new byte[pixMap.rowBytes]; for (int y = 0; y < bitmapHeight; y++) { stream.read(buf, pixMap.rowBytes); memcpy(_surface->getBasePtr(0, y), buf, bitmapWidth); } delete[] buf; } else { error("CicnDecoder::loadStream(): Invalid pixel size %d", pixMap.pixelSize); } return true; } } // End of namespace Image