2017-03-11 13:43:42 +00:00
|
|
|
#include <algorithm>
|
|
|
|
|
2019-10-11 15:34:38 +00:00
|
|
|
#include "base/colorutil.h"
|
|
|
|
#include "base/timeutil.h"
|
2016-12-27 21:26:49 +00:00
|
|
|
#include "thin3d/thin3d.h"
|
|
|
|
#include "image/zim_load.h"
|
|
|
|
#include "image/png_load.h"
|
2017-03-11 13:43:42 +00:00
|
|
|
#include "math/math_util.h"
|
2019-10-11 15:34:38 +00:00
|
|
|
#include "math/curves.h"
|
2016-12-27 21:26:49 +00:00
|
|
|
#include "file/vfs.h"
|
|
|
|
#include "ext/jpge/jpgd.h"
|
2019-10-11 15:34:38 +00:00
|
|
|
#include "ui/view.h"
|
|
|
|
#include "ui/ui_context.h"
|
|
|
|
#include "gfx_es2/draw_buffer.h"
|
2018-03-27 21:10:33 +00:00
|
|
|
#include "Common/Log.h"
|
2019-10-11 15:34:38 +00:00
|
|
|
#include "UI/TextureUtil.h"
|
|
|
|
#include "UI/GameInfoCache.h"
|
2016-12-27 21:26:49 +00:00
|
|
|
|
|
|
|
static Draw::DataFormat ZimToT3DFormat(int zim) {
|
|
|
|
switch (zim) {
|
|
|
|
case ZIM_RGBA8888: return Draw::DataFormat::R8G8B8A8_UNORM;
|
|
|
|
default: return Draw::DataFormat::R8G8B8A8_UNORM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static ImageFileType DetectImageFileType(const uint8_t *data, size_t size) {
|
|
|
|
if (!memcmp(data, "ZIMG", 4)) {
|
|
|
|
return ZIM;
|
|
|
|
}
|
|
|
|
else if (!memcmp(data, "\x89\x50\x4E\x47", 4)) {
|
|
|
|
return PNG;
|
|
|
|
}
|
|
|
|
else if (!memcmp(data, "\xff\xd8\xff\xe0", 4)) {
|
|
|
|
return JPEG;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return TYPE_UNKNOWN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool LoadTextureLevels(const uint8_t *data, size_t size, ImageFileType type, int width[16], int height[16], int *num_levels, Draw::DataFormat *fmt, uint8_t *image[16], int *zim_flags) {
|
|
|
|
if (type == DETECT) {
|
|
|
|
type = DetectImageFileType(data, size);
|
|
|
|
}
|
|
|
|
if (type == TYPE_UNKNOWN) {
|
2018-03-27 21:10:33 +00:00
|
|
|
ELOG("File (size: %d) has unknown format", (int)size);
|
2016-12-27 21:26:49 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*num_levels = 0;
|
|
|
|
*zim_flags = 0;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case ZIM:
|
|
|
|
{
|
|
|
|
*num_levels = LoadZIMPtr((const uint8_t *)data, size, width, height, zim_flags, image);
|
|
|
|
*fmt = ZimToT3DFormat(*zim_flags & ZIM_FORMAT_MASK);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PNG:
|
|
|
|
if (1 == pngLoadPtr((const unsigned char *)data, size, &width[0], &height[0], &image[0], false)) {
|
|
|
|
*num_levels = 1;
|
|
|
|
*fmt = Draw::DataFormat::R8G8B8A8_UNORM;
|
2017-03-05 11:30:42 +00:00
|
|
|
if (!image[0]) {
|
|
|
|
ELOG("WTF");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ELOG("PNG load failed");
|
|
|
|
return false;
|
2016-12-27 21:26:49 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JPEG:
|
|
|
|
{
|
|
|
|
int actual_components = 0;
|
|
|
|
unsigned char *jpegBuf = jpgd::decompress_jpeg_image_from_memory(data, (int)size, &width[0], &height[0], &actual_components, 4);
|
|
|
|
if (jpegBuf) {
|
|
|
|
*num_levels = 1;
|
|
|
|
*fmt = Draw::DataFormat::R8G8B8A8_UNORM;
|
|
|
|
image[0] = (uint8_t *)jpegBuf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2018-03-27 21:10:33 +00:00
|
|
|
ELOG("Unsupported image format %d", (int)type);
|
2016-12-27 21:26:49 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return *num_levels > 0;
|
|
|
|
}
|
|
|
|
|
2017-03-11 13:43:42 +00:00
|
|
|
bool ManagedTexture::LoadFromFileData(const uint8_t *data, size_t dataSize, ImageFileType type, bool generateMips) {
|
2018-03-27 21:10:33 +00:00
|
|
|
generateMips_ = generateMips;
|
2017-01-16 16:43:07 +00:00
|
|
|
using namespace Draw;
|
|
|
|
|
2017-01-28 22:16:37 +00:00
|
|
|
int width[16]{}, height[16]{};
|
|
|
|
uint8_t *image[16]{};
|
2016-12-27 21:26:49 +00:00
|
|
|
|
2017-01-28 22:16:37 +00:00
|
|
|
int num_levels = 0;
|
|
|
|
int zim_flags = 0;
|
2017-01-16 16:43:07 +00:00
|
|
|
DataFormat fmt;
|
2016-12-27 21:26:49 +00:00
|
|
|
if (!LoadTextureLevels(data, dataSize, type, width, height, &num_levels, &fmt, image, &zim_flags)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-03-05 11:30:42 +00:00
|
|
|
if (!image[0]) {
|
|
|
|
Crash();
|
|
|
|
}
|
|
|
|
|
2016-12-27 21:26:49 +00:00
|
|
|
if (num_levels < 0 || num_levels >= 16) {
|
|
|
|
ELOG("Invalid num_levels: %d. Falling back to one. Image: %dx%d", num_levels, width[0], height[0]);
|
|
|
|
num_levels = 1;
|
|
|
|
}
|
2017-01-16 16:34:53 +00:00
|
|
|
|
2018-11-22 22:53:58 +00:00
|
|
|
// Free the old texture, if any.
|
2017-01-28 22:16:37 +00:00
|
|
|
if (texture_) {
|
2017-01-16 16:34:53 +00:00
|
|
|
delete texture_;
|
2017-01-28 22:16:37 +00:00
|
|
|
texture_ = nullptr;
|
|
|
|
}
|
2017-01-16 16:34:53 +00:00
|
|
|
|
2017-03-11 13:43:42 +00:00
|
|
|
int potentialLevels = std::min(log2i(width[0]), log2i(height[0]));
|
|
|
|
|
2017-01-16 16:43:07 +00:00
|
|
|
TextureDesc desc{};
|
|
|
|
desc.type = TextureType::LINEAR2D;
|
|
|
|
desc.format = fmt;
|
|
|
|
desc.width = width[0];
|
|
|
|
desc.height = height[0];
|
|
|
|
desc.depth = 1;
|
2017-03-11 13:43:42 +00:00
|
|
|
desc.mipLevels = generateMips ? potentialLevels : num_levels;
|
|
|
|
desc.generateMips = generateMips && potentialLevels > num_levels;
|
2018-04-07 04:32:33 +00:00
|
|
|
desc.tag = "LoadedFileData";
|
2017-01-17 17:31:44 +00:00
|
|
|
for (int i = 0; i < num_levels; i++) {
|
|
|
|
desc.initData.push_back(image[i]);
|
|
|
|
}
|
2017-01-16 16:43:07 +00:00
|
|
|
texture_ = draw_->CreateTexture(desc);
|
2016-12-27 21:26:49 +00:00
|
|
|
for (int i = 0; i < num_levels; i++) {
|
2017-01-17 17:31:44 +00:00
|
|
|
if (image[i])
|
2016-12-27 21:26:49 +00:00
|
|
|
free(image[i]);
|
|
|
|
}
|
2018-11-22 22:53:58 +00:00
|
|
|
return texture_;
|
2016-12-27 21:26:49 +00:00
|
|
|
}
|
|
|
|
|
2017-03-11 13:43:42 +00:00
|
|
|
bool ManagedTexture::LoadFromFile(const std::string &filename, ImageFileType type, bool generateMips) {
|
2018-03-27 21:10:33 +00:00
|
|
|
generateMips_ = generateMips;
|
2016-12-27 21:26:49 +00:00
|
|
|
size_t fileSize;
|
|
|
|
uint8_t *buffer = VFSReadFile(filename.c_str(), &fileSize);
|
|
|
|
if (!buffer) {
|
2018-03-27 21:10:33 +00:00
|
|
|
filename_ = "";
|
|
|
|
ELOG("Failed to read file '%s'", filename.c_str());
|
2016-12-27 21:26:49 +00:00
|
|
|
return false;
|
|
|
|
}
|
2017-03-11 13:43:42 +00:00
|
|
|
bool retval = LoadFromFileData(buffer, fileSize, type, generateMips);
|
2016-12-27 21:26:49 +00:00
|
|
|
if (retval) {
|
|
|
|
filename_ = filename;
|
2017-02-06 10:55:54 +00:00
|
|
|
} else {
|
2018-03-27 21:10:33 +00:00
|
|
|
filename_ = "";
|
|
|
|
ELOG("Failed to load texture '%s'", filename.c_str());
|
2016-12-27 21:26:49 +00:00
|
|
|
}
|
|
|
|
delete[] buffer;
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2017-05-18 10:52:03 +00:00
|
|
|
std::unique_ptr<ManagedTexture> CreateTextureFromFile(Draw::DrawContext *draw, const char *filename, ImageFileType type, bool generateMips) {
|
2017-02-06 10:55:54 +00:00
|
|
|
if (!draw)
|
2017-05-18 10:52:03 +00:00
|
|
|
return std::unique_ptr<ManagedTexture>();
|
2017-05-18 10:41:42 +00:00
|
|
|
// TODO: Load the texture on a background thread.
|
2016-12-27 21:26:49 +00:00
|
|
|
ManagedTexture *mtex = new ManagedTexture(draw);
|
2017-03-11 13:43:42 +00:00
|
|
|
if (!mtex->LoadFromFile(filename, type, generateMips)) {
|
2016-12-27 21:26:49 +00:00
|
|
|
delete mtex;
|
2017-05-18 10:52:03 +00:00
|
|
|
return std::unique_ptr<ManagedTexture>();
|
2016-12-27 21:26:49 +00:00
|
|
|
}
|
2018-03-01 12:50:56 +00:00
|
|
|
return std::unique_ptr<ManagedTexture>(mtex);
|
2016-12-27 21:26:49 +00:00
|
|
|
}
|
|
|
|
|
2018-03-27 21:10:33 +00:00
|
|
|
void ManagedTexture::DeviceLost() {
|
|
|
|
ILOG("ManagedTexture::DeviceLost(%s)", filename_.c_str());
|
|
|
|
if (texture_)
|
|
|
|
texture_->Release();
|
|
|
|
texture_ = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ManagedTexture::DeviceRestored(Draw::DrawContext *draw) {
|
|
|
|
ILOG("ManagedTexture::DeviceRestored(%s)", filename_.c_str());
|
|
|
|
_assert_(!texture_);
|
|
|
|
draw_ = draw;
|
|
|
|
// Vulkan: Can't load textures before the first frame has started.
|
|
|
|
// Should probably try to lift that restriction again someday..
|
|
|
|
loadPending_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Draw::Texture *ManagedTexture::GetTexture() {
|
|
|
|
if (loadPending_) {
|
|
|
|
if (!LoadFromFile(filename_, ImageFileType::DETECT, generateMips_)) {
|
|
|
|
ELOG("ManagedTexture failed: '%s'", filename_.c_str());
|
|
|
|
}
|
|
|
|
loadPending_ = false;
|
|
|
|
}
|
|
|
|
return texture_;
|
|
|
|
}
|
|
|
|
|
2016-12-27 21:26:49 +00:00
|
|
|
// TODO: Remove the code duplication between this and LoadFromFileData
|
2017-05-18 10:52:03 +00:00
|
|
|
std::unique_ptr<ManagedTexture> CreateTextureFromFileData(Draw::DrawContext *draw, const uint8_t *data, int size, ImageFileType type, bool generateMips) {
|
2017-02-06 10:55:54 +00:00
|
|
|
if (!draw)
|
2017-05-18 10:52:03 +00:00
|
|
|
return std::unique_ptr<ManagedTexture>();
|
2016-12-27 21:26:49 +00:00
|
|
|
ManagedTexture *mtex = new ManagedTexture(draw);
|
2018-11-19 07:27:46 +00:00
|
|
|
if (mtex->LoadFromFileData(data, size, type, generateMips)) {
|
|
|
|
return std::unique_ptr<ManagedTexture>(mtex);
|
|
|
|
} else {
|
|
|
|
// Best to return a null pointer if we fail!
|
|
|
|
delete mtex;
|
|
|
|
return std::unique_ptr<ManagedTexture>();
|
|
|
|
}
|
2018-03-01 11:11:10 +00:00
|
|
|
}
|
2019-10-11 15:34:38 +00:00
|
|
|
|
|
|
|
void GameIconView::GetContentDimensions(const UIContext &dc, float &w, float &h) const {
|
|
|
|
w = textureWidth_;
|
|
|
|
h = textureHeight_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameIconView::Draw(UIContext &dc) {
|
|
|
|
using namespace UI;
|
|
|
|
std::shared_ptr<GameInfo> info = g_gameInfoCache->GetInfo(NULL, gamePath_, GAMEINFO_WANTBG | GAMEINFO_WANTSIZE);
|
|
|
|
|
|
|
|
if (!info->icon.texture) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-03-24 00:43:58 +00:00
|
|
|
textureWidth_ = info->icon.texture->Width() * scale_;
|
|
|
|
textureHeight_ = info->icon.texture->Height() * scale_;
|
2019-10-11 15:34:38 +00:00
|
|
|
|
|
|
|
// Fade icon with the backgrounds.
|
|
|
|
double loadTime = info->icon.timeLoaded;
|
|
|
|
if (info->pic1.texture) {
|
|
|
|
loadTime = std::max(loadTime, info->pic1.timeLoaded);
|
|
|
|
}
|
|
|
|
if (info->pic0.texture) {
|
|
|
|
loadTime = std::max(loadTime, info->pic0.timeLoaded);
|
|
|
|
}
|
|
|
|
uint32_t color = whiteAlpha(ease((time_now_d() - loadTime) * 3));
|
|
|
|
|
2020-03-24 00:43:58 +00:00
|
|
|
// Adjust size so we don't stretch the image vertically or horizontally.
|
|
|
|
// Make sure it's not wider than 144 (like Doom Legacy homebrew), ugly in the grid mode.
|
|
|
|
float nw = std::min(bounds_.h * textureWidth_ / textureHeight_, (float)bounds_.w);
|
|
|
|
|
2019-10-11 15:34:38 +00:00
|
|
|
dc.Flush();
|
|
|
|
dc.GetDrawContext()->BindTexture(0, info->icon.texture->GetTexture());
|
2020-03-24 00:43:58 +00:00
|
|
|
dc.Draw()->Rect(bounds_.x, bounds_.y, nw, bounds_.h, color);
|
2019-10-11 15:34:38 +00:00
|
|
|
dc.Flush();
|
|
|
|
dc.RebindTexture();
|
|
|
|
}
|