CheatMenuSA/CheatMenu/Paint.cpp

316 lines
8.9 KiB
C++
Raw Normal View History

// Portion of this source is taken from MoonAdditions https://github.com/THE-FYP/MoonAdditions
// To keep the licensing simple this file would go under MIT License, GPLv3 won't apply
// Copyright (c) 2012 DK22Pac
// Copyright (c) 2017 FYP
// Copyright (c) 2021 Grinch_
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
2020-12-02 16:19:16 -05:00
#include "pch.h"
#include "Paint.h"
#include "NodeName.h"
#include "Util.h"
2020-12-02 16:19:16 -05:00
2021-02-25 17:45:41 -05:00
void Paint::RenderEvent(CVehicle* pVeh)
2020-12-02 16:19:16 -05:00
{
2021-06-18 12:49:11 -04:00
VehData& data = m_VehData.Get(pVeh);
2021-02-25 17:45:41 -05:00
// reset custom color if color id changed
if (pVeh->m_nPrimaryColor != data.primary_color
|| pVeh->m_nSecondaryColor != data.secondary_color)
2020-12-02 16:19:16 -05:00
{
2021-02-25 17:45:41 -05:00
for (auto& it : data.materialProperties)
data.resetMaterialColor(it.first);
data.primary_color = pVeh->m_nPrimaryColor;
data.secondary_color = pVeh->m_nSecondaryColor;
2021-02-24 16:54:45 -05:00
}
2021-02-25 17:45:41 -05:00
for (auto& it : data.materialProperties)
{
2021-02-25 17:45:41 -05:00
if (it.second._recolor)
{
2021-02-25 17:45:41 -05:00
it.second._originalColor = it.first->color;
it.first->color = it.second._color;
it.second._originalGeometryFlags = it.second._geometry->flags;
it.second._geometry->flags |= rpGEOMETRYMODULATEMATERIALCOLOR;
}
2021-02-25 17:45:41 -05:00
if (it.second._retexture)
{
2021-02-25 17:45:41 -05:00
auto tex = it.second._texture.lock();
if (tex)
{
2021-02-25 17:45:41 -05:00
it.second._originalTexture = it.first->texture;
it.first->texture = tex.get();
}
2021-02-25 17:45:41 -05:00
else
{
2021-02-25 17:45:41 -05:00
it.second._retexture = false;
}
}
2021-02-25 17:45:41 -05:00
}
}
2020-12-02 16:19:16 -05:00
2021-02-25 17:45:41 -05:00
void Paint::ResetAfterRenderEvent(CVehicle* pVeh)
{
2021-06-18 12:49:11 -04:00
for (auto& it : m_VehData.Get(pVeh).materialProperties)
2021-02-25 17:45:41 -05:00
{
if (it.second._recolor)
{
2021-02-25 17:45:41 -05:00
it.first->color = it.second._originalColor;
it.second._geometry->flags = it.second._originalGeometryFlags;
}
if (it.second._retexture)
{
it.first->texture = it.second._originalTexture;
}
}
}
Paint::Paint()
{
Events::processScriptsEvent += [this]
2021-02-25 17:45:41 -05:00
{
2021-06-18 12:49:11 -04:00
if (!m_bImagesLoaded)
2021-02-25 17:45:41 -05:00
{
for (auto& p : fs::recursive_directory_iterator(PLUGIN_PATH((char*)"\\CheatMenu\\vehicles\\paintjobs\\")))
{
if (p.path().extension() == ".png")
{
std::string file_name = p.path().stem().string();
2021-06-18 12:49:11 -04:00
m_Textures[file_name] = std::make_shared<RwTexture>(*(Util::LoadTextureFromPngFile(p.path())));
}
}
2021-06-18 12:49:11 -04:00
m_bImagesLoaded = true;
}
};
2021-02-25 17:45:41 -05:00
Events::vehicleRenderEvent.before += RenderEvent;
Events::vehicleResetAfterRender += ResetAfterRenderEvent;
}
Paint::~Paint()
{
Events::vehicleRenderEvent.before -= RenderEvent;
Events::vehicleResetAfterRender -= ResetAfterRenderEvent;
2020-12-02 16:19:16 -05:00
}
void Paint::VehData::setMaterialColor(RpMaterial* material, RpGeometry* geometry, RwRGBA color, bool filter_mat)
{
auto& matProps = materialProperties[material];
2021-02-24 16:54:45 -05:00
if (!filter_mat
|| (material->color.red == 0x3C && material->color.green == 0xFF && material->color.blue == 0x00)
|| (material->color.red == 0xFF && material->color.green == 0x00 && material->color.blue == 0xAF))
{
matProps._recolor = true;
matProps._color = color;
matProps._geometry = geometry;
}
}
void Paint::VehData::setMaterialTexture(RpMaterial* material, std::shared_ptr<RwTexture> texture, bool filter_mat)
{
auto& matProps = materialProperties[material];
if (!filter_mat
|| (material->color.red == 0x3C && material->color.green == 0xFF && material->color.blue == 0x00)
|| (material->color.red == 0xFF && material->color.green == 0x00 && material->color.blue == 0xAF))
{
matProps._retexture = true;
matProps._texture = texture;
}
}
void Paint::VehData::resetMaterialColor(RpMaterial* material)
{
auto& matProps = materialProperties[material];
matProps._recolor = false;
2021-06-18 12:49:11 -04:00
matProps._color = {0, 0, 0, 0};
}
void Paint::VehData::resetMaterialTexture(RpMaterial* material)
{
auto& matProps = materialProperties[material];
matProps._retexture = false;
matProps._texture.reset();
}
2021-02-24 16:54:45 -05:00
void Paint::NodeWrapperRecursive(RwFrame* frame, CVehicle* pVeh, std::function<void(RwFrame*)> func)
2020-12-02 16:19:16 -05:00
{
if (frame)
{
func(frame);
2021-02-24 16:54:45 -05:00
if (RwFrame* newFrame = frame->child)
2020-12-02 16:19:16 -05:00
NodeWrapperRecursive(newFrame, pVeh, func);
2021-02-24 16:54:45 -05:00
if (RwFrame* newFrame = frame->next)
2020-12-02 16:19:16 -05:00
NodeWrapperRecursive(newFrame, pVeh, func);
}
return;
}
void Paint::UpdateNodeListRecursive(CVehicle* pVeh)
{
2021-02-24 16:54:45 -05:00
RwFrame* frame = (RwFrame*)pVeh->m_pRwClump->object.parent;
2020-12-02 16:19:16 -05:00
2021-02-24 16:54:45 -05:00
NodeWrapperRecursive(frame, pVeh, [](RwFrame* frame)
2020-12-02 16:19:16 -05:00
{
const std::string name = GetFrameNodeName(frame);
if (!(std::find(veh_nodes::names_vec.begin(), veh_nodes::names_vec.end(), name) != veh_nodes::names_vec.end()))
veh_nodes::names_vec.push_back(name);
});
}
void Paint::SetNodeColor(CVehicle* pVeh, std::string node_name, CRGBA color, bool filter_mat)
{
2021-02-24 16:54:45 -05:00
RwFrame* frame = (RwFrame*)pVeh->m_pRwClump->object.parent;
2020-12-02 16:19:16 -05:00
2021-02-24 16:54:45 -05:00
NodeWrapperRecursive(frame, pVeh, [&](RwFrame* frame)
2020-12-02 16:19:16 -05:00
{
const std::string name = GetFrameNodeName(frame);
struct ST
{
CRGBA _color;
bool _filter;
} st;
st._color = color;
st._filter = filter_mat;
if (node_name == "Default" || node_name == name)
{
2021-06-18 12:49:11 -04:00
RwFrameForAllObjects(frame, [](RwObject* object, void* data) -> RwObject*
{
2020-12-02 16:19:16 -05:00
if (object->type == rpATOMIC)
{
2021-02-24 16:54:45 -05:00
RpAtomic* atomic = reinterpret_cast<RpAtomic*>(object);
2020-12-02 16:19:16 -05:00
ST* st = reinterpret_cast<ST*>(data);
CRGBA* color = &st->_color;
bool filter_mat = st->_filter;
2021-06-18 12:49:11 -04:00
VehData& data = m_VehData.Get(FindPlayerPed()->m_pVehicle);
2020-12-02 16:19:16 -05:00
for (int i = 0; i < atomic->geometry->matList.numMaterials; ++i)
2021-06-18 12:49:11 -04:00
data.setMaterialColor(atomic->geometry->matList.materials[i], atomic->geometry,
{color->r, color->g, color->b, 255}, filter_mat);
2020-12-02 16:19:16 -05:00
}
return object;
}, &st);
}
});
}
void Paint::SetNodeTexture(CVehicle* pVeh, std::string node_name, std::string texturename, bool filter_mat)
{
2021-02-24 16:54:45 -05:00
RwFrame* frame = (RwFrame*)pVeh->m_pRwClump->object.parent;
NodeWrapperRecursive(frame, pVeh, [&](RwFrame* frame)
2020-12-02 16:19:16 -05:00
{
const std::string name = GetFrameNodeName(frame);
struct ST
{
std::string _texturename;
std::map<std::string, std::shared_ptr<RwTexture>> _textures;
bool _filter;
} st;
2021-06-18 12:49:11 -04:00
st._textures = m_Textures;
2020-12-02 16:19:16 -05:00
st._texturename = texturename;
st._filter = filter_mat;
if (node_name == "Default" || node_name == name)
{
2021-06-18 12:49:11 -04:00
RwFrameForAllObjects(frame, [](RwObject* object, void* data) -> RwObject*
{
2020-12-02 16:19:16 -05:00
if (object->type == rpATOMIC)
{
2021-02-24 16:54:45 -05:00
RpAtomic* atomic = reinterpret_cast<RpAtomic*>(object);
2020-12-02 16:19:16 -05:00
ST* st = reinterpret_cast<ST*>(data);
2021-06-18 12:49:11 -04:00
VehData& data = m_VehData.Get(FindPlayerPed()->m_pVehicle);
2020-12-02 16:19:16 -05:00
for (int i = 0; i < atomic->geometry->matList.numMaterials; ++i)
2021-06-18 12:49:11 -04:00
data.setMaterialTexture(atomic->geometry->matList.materials[i], st->_textures[st->_texturename],
st->_filter);
2020-12-02 16:19:16 -05:00
}
return object;
}, &st);
}
});
}
2021-02-24 16:54:45 -05:00
void Paint::ResetNodeColor(CVehicle* pVeh, std::string node_name)
2020-12-02 16:19:16 -05:00
{
2021-02-24 16:54:45 -05:00
RwFrame* frame = (RwFrame*)pVeh->m_pRwClump->object.parent;
2020-12-02 16:19:16 -05:00
2021-02-24 16:54:45 -05:00
NodeWrapperRecursive(frame, pVeh, [&](RwFrame* frame)
2020-12-02 16:19:16 -05:00
{
const std::string name = GetFrameNodeName(frame);
if (node_name == "Default" || node_name == name)
{
2021-06-18 12:49:11 -04:00
RwFrameForAllObjects(frame, [](RwObject* object, void* data) -> RwObject*
{
2020-12-02 16:19:16 -05:00
if (object->type == rpATOMIC)
{
2021-02-24 16:54:45 -05:00
RpAtomic* atomic = reinterpret_cast<RpAtomic*>(object);
2021-06-18 12:49:11 -04:00
VehData& data = m_VehData.Get(FindPlayerPed()->m_pVehicle);
2020-12-02 16:19:16 -05:00
for (int i = 0; i < atomic->geometry->matList.numMaterials; ++i)
data.resetMaterialColor(atomic->geometry->matList.materials[i]);
2020-12-02 16:19:16 -05:00
}
return object;
}, nullptr);
}
});
}
2021-02-24 16:54:45 -05:00
void Paint::ResetNodeTexture(CVehicle* pVeh, std::string node_name)
2020-12-02 16:19:16 -05:00
{
2021-02-24 16:54:45 -05:00
RwFrame* frame = (RwFrame*)pVeh->m_pRwClump->object.parent;
2020-12-02 16:19:16 -05:00
2021-02-24 16:54:45 -05:00
NodeWrapperRecursive(frame, pVeh, [&](RwFrame* frame)
2020-12-02 16:19:16 -05:00
{
const std::string name = GetFrameNodeName(frame);
if (node_name == "Default" || node_name == name)
{
2021-06-18 12:49:11 -04:00
RwFrameForAllObjects(frame, [](RwObject* object, void* data) -> RwObject*
{
2020-12-02 16:19:16 -05:00
if (object->type == rpATOMIC)
{
2021-02-24 16:54:45 -05:00
RpAtomic* atomic = reinterpret_cast<RpAtomic*>(object);
2020-12-02 16:19:16 -05:00
2021-06-18 12:49:11 -04:00
VehData& data = m_VehData.Get(FindPlayerPed()->m_pVehicle);
2020-12-02 16:19:16 -05:00
for (int i = 0; i < atomic->geometry->matList.numMaterials; ++i)
data.resetMaterialTexture(atomic->geometry->matList.materials[i]);
2020-12-02 16:19:16 -05:00
}
return object;
}, nullptr);
}
});
2021-06-18 12:49:11 -04:00
}