2020-06-25 06:32:42 +01:00
|
|
|
#include "config.h"
|
2018-09-26 00:13:33 +01:00
|
|
|
#include "imageproviders.h"
|
2018-12-20 23:30:35 +00:00
|
|
|
#include "log.h"
|
2018-09-26 00:13:33 +01:00
|
|
|
#include <QPainter>
|
|
|
|
|
|
|
|
QImage getCollisionMetatileImage(Block block) {
|
|
|
|
return getCollisionMetatileImage(block.collision, block.elevation);
|
|
|
|
}
|
|
|
|
|
|
|
|
QImage getCollisionMetatileImage(int collision, int elevation) {
|
|
|
|
int x = collision * 16;
|
|
|
|
int y = elevation * 16;
|
|
|
|
QPixmap collisionImage = QPixmap(":/images/collisions.png").copy(x, y, 16, 16);
|
|
|
|
return collisionImage.toImage();
|
|
|
|
}
|
|
|
|
|
2020-07-02 02:43:19 +01:00
|
|
|
QImage getMetatileImage(
|
2021-12-01 21:06:14 +00:00
|
|
|
uint16_t metatileId,
|
2020-07-02 02:43:19 +01:00
|
|
|
Tileset *primaryTileset,
|
|
|
|
Tileset *secondaryTileset,
|
|
|
|
QList<int> layerOrder,
|
|
|
|
QList<float> layerOpacity,
|
|
|
|
bool useTruePalettes)
|
|
|
|
{
|
2021-12-01 21:06:14 +00:00
|
|
|
Metatile* metatile = Tileset::getMetatile(metatileId, primaryTileset, secondaryTileset);
|
|
|
|
Tileset* blockTileset = Tileset::getBlockTileset(metatileId, primaryTileset, secondaryTileset);
|
|
|
|
if (!metatile || !blockTileset) {
|
|
|
|
QImage metatile_image(16, 16, QImage::Format_RGBA8888);
|
2020-04-09 20:12:52 +01:00
|
|
|
metatile_image.fill(Qt::magenta);
|
2018-09-26 00:13:33 +01:00
|
|
|
return metatile_image;
|
|
|
|
}
|
2021-12-01 21:06:14 +00:00
|
|
|
return getMetatileImage(metatile, primaryTileset, secondaryTileset, layerOrder, layerOpacity, useTruePalettes);
|
|
|
|
}
|
2018-09-26 00:13:33 +01:00
|
|
|
|
2021-12-01 21:06:14 +00:00
|
|
|
QImage getMetatileImage(
|
|
|
|
Metatile *metatile,
|
|
|
|
Tileset *primaryTileset,
|
|
|
|
Tileset *secondaryTileset,
|
|
|
|
QList<int> layerOrder,
|
|
|
|
QList<float> layerOpacity,
|
|
|
|
bool useTruePalettes)
|
|
|
|
{
|
|
|
|
QImage metatile_image(16, 16, QImage::Format_RGBA8888);
|
|
|
|
if (!metatile) {
|
2020-04-09 20:12:52 +01:00
|
|
|
metatile_image.fill(Qt::magenta);
|
2018-09-26 00:13:33 +01:00
|
|
|
return metatile_image;
|
|
|
|
}
|
2021-12-01 21:06:14 +00:00
|
|
|
metatile_image.fill(Qt::black);
|
|
|
|
|
2020-05-03 16:31:44 +01:00
|
|
|
QList<QList<QRgb>> palettes = Tileset::getBlockPalettes(primaryTileset, secondaryTileset, useTruePalettes);
|
2018-09-26 00:13:33 +01:00
|
|
|
|
|
|
|
QPainter metatile_painter(&metatile_image);
|
2020-06-25 06:32:42 +01:00
|
|
|
bool isTripleLayerMetatile = projectConfig.getTripleLayerMetatilesEnabled();
|
|
|
|
int numLayers = isTripleLayerMetatile ? 3: 2;
|
|
|
|
for (int layer = 0; layer < numLayers; layer++)
|
2018-09-26 00:13:33 +01:00
|
|
|
for (int y = 0; y < 2; y++)
|
|
|
|
for (int x = 0; x < 2; x++) {
|
2020-07-02 02:19:08 +01:00
|
|
|
int l = layerOrder.size() >= numLayers ? layerOrder[layer] : layer;
|
2020-07-02 16:31:42 +01:00
|
|
|
int bottomLayer = layerOrder.size() >= numLayers ? layerOrder[0] : 0;
|
2021-12-01 21:06:14 +00:00
|
|
|
Tile tile = metatile->tiles.value((y * 2) + x + (l * 4));
|
|
|
|
QImage tile_image = getTileImage(tile.tileId, primaryTileset, secondaryTileset);
|
2018-09-26 00:13:33 +01:00
|
|
|
if (tile_image.isNull()) {
|
|
|
|
// Some metatiles specify tiles that are outside the valid range.
|
|
|
|
// These are treated as completely transparent, so they can be skipped without
|
2018-09-27 19:27:57 +01:00
|
|
|
// being drawn unless they're on the bottom layer, in which case we need
|
|
|
|
// a placeholder because garbage will be drawn otherwise.
|
2020-07-02 16:31:42 +01:00
|
|
|
if (l == bottomLayer) {
|
2018-09-27 19:27:57 +01:00
|
|
|
metatile_painter.fillRect(x * 8, y * 8, 8, 8, palettes.value(0).value(0));
|
|
|
|
}
|
2018-09-26 00:13:33 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Colorize the metatile tiles with its palette.
|
2021-12-01 21:06:14 +00:00
|
|
|
if (tile.palette < palettes.length()) {
|
|
|
|
QList<QRgb> palette = palettes.value(tile.palette);
|
2018-09-26 00:13:33 +01:00
|
|
|
for (int j = 0; j < palette.length(); j++) {
|
|
|
|
tile_image.setColor(j, palette.value(j));
|
|
|
|
}
|
|
|
|
} else {
|
2021-12-01 21:06:14 +00:00
|
|
|
logWarn(QString("Tile '%1' is referring to invalid palette number: '%2'").arg(tile.tileId).arg(tile.palette));
|
2018-09-26 00:13:33 +01:00
|
|
|
}
|
|
|
|
|
2020-07-02 02:59:01 +01:00
|
|
|
QPoint origin = QPoint(x*8, y*8);
|
|
|
|
float opacity = layerOpacity.size() >= numLayers ? layerOpacity[l] : 1.0;
|
|
|
|
if (opacity < 1.0) {
|
|
|
|
int alpha = 255 * opacity;
|
|
|
|
for (int c = 0; c < tile_image.colorCount(); c++) {
|
|
|
|
QColor color(tile_image.color(c));
|
|
|
|
color.setAlpha(alpha);
|
|
|
|
tile_image.setColor(c, color.rgba());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-26 00:13:33 +01:00
|
|
|
// The top layer of the metatile has its first color displayed at transparent.
|
2020-07-02 16:31:42 +01:00
|
|
|
if (l != bottomLayer) {
|
2018-09-26 00:13:33 +01:00
|
|
|
QColor color(tile_image.color(0));
|
|
|
|
color.setAlpha(0);
|
|
|
|
tile_image.setColor(0, color.rgba());
|
|
|
|
}
|
|
|
|
|
2021-12-01 21:06:14 +00:00
|
|
|
metatile_painter.drawImage(origin, tile_image.mirrored(tile.xflip, tile.yflip));
|
2018-09-26 00:13:33 +01:00
|
|
|
}
|
|
|
|
metatile_painter.end();
|
|
|
|
|
|
|
|
return metatile_image;
|
|
|
|
}
|
|
|
|
|
2021-12-01 21:06:14 +00:00
|
|
|
QImage getTileImage(uint16_t tileId, Tileset *primaryTileset, Tileset *secondaryTileset) {
|
|
|
|
Tileset *tileset = Tileset::getBlockTileset(tileId, primaryTileset, secondaryTileset);
|
|
|
|
int local_index = Metatile::getBlockIndex(tileId);
|
2021-02-17 02:45:54 +00:00
|
|
|
if (!tileset) {
|
2018-09-26 00:13:33 +01:00
|
|
|
return QImage();
|
|
|
|
}
|
2021-02-17 02:45:54 +00:00
|
|
|
return tileset->tiles.value(local_index, QImage());
|
2018-09-26 00:13:33 +01:00
|
|
|
}
|
2018-09-30 18:33:58 +01:00
|
|
|
|
2021-12-01 21:06:14 +00:00
|
|
|
QImage getColoredTileImage(uint16_t tileId, Tileset *primaryTileset, Tileset *secondaryTileset, QList<QRgb> palette) {
|
|
|
|
QImage tileImage = getTileImage(tileId, primaryTileset, secondaryTileset);
|
2018-10-03 01:01:21 +01:00
|
|
|
if (tileImage.isNull()) {
|
2021-12-18 07:27:23 +00:00
|
|
|
tileImage = QImage(8, 8, QImage::Format_RGBA8888);
|
2018-10-03 01:01:21 +01:00
|
|
|
QPainter painter(&tileImage);
|
2021-12-18 07:27:23 +00:00
|
|
|
painter.fillRect(0, 0, 8, 8, palette.at(0));
|
2018-10-03 01:01:21 +01:00
|
|
|
} else {
|
|
|
|
for (int i = 0; i < 16; i++) {
|
|
|
|
tileImage.setColor(i, palette.at(i));
|
|
|
|
}
|
2018-09-30 18:33:58 +01:00
|
|
|
}
|
2018-10-03 01:01:21 +01:00
|
|
|
|
2018-09-30 18:33:58 +01:00
|
|
|
return tileImage;
|
|
|
|
}
|
2019-06-17 18:00:31 +01:00
|
|
|
|
2021-12-01 21:06:14 +00:00
|
|
|
QImage getPalettedTileImage(uint16_t tileId, Tileset *primaryTileset, Tileset *secondaryTileset, int paletteId, bool useTruePalettes) {
|
2020-05-03 16:31:44 +01:00
|
|
|
QList<QRgb> palette = Tileset::getPalette(paletteId, primaryTileset, secondaryTileset, useTruePalettes);
|
2021-12-01 21:06:14 +00:00
|
|
|
return getColoredTileImage(tileId, primaryTileset, secondaryTileset, palette);
|
2019-06-17 18:00:31 +01:00
|
|
|
}
|
|
|
|
|
2021-12-01 21:06:14 +00:00
|
|
|
QImage getGreyscaleTileImage(uint16_t tileId, Tileset *primaryTileset, Tileset *secondaryTileset) {
|
|
|
|
return getColoredTileImage(tileId, primaryTileset, secondaryTileset, greyscalePalette);
|
2019-06-17 18:00:31 +01:00
|
|
|
}
|