VideoCommon: set individual texture asset filter/wrap values when loading from json

This commit is contained in:
iwubcode 2024-05-31 23:31:10 -05:00
parent 75a0a1ed67
commit 1c3402502b
1 changed files with 124 additions and 67 deletions

View File

@ -3,6 +3,9 @@
#include "VideoCommon/Assets/TextureAsset.h" #include "VideoCommon/Assets/TextureAsset.h"
#include <optional>
#include "Common/JsonUtil.h"
#include "Common/Logging/Log.h" #include "Common/Logging/Log.h"
#include "Common/StringUtil.h" #include "Common/StringUtil.h"
#include "VideoCommon/BPMemory.h" #include "VideoCommon/BPMemory.h"
@ -11,6 +14,45 @@ namespace VideoCommon
{ {
namespace namespace
{ {
std::optional<WrapMode> ReadWrapModeFromJSON(const picojson::object& json, const std::string& uv)
{
auto uv_mode = ReadStringFromJson(json, uv).value_or("");
Common::ToLower(&uv_mode);
if (uv_mode == "clamp")
{
return WrapMode::Clamp;
}
else if (uv_mode == "repeat")
{
return WrapMode::Repeat;
}
else if (uv_mode == "mirror")
{
return WrapMode::Mirror;
}
return std::nullopt;
}
std::optional<FilterMode> ReadFilterModeFromJSON(const picojson::object& json,
const std::string& filter)
{
auto filter_mode = ReadStringFromJson(json, filter).value_or("");
Common::ToLower(&filter_mode);
if (filter_mode == "linear")
{
return FilterMode::Linear;
}
else if (filter_mode == "near")
{
return FilterMode::Near;
}
return std::nullopt;
}
bool ParseSampler(const VideoCommon::CustomAssetLibrary::AssetID& asset_id, bool ParseSampler(const VideoCommon::CustomAssetLibrary::AssetID& asset_id,
const picojson::object& json, SamplerState* sampler) const picojson::object& json, SamplerState* sampler)
{ {
@ -19,78 +61,93 @@ bool ParseSampler(const VideoCommon::CustomAssetLibrary::AssetID& asset_id,
*sampler = RenderState::GetLinearSamplerState(); *sampler = RenderState::GetLinearSamplerState();
const auto sampler_state_mode_iter = json.find("texture_mode"); const auto sampler_state_wrap_iter = json.find("wrap_mode");
if (sampler_state_mode_iter == json.end()) if (sampler_state_wrap_iter != json.end())
{ {
ERROR_LOG_FMT(VIDEO, "Asset '{}' failed to parse json, 'texture_mode' not found", asset_id); if (!sampler_state_wrap_iter->second.is<picojson::object>())
return false; {
} ERROR_LOG_FMT(VIDEO, "Asset '{}' failed to parse json, 'wrap_mode' is not the right type",
if (!sampler_state_mode_iter->second.is<std::string>()) asset_id);
{ return false;
ERROR_LOG_FMT(VIDEO, "Asset '{}' failed to parse json, 'texture_mode' is not the right type", }
asset_id); const auto sampler_state_wrap_obj = sampler_state_wrap_iter->second.get<picojson::object>();
return false;
}
std::string sampler_state_mode = sampler_state_mode_iter->second.to_str();
Common::ToLower(&sampler_state_mode);
if (sampler_state_mode == "clamp") if (const auto mode = ReadWrapModeFromJSON(sampler_state_wrap_obj, "u"))
{ {
sampler->tm0.wrap_u = WrapMode::Clamp; sampler->tm0.wrap_u = *mode;
sampler->tm0.wrap_v = WrapMode::Clamp; }
} else
else if (sampler_state_mode == "repeat") {
{ ERROR_LOG_FMT(VIDEO,
sampler->tm0.wrap_u = WrapMode::Repeat; "Asset '{}' failed to parse json, 'wrap_mode[u]' has an invalid "
sampler->tm0.wrap_v = WrapMode::Repeat; "value",
} asset_id);
else if (sampler_state_mode == "mirrored_repeat") return false;
{ }
sampler->tm0.wrap_u = WrapMode::Mirror;
sampler->tm0.wrap_v = WrapMode::Mirror; if (const auto mode = ReadWrapModeFromJSON(sampler_state_wrap_obj, "v"))
} {
else sampler->tm0.wrap_v = *mode;
{ }
ERROR_LOG_FMT(VIDEO, else
"Asset '{}' failed to parse json, 'texture_mode' has an invalid " {
"value '{}'", ERROR_LOG_FMT(VIDEO,
asset_id, sampler_state_mode); "Asset '{}' failed to parse json, 'wrap_mode[v]' has an invalid "
return false; "value",
asset_id);
return false;
}
} }
const auto sampler_state_filter_iter = json.find("texture_filter"); const auto sampler_state_filter_iter = json.find("filter_mode");
if (sampler_state_filter_iter == json.end()) if (sampler_state_filter_iter != json.end())
{ {
ERROR_LOG_FMT(VIDEO, "Asset '{}' failed to parse json, 'texture_filter' not found", asset_id); if (!sampler_state_filter_iter->second.is<picojson::object>())
return false; {
} ERROR_LOG_FMT(VIDEO, "Asset '{}' failed to parse json, 'filter_mode' is not the right type",
if (!sampler_state_filter_iter->second.is<std::string>()) asset_id);
{ return false;
ERROR_LOG_FMT(VIDEO, "Asset '{}' failed to parse json, 'texture_filter' is not the right type", }
asset_id); const auto sampler_state_filter_obj = sampler_state_filter_iter->second.get<picojson::object>();
return false;
} if (const auto mode = ReadFilterModeFromJSON(sampler_state_filter_obj, "min"))
std::string sampler_state_filter = sampler_state_filter_iter->second.to_str(); {
Common::ToLower(&sampler_state_filter); sampler->tm0.min_filter = *mode;
if (sampler_state_filter == "linear") }
{ else
sampler->tm0.min_filter = FilterMode::Linear; {
sampler->tm0.mag_filter = FilterMode::Linear; ERROR_LOG_FMT(VIDEO,
sampler->tm0.mipmap_filter = FilterMode::Linear; "Asset '{}' failed to parse json, 'filter_mode[min]' has an invalid "
} "value",
else if (sampler_state_filter == "near") asset_id);
{ return false;
sampler->tm0.min_filter = FilterMode::Near; }
sampler->tm0.mag_filter = FilterMode::Near;
sampler->tm0.mipmap_filter = FilterMode::Near; if (const auto mode = ReadFilterModeFromJSON(sampler_state_filter_obj, "mag"))
} {
else sampler->tm0.mag_filter = *mode;
{ }
ERROR_LOG_FMT(VIDEO, else
"Asset '{}' failed to parse json, 'texture_filter' has an invalid " {
"value '{}'", ERROR_LOG_FMT(VIDEO,
asset_id, sampler_state_filter); "Asset '{}' failed to parse json, 'filter_mode[mag]' has an invalid "
return false; "value",
asset_id);
return false;
}
if (const auto mode = ReadFilterModeFromJSON(sampler_state_filter_obj, "mipmap"))
{
sampler->tm0.mipmap_filter = *mode;
}
else
{
ERROR_LOG_FMT(VIDEO,
"Asset '{}' failed to parse json, 'filter_mode[mipmap]' has an invalid "
"value",
asset_id);
return false;
}
} }
return true; return true;