2024-10-28 15:57:39 +01:00
|
|
|
// Luanti
|
|
|
|
// SPDX-License-Identifier: LGPL-2.1-or-later
|
|
|
|
// Copyright (C) 2020 Hugues Ross <hugues.ross@gmail.com>
|
2020-04-14 20:41:29 +02:00
|
|
|
|
|
|
|
#include "texture_override.h"
|
|
|
|
|
|
|
|
#include "log.h"
|
2024-05-08 20:37:10 +02:00
|
|
|
#include "filesys.h"
|
2020-04-14 20:41:29 +02:00
|
|
|
#include "util/string.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <fstream>
|
2024-10-09 16:24:44 +02:00
|
|
|
#include <map>
|
2020-04-14 20:41:29 +02:00
|
|
|
|
2020-08-04 20:12:47 +02:00
|
|
|
#define override_cast static_cast<override_t>
|
|
|
|
|
2022-08-13 08:52:16 +02:00
|
|
|
static const std::map<std::string, OverrideTarget> override_LUT = {
|
|
|
|
{ "top", OverrideTarget::TOP },
|
|
|
|
{ "bottom", OverrideTarget::BOTTOM },
|
|
|
|
{ "left", OverrideTarget::LEFT },
|
|
|
|
{ "right", OverrideTarget::RIGHT },
|
|
|
|
{ "front", OverrideTarget::FRONT },
|
|
|
|
{ "back", OverrideTarget::BACK },
|
|
|
|
{ "inventory", OverrideTarget::INVENTORY },
|
|
|
|
{ "wield", OverrideTarget::WIELD },
|
|
|
|
{ "special1", OverrideTarget::SPECIAL_1 },
|
|
|
|
{ "special2", OverrideTarget::SPECIAL_2 },
|
|
|
|
{ "special3", OverrideTarget::SPECIAL_3 },
|
|
|
|
{ "special4", OverrideTarget::SPECIAL_4 },
|
|
|
|
{ "special5", OverrideTarget::SPECIAL_5 },
|
|
|
|
{ "special6", OverrideTarget::SPECIAL_6 },
|
|
|
|
{ "sides", OverrideTarget::SIDES },
|
|
|
|
{ "all", OverrideTarget::ALL_FACES },
|
|
|
|
{ "*", OverrideTarget::ALL_FACES }
|
|
|
|
};
|
|
|
|
|
2023-12-23 13:02:17 +01:00
|
|
|
TextureOverrideSource::TextureOverrideSource(const std::string &filepath)
|
2020-04-14 20:41:29 +02:00
|
|
|
{
|
2024-05-08 20:37:10 +02:00
|
|
|
auto infile = open_ifstream(filepath.c_str(), false);
|
2020-04-14 20:41:29 +02:00
|
|
|
std::string line;
|
|
|
|
int line_index = 0;
|
|
|
|
while (std::getline(infile, line)) {
|
|
|
|
line_index++;
|
|
|
|
|
|
|
|
// Also trim '\r' on DOS-style files
|
|
|
|
line = trim(line);
|
|
|
|
|
|
|
|
// Ignore empty lines and comments
|
|
|
|
if (line.empty() || line[0] == '#')
|
|
|
|
continue;
|
|
|
|
|
2022-08-13 08:52:16 +02:00
|
|
|
// Format: mod_name:item_name target1[,...] texture_name.png
|
2020-04-14 20:41:29 +02:00
|
|
|
std::vector<std::string> splitted = str_split(line, ' ');
|
2022-08-13 08:52:16 +02:00
|
|
|
if (splitted.size() < 3) {
|
2020-04-14 20:41:29 +02:00
|
|
|
warningstream << filepath << ":" << line_index
|
|
|
|
<< " Syntax error in texture override \"" << line
|
|
|
|
<< "\": Expected 3 arguments, got " << splitted.size()
|
|
|
|
<< std::endl;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
TextureOverride texture_override = {};
|
|
|
|
texture_override.id = splitted[0];
|
|
|
|
texture_override.texture = splitted[2];
|
|
|
|
|
|
|
|
// Parse the target mask
|
|
|
|
std::vector<std::string> targets = str_split(splitted[1], ',');
|
|
|
|
for (const std::string &target : targets) {
|
2022-08-13 08:52:16 +02:00
|
|
|
std::vector<std::string> kvpair = str_split(target, '=');
|
|
|
|
if (kvpair.size() == 2) {
|
|
|
|
// Key-value pairs
|
|
|
|
if (kvpair[0] == "align_world") {
|
|
|
|
// Global textures
|
|
|
|
texture_override.world_scale = stoi(kvpair[1], 0, U8_MAX);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (kvpair.size() == 1) {
|
|
|
|
// Regular override flags
|
|
|
|
auto pair = override_LUT.find(target);
|
|
|
|
|
|
|
|
if (pair != override_LUT.end()) {
|
|
|
|
texture_override.target |= override_cast(pair->second);
|
|
|
|
continue;
|
|
|
|
}
|
2020-04-14 20:41:29 +02:00
|
|
|
}
|
2022-08-13 08:52:16 +02:00
|
|
|
|
|
|
|
// Report invalid target
|
|
|
|
warningstream << filepath << ":" << line_index
|
|
|
|
<< " Syntax error in texture override \"" << line
|
|
|
|
<< "\": Unknown target \"" << target << "\""
|
|
|
|
<< std::endl;
|
2023-05-18 20:34:18 +02:00
|
|
|
|
2020-04-14 20:41:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// If there are no valid targets, skip adding this override
|
2020-08-04 20:12:47 +02:00
|
|
|
if (texture_override.target == override_cast(OverrideTarget::INVALID)) {
|
2020-04-14 20:41:29 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_overrides.push_back(texture_override);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Get all overrides that apply to item definitions
|
2023-12-23 13:02:17 +01:00
|
|
|
std::vector<TextureOverride> TextureOverrideSource::getItemTextureOverrides() const
|
2020-04-14 20:41:29 +02:00
|
|
|
{
|
|
|
|
std::vector<TextureOverride> found_overrides;
|
|
|
|
|
|
|
|
for (const TextureOverride &texture_override : m_overrides) {
|
|
|
|
if (texture_override.hasTarget(OverrideTarget::ITEM_TARGETS))
|
|
|
|
found_overrides.push_back(texture_override);
|
|
|
|
}
|
|
|
|
|
|
|
|
return found_overrides;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Get all overrides that apply to node definitions
|
2023-12-23 13:02:17 +01:00
|
|
|
std::vector<TextureOverride> TextureOverrideSource::getNodeTileOverrides() const
|
2020-04-14 20:41:29 +02:00
|
|
|
{
|
|
|
|
std::vector<TextureOverride> found_overrides;
|
|
|
|
|
|
|
|
for (const TextureOverride &texture_override : m_overrides) {
|
2020-08-04 20:12:47 +02:00
|
|
|
if (texture_override.hasTarget(OverrideTarget::NODE_TARGETS))
|
2020-04-14 20:41:29 +02:00
|
|
|
found_overrides.push_back(texture_override);
|
|
|
|
}
|
|
|
|
|
|
|
|
return found_overrides;
|
|
|
|
}
|