2023-06-20 14:33:09 +10:00
|
|
|
#include "StarImageMetadataDatabase.hpp"
|
|
|
|
#include "StarFile.hpp"
|
|
|
|
#include "StarImage.hpp"
|
|
|
|
#include "StarImageProcessing.hpp"
|
|
|
|
#include "StarLogging.hpp"
|
|
|
|
#include "StarEncode.hpp"
|
|
|
|
#include "StarGameTypes.hpp"
|
|
|
|
#include "StarRoot.hpp"
|
|
|
|
#include "StarAssets.hpp"
|
|
|
|
|
|
|
|
namespace Star {
|
|
|
|
|
2023-06-24 22:49:47 +10:00
|
|
|
Vec2U ImageMetadataDatabase::imageSize(AssetPath const& path) const {
|
2023-06-20 14:33:09 +10:00
|
|
|
MutexLocker locker(m_mutex);
|
|
|
|
auto i = m_sizeCache.find(path);
|
|
|
|
if (i != m_sizeCache.end())
|
|
|
|
return i->second;
|
|
|
|
|
|
|
|
locker.unlock();
|
|
|
|
Vec2U size = calculateImageSize(path);
|
|
|
|
|
|
|
|
locker.lock();
|
|
|
|
m_sizeCache[path] = size;
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2023-06-24 22:49:47 +10:00
|
|
|
List<Vec2I> ImageMetadataDatabase::imageSpaces(AssetPath const& path, Vec2F position, float fillLimit, bool flip) const {
|
2023-06-20 14:33:09 +10:00
|
|
|
SpacesEntry key = make_tuple(path, Vec2I::round(position), fillLimit, flip);
|
|
|
|
|
|
|
|
MutexLocker locker(m_mutex);
|
|
|
|
auto i = m_spacesCache.find(key);
|
|
|
|
if (i != m_spacesCache.end()) {
|
|
|
|
return i->second;
|
|
|
|
}
|
|
|
|
|
2023-06-24 22:49:47 +10:00
|
|
|
auto filteredPath = filterProcessing(path);
|
2023-06-20 14:33:09 +10:00
|
|
|
SpacesEntry filteredKey = make_tuple(filteredPath, Vec2I::round(position), fillLimit, flip);
|
|
|
|
|
|
|
|
auto j = m_spacesCache.find(filteredKey);
|
|
|
|
if (j != m_spacesCache.end()) {
|
|
|
|
auto spaces = j->second;
|
|
|
|
m_spacesCache[key] = spaces;
|
|
|
|
return spaces;
|
|
|
|
}
|
|
|
|
|
|
|
|
locker.unlock();
|
|
|
|
|
|
|
|
auto image = Root::singleton().assets()->image(filteredPath);
|
|
|
|
int imageWidth = image->width();
|
|
|
|
int imageHeight = image->height();
|
|
|
|
|
|
|
|
Vec2I min((position / TilePixels).floor());
|
|
|
|
Vec2I max(((Vec2F(imageWidth, imageHeight) + position) / TilePixels).ceil());
|
|
|
|
|
|
|
|
List<Vec2I> spaces;
|
|
|
|
|
|
|
|
for (int yspace = min[1]; yspace < max[1]; ++yspace) {
|
|
|
|
for (int xspace = min[0]; xspace < max[0]; ++xspace) {
|
|
|
|
float fillRatio = 0.0f;
|
|
|
|
|
|
|
|
for (int y = 0; y < (int)TilePixels; ++y) {
|
|
|
|
int ypixel = round(yspace * (int)TilePixels + y - position[1]);
|
|
|
|
if (ypixel < 0 || ypixel >= imageHeight)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (int x = 0; x < (int)TilePixels; ++x) {
|
|
|
|
int xpixel = round(xspace * (int)TilePixels + x - position[0]);
|
|
|
|
if (flip)
|
|
|
|
xpixel = imageWidth - 1 - xpixel;
|
|
|
|
|
|
|
|
if (xpixel < 0 || xpixel >= imageWidth)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (image->get(xpixel, ypixel)[3] > 0)
|
|
|
|
fillRatio += 1.0f / square(TilePixels);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fillRatio >= fillLimit)
|
|
|
|
spaces.append(Vec2I(xspace, yspace));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
locker.lock();
|
|
|
|
m_spacesCache[key] = spaces;
|
|
|
|
m_spacesCache[filteredKey] = spaces;
|
|
|
|
|
|
|
|
return spaces;
|
|
|
|
}
|
|
|
|
|
2023-06-24 22:49:47 +10:00
|
|
|
RectU ImageMetadataDatabase::nonEmptyRegion(AssetPath const& path) const {
|
2023-06-20 14:33:09 +10:00
|
|
|
MutexLocker locker(m_mutex);
|
|
|
|
auto i = m_regionCache.find(path);
|
|
|
|
if (i != m_regionCache.end()) {
|
|
|
|
return i->second;
|
|
|
|
}
|
|
|
|
|
2023-06-24 22:49:47 +10:00
|
|
|
auto filteredPath = filterProcessing(path);
|
2023-06-20 14:33:09 +10:00
|
|
|
auto j = m_regionCache.find(filteredPath);
|
|
|
|
if (j != m_regionCache.end()) {
|
|
|
|
m_regionCache[path] = j->second;
|
|
|
|
return j->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
locker.unlock();
|
|
|
|
auto image = Root::singleton().assets()->image(filteredPath);
|
|
|
|
RectU region = RectU::null();
|
|
|
|
image->forEachPixel([®ion](unsigned x, unsigned y, Vec4B const& pixel) {
|
|
|
|
if (pixel[3] > 0)
|
|
|
|
region.combine(RectU::withSize({x, y}, {1, 1}));
|
|
|
|
});
|
|
|
|
|
|
|
|
locker.lock();
|
|
|
|
m_regionCache[path] = region;
|
|
|
|
m_regionCache[filteredPath] = region;
|
|
|
|
|
|
|
|
return region;
|
|
|
|
}
|
|
|
|
|
2023-06-24 22:49:47 +10:00
|
|
|
AssetPath ImageMetadataDatabase::filterProcessing(AssetPath const& path) {
|
|
|
|
AssetPath newPath = { path.basePath, path.subPath, {} };
|
2023-06-24 01:30:55 +10:00
|
|
|
|
2023-06-26 01:42:18 +10:00
|
|
|
String filtered;
|
2023-06-27 03:38:57 +10:00
|
|
|
for (auto& directives : path.directives.list())
|
|
|
|
directives.loadOperations();
|
2023-06-26 01:42:18 +10:00
|
|
|
path.directives.forEach([&](auto const& entry, Directives const& directives) {
|
2023-06-24 19:41:52 +10:00
|
|
|
ImageOperation const& operation = entry.operation;
|
2023-06-26 01:42:18 +10:00
|
|
|
if (!(operation.is<HueShiftImageOperation>() ||
|
|
|
|
operation.is<SaturationShiftImageOperation>() ||
|
|
|
|
operation.is<BrightnessMultiplyImageOperation>() ||
|
|
|
|
operation.is<FadeToColorImageOperation>() ||
|
|
|
|
operation.is<ScanLinesImageOperation>() ||
|
|
|
|
operation.is<SetColorImageOperation>())) {
|
|
|
|
filtered += "?";
|
2024-04-22 06:07:59 +10:00
|
|
|
filtered += entry.string(*directives);
|
2023-06-24 01:30:55 +10:00
|
|
|
}
|
2023-06-26 01:42:18 +10:00
|
|
|
});
|
2023-06-20 14:33:09 +10:00
|
|
|
|
2024-02-19 16:55:19 +01:00
|
|
|
newPath.directives = std::move(filtered);
|
2023-06-24 22:49:47 +10:00
|
|
|
return newPath;
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
2023-06-24 22:49:47 +10:00
|
|
|
Vec2U ImageMetadataDatabase::calculateImageSize(AssetPath const& path) const {
|
2023-06-20 14:33:09 +10:00
|
|
|
// Carefully calculate an image's size while trying not to actually load it.
|
|
|
|
// In error cases, this will fall back to calling Assets::image, so that image
|
|
|
|
// can possibly produce a missing image asset or properly report the error.
|
|
|
|
|
|
|
|
auto assets = Root::singleton().assets();
|
|
|
|
|
|
|
|
auto fallback = [&assets, &path]() {
|
|
|
|
return assets->image(path)->size();
|
|
|
|
};
|
|
|
|
|
2023-06-24 22:49:47 +10:00
|
|
|
if (!assets->assetExists(path.basePath)) {
|
2023-06-20 14:33:09 +10:00
|
|
|
return fallback();
|
|
|
|
}
|
|
|
|
|
|
|
|
Vec2U imageSize;
|
2023-06-24 22:49:47 +10:00
|
|
|
if (path.subPath) {
|
|
|
|
auto frames = assets->imageFrames(path.basePath);
|
2023-06-20 14:33:09 +10:00
|
|
|
if (!frames)
|
|
|
|
return fallback();
|
|
|
|
|
2023-06-24 22:49:47 +10:00
|
|
|
if (auto rect = frames->getRect(*path.subPath))
|
2023-06-20 14:33:09 +10:00
|
|
|
imageSize = rect->size();
|
|
|
|
else
|
|
|
|
return fallback();
|
|
|
|
} else {
|
|
|
|
// We ensure that the base image size is cached even when given directives,
|
|
|
|
// so we don't have to call Image::readPngMetadata on the same file more
|
|
|
|
// than once.
|
|
|
|
MutexLocker locker(m_mutex);
|
2023-06-24 22:49:47 +10:00
|
|
|
if (auto size = m_sizeCache.maybe(path.basePath)) {
|
2023-06-20 14:33:09 +10:00
|
|
|
imageSize = *size;
|
|
|
|
} else {
|
|
|
|
locker.unlock();
|
2024-09-10 23:04:09 -04:00
|
|
|
auto file = assets->openFile(path.basePath);
|
|
|
|
if (Image::isPng(file))
|
|
|
|
imageSize = get<0>(Image::readPngMetadata(file));
|
|
|
|
else
|
|
|
|
imageSize = fallback();
|
2023-06-20 14:33:09 +10:00
|
|
|
locker.lock();
|
2023-06-24 22:49:47 +10:00
|
|
|
m_sizeCache[path.basePath] = imageSize;
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct OperationSizeAdjust {
|
|
|
|
Vec2U& imageSize;
|
|
|
|
bool hasError;
|
|
|
|
|
|
|
|
OperationSizeAdjust(Vec2U& size) : imageSize(size), hasError(false) {};
|
|
|
|
|
2023-06-27 03:38:57 +10:00
|
|
|
void operator()(NullImageOperation const&) {}
|
|
|
|
|
2023-06-25 14:00:20 +10:00
|
|
|
void operator()(ErrorImageOperation const&) {}
|
|
|
|
|
2023-06-20 14:33:09 +10:00
|
|
|
void operator()(HueShiftImageOperation const&) {}
|
|
|
|
|
|
|
|
void operator()(SaturationShiftImageOperation const&) {}
|
|
|
|
|
|
|
|
void operator()(BrightnessMultiplyImageOperation const&) {}
|
|
|
|
|
|
|
|
void operator()(FadeToColorImageOperation const&) {}
|
|
|
|
|
|
|
|
void operator()(ScanLinesImageOperation const&) {}
|
|
|
|
|
|
|
|
void operator()(SetColorImageOperation const&) {}
|
|
|
|
|
|
|
|
void operator()(ColorReplaceImageOperation const&) {}
|
|
|
|
|
|
|
|
void operator()(AlphaMaskImageOperation const&) {}
|
|
|
|
|
|
|
|
void operator()(BlendImageOperation const&) {}
|
|
|
|
|
|
|
|
void operator()(MultiplyImageOperation const&) {}
|
|
|
|
|
|
|
|
void operator()(BorderImageOperation const& bio) {
|
|
|
|
imageSize += Vec2U::filled(bio.pixels * 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
void operator()(ScaleImageOperation const& sio) {
|
|
|
|
imageSize = Vec2U::round(vmult(Vec2F(imageSize), sio.scale));
|
|
|
|
}
|
|
|
|
|
|
|
|
void operator()(CropImageOperation const& cio) {
|
|
|
|
if (cio.subset.isEmpty() ||
|
|
|
|
cio.subset.xMin() < 0 ||
|
|
|
|
cio.subset.yMin() < 0 ||
|
|
|
|
(unsigned)cio.subset.xMax() > imageSize[0] ||
|
|
|
|
(unsigned)cio.subset.yMax() > imageSize[1]) {
|
|
|
|
hasError = true;
|
|
|
|
} else {
|
|
|
|
imageSize = Vec2U(cio.subset.size());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void operator()(FlipImageOperation const&) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
OperationSizeAdjust osa(imageSize);
|
|
|
|
|
2023-06-27 03:38:57 +10:00
|
|
|
for (auto& directives : path.directives.list())
|
|
|
|
directives.loadOperations();
|
|
|
|
|
2024-02-28 18:11:55 +01:00
|
|
|
bool complete = path.directives.forEachAbortable([&](auto const& entry, Directives const&) -> bool {
|
2023-06-24 19:41:52 +10:00
|
|
|
entry.operation.call(osa);
|
|
|
|
return !osa.hasError;
|
2023-06-24 01:30:55 +10:00
|
|
|
});
|
2023-06-20 14:33:09 +10:00
|
|
|
|
2023-06-24 01:30:55 +10:00
|
|
|
if (!complete)
|
|
|
|
return fallback();
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
return imageSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|