2019-10-05 21:44:28 +00:00
|
|
|
/*
|
|
|
|
** arttexture.cpp
|
|
|
|
** Texture class for ART-based hightiles
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright 2019 Christoph Oelckers
|
|
|
|
** All rights reserved.
|
|
|
|
**
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
** are met:
|
|
|
|
**
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "files.h"
|
|
|
|
#include "templates.h"
|
|
|
|
#include "bitmap.h"
|
|
|
|
#include "image.h"
|
|
|
|
#include "cache1d.h"
|
|
|
|
#include "build.h"
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// an AET texture
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
class FArtTexture : public FImageSource
|
|
|
|
{
|
|
|
|
|
|
|
|
public:
|
|
|
|
FArtTexture (int width, int height, int p);
|
2019-10-11 17:21:36 +00:00
|
|
|
void CreatePalettedPixels(uint8_t* buffer);
|
2019-10-05 21:44:28 +00:00
|
|
|
int CopyPixels(FBitmap *bmp, int conversion) override;
|
|
|
|
int32_t picanmdisk; // Todo: Get this out again on the other side.
|
|
|
|
};
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FImageSource *ArtImage_TryCreate(FileReader & file)
|
|
|
|
{
|
|
|
|
auto buffer = file.Read();
|
|
|
|
|
2019-10-14 21:11:01 +00:00
|
|
|
// Cannot load if smaller than the header.
|
|
|
|
if (buffer.Size() < ARTv1_UNITOFFSET) return nullptr;
|
|
|
|
uint32_t* header = (uint32_t *) buffer.Data();
|
|
|
|
int ver = LittleLong(header[0]);
|
|
|
|
if (ver != 1) return nullptr;
|
2019-10-05 21:44:28 +00:00
|
|
|
|
2019-10-14 21:11:01 +00:00
|
|
|
// Only allow files with one tile.
|
|
|
|
int firsttile = LittleLong(header[1]);
|
|
|
|
int lasttile = LittleLong(header[2]);
|
|
|
|
if (firsttile != lasttile) return nullptr;
|
|
|
|
|
|
|
|
int32_t picanmdisk = LittleLong(header[5]);
|
|
|
|
int Width = LittleShort(B_UNBUF16(&buffer[16]));
|
|
|
|
int Height = LittleShort(B_UNBUF16(&buffer[18]));
|
2019-10-05 21:44:28 +00:00
|
|
|
|
|
|
|
if (Width <= 0 || Height <= 0)
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t NumPixels = Width * Height;
|
|
|
|
|
|
|
|
if (ARTv1_UNITOFFSET + NumPixels > (int)buffer.Size())
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return new FArtTexture(Width, Height, picanmdisk);
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FArtTexture::FArtTexture(int width, int height, int p)
|
|
|
|
{
|
|
|
|
Width = width;
|
|
|
|
Height = height;
|
|
|
|
picanmdisk = p;
|
|
|
|
}
|
|
|
|
|
2019-10-11 17:21:36 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// This will never be called by the software renderer but let's be safe.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void FArtTexture::CreatePalettedPixels(uint8_t* buffer)
|
|
|
|
{
|
2019-12-07 09:14:51 +00:00
|
|
|
FileReader fr = fileSystem.OpenFileReader(Name, 0);
|
2019-10-11 17:21:36 +00:00
|
|
|
if (!fr.isOpen()) return;
|
|
|
|
int numpixels = Width * Height;
|
|
|
|
fr.Read(buffer, numpixels);
|
|
|
|
}
|
|
|
|
|
2019-10-05 21:44:28 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// FArtTexture::CopyPixels
|
|
|
|
//
|
|
|
|
// This format is special because it ignores the game palettes and
|
|
|
|
// only outputs a true color image with the primary palette.
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
int FArtTexture::CopyPixels(FBitmap *bmp, int conversion)
|
|
|
|
{
|
|
|
|
// Treat both buffer as linear contiguous blocks.
|
|
|
|
// Both Src and Dst are ordered the same with no padding.
|
|
|
|
int numpixels = Width * Height;
|
|
|
|
bool hasalpha = false;
|
2019-12-07 09:14:51 +00:00
|
|
|
FileReader fr = fileSystem.OpenFileReader(Name, 0);
|
2019-10-05 21:44:28 +00:00
|
|
|
if (!fr.isOpen()) return 0;
|
|
|
|
TArray<uint8_t> source(numpixels, true);
|
|
|
|
fr.Read(source.Data(), numpixels);
|
|
|
|
auto dest = bmp->GetPixels();
|
|
|
|
|
|
|
|
for (int y = 0; y < numpixels; ++y)
|
|
|
|
{
|
|
|
|
int index = source[y];
|
|
|
|
if (index == 255)
|
|
|
|
{
|
|
|
|
hasalpha = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
dest[0] = palette[index];
|
|
|
|
dest[1] = palette[index+1];
|
|
|
|
dest[2] = palette[index+2];
|
|
|
|
dest[3] = 255;
|
|
|
|
dest += 4;
|
|
|
|
}
|
|
|
|
bMasked = hasalpha;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|