osb/source/game/StarImageMetadataDatabase.cpp

272 lines
7.6 KiB
C++
Raw Normal View History

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 {
ImageMetadataDatabase::ImageMetadataDatabase() {
MutexLocker locker(m_mutex);
int timeSmear = 2000;
int64_t timeToLive = 60000;
m_sizeCache.setTimeSmear(timeSmear);
m_spacesCache.setTimeSmear(timeSmear);
m_regionCache.setTimeSmear(timeSmear);
m_sizeCache.setTimeToLive(timeToLive);
m_spacesCache.setTimeToLive(timeToLive);
m_regionCache.setTimeToLive(timeToLive);
}
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);
if (auto cached = m_sizeCache.ptr(path))
return *cached;
2023-06-20 14:33:09 +10:00
locker.unlock();
Vec2U size = calculateImageSize(path);
locker.lock();
m_sizeCache.set(path, size);
2023-06-20 14:33:09 +10:00
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);
if (auto cached = m_spacesCache.ptr(key)) {
return *cached;
2023-06-20 14:33:09 +10:00
}
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);
if (auto cached = m_spacesCache.ptr(filteredKey)) {
m_spacesCache.set(key, *cached);
return *cached;
2023-06-20 14:33:09 +10:00
}
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.set(key, spaces);
m_spacesCache.set(filteredKey, spaces);
2023-06-20 14:33:09 +10:00
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);
if (auto cached = m_regionCache.ptr(path)) {
return *cached;
2023-06-20 14:33:09 +10:00
}
2023-06-24 22:49:47 +10:00
auto filteredPath = filterProcessing(path);
if (auto cached = m_regionCache.ptr(filteredPath)) {
m_regionCache.set(path, *cached);
return *cached;
2023-06-20 14:33:09 +10:00
}
locker.unlock();
auto image = Root::singleton().assets()->image(filteredPath);
RectU region = RectU::null();
image->forEachPixel([&region](unsigned x, unsigned y, Vec4B const& pixel) {
if (pixel[3] > 0)
region.combine(RectU::withSize({x, y}, {1, 1}));
});
locker.lock();
m_regionCache.set(path, region);
m_regionCache.set(filteredPath, region);
2023-06-20 14:33:09 +10:00
return region;
}
void ImageMetadataDatabase::cleanup() const {
MutexLocker locker(m_mutex);
m_sizeCache.cleanup();
m_spacesCache.cleanup();
m_regionCache.cleanup();
}
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
String filtered;
for (auto& directives : path.directives.list())
directives.loadOperations();
path.directives.forEach([&](auto const& entry, Directives const& directives) {
2023-06-24 19:41:52 +10:00
ImageOperation const& operation = entry.operation;
if (!(operation.is<HueShiftImageOperation>() ||
operation.is<SaturationShiftImageOperation>() ||
operation.is<BrightnessMultiplyImageOperation>() ||
operation.is<FadeToColorImageOperation>() ||
operation.is<ScanLinesImageOperation>() ||
operation.is<SetColorImageOperation>())) {
filtered += "?";
filtered += entry.string(*directives);
2023-06-24 01:30:55 +10:00
}
});
2023-06-20 14:33:09 +10: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);
if (auto size = m_sizeCache.ptr(path.basePath)) {
2023-06-20 14:33:09 +10:00
imageSize = *size;
} else {
locker.unlock();
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();
m_sizeCache.set(path.basePath, imageSize);
2023-06-20 14:33:09 +10:00
}
}
struct OperationSizeAdjust {
Vec2U& imageSize;
bool hasError;
OperationSizeAdjust(Vec2U& size) : imageSize(size), hasError(false) {};
void operator()(NullImageOperation const&) {}
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);
for (auto& directives : path.directives.list())
directives.loadOperations();
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;
}
}