mirror of
https://github.com/HamletDuFromage/aio-switch-updater.git
synced 2025-01-19 12:51:32 +00:00
356 lines
No EOL
11 KiB
C++
356 lines
No EOL
11 KiB
C++
#include "color_swapper.hpp"
|
|
#include <iomanip>
|
|
#include <string>
|
|
#include <fstream>
|
|
#include <sstream>
|
|
#include <filesystem>
|
|
#include "constants.hpp"
|
|
#include "progress_event.hpp"
|
|
#include "utils.hpp"
|
|
#include "fs.hpp"
|
|
|
|
using json = nlohmann::json;
|
|
|
|
namespace ColorSwapper {
|
|
|
|
int hexToBGR(const std::string& hex){
|
|
std::string R = hex.substr(0, 2);
|
|
std::string G = hex.substr(2, 2);
|
|
std::string B = hex.substr(4, 2);
|
|
return std::stoi(B + G + R, 0, 16);
|
|
}
|
|
|
|
std::string BGRToHex(int v){
|
|
std::stringstream ss;
|
|
v = ((v & 0xFF) << 16) + (v & 0xFF00) + (v >> 16) + 256;
|
|
ss << std::setfill('0') << std::setw(6) << std::right << std::hex << v;
|
|
return ss.str();
|
|
}
|
|
|
|
bool isHexaAnd3Bytes(const std::string& str){
|
|
if(str.size()!=6) return false;
|
|
for(char const &c : str){
|
|
if(!isxdigit(c)) return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
}
|
|
|
|
namespace JC {
|
|
|
|
int setColor(const std::vector<int>& colors){
|
|
Result pads, ljc, rjc;
|
|
int res = 0;
|
|
s32 nbEntries;
|
|
HidsysUniquePadId UniquePadIds[2] = {};
|
|
|
|
pads = hidsysGetUniquePadsFromNpad(HidNpadIdType_Handheld, UniquePadIds, 2,&nbEntries);
|
|
if(R_SUCCEEDED(pads)){
|
|
ljc = hiddbgUpdateControllerColor(colors[0], colors[1], UniquePadIds[0]);
|
|
if (R_FAILED(ljc)) res +=1;
|
|
rjc = hiddbgUpdateControllerColor(colors[2], colors[3], UniquePadIds[1]);
|
|
if (R_FAILED(rjc)) res +=2;
|
|
}
|
|
else{
|
|
res +=4;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
int backupToJSON(json &profiles, const std::string& path){
|
|
HidNpadControllerColor color_left;
|
|
HidNpadControllerColor color_right;
|
|
Result res = hidGetNpadControllerColorSplit(HidNpadIdType_Handheld, &color_left, &color_right);
|
|
std::vector<int> oldBackups;
|
|
if (R_SUCCEEDED(res)) {
|
|
int i = 0;
|
|
for (const auto& x : profiles.items()){
|
|
if(x.value()["name"] == "_backup") {
|
|
oldBackups.push_back(i);
|
|
}
|
|
i++;
|
|
}
|
|
for (auto &k : oldBackups){
|
|
profiles.erase(profiles.begin() + k);
|
|
}
|
|
json newBackup = json::object({
|
|
{"name", "_backup"},
|
|
{"L_JC", ColorSwapper::BGRToHex(color_left.main)},
|
|
{"L_BTN", ColorSwapper::BGRToHex(color_left.sub)},
|
|
{"R_JC", ColorSwapper::BGRToHex(color_right.main)},
|
|
{"R_BTN", ColorSwapper::BGRToHex(color_right.sub)}
|
|
});
|
|
profiles.push_back(newBackup);
|
|
fs::writeJsonToFile(profiles, path);
|
|
return 0;
|
|
}
|
|
else{
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
json backupProfile(){
|
|
json newBackup;
|
|
HidNpadControllerColor color_left;
|
|
HidNpadControllerColor color_right;
|
|
Result res = hidGetNpadControllerColorSplit(HidNpadIdType_Handheld, &color_left, &color_right);
|
|
if (R_SUCCEEDED(res)) {
|
|
newBackup = json::object({
|
|
{"name", "_backup"},
|
|
{"L_JC", ColorSwapper::BGRToHex(color_left.main)},
|
|
{"L_BTN", ColorSwapper::BGRToHex(color_left.sub)},
|
|
{"R_JC", ColorSwapper::BGRToHex(color_right.main)},
|
|
{"R_BTN", ColorSwapper::BGRToHex(color_right.sub)}
|
|
});
|
|
}
|
|
return newBackup;
|
|
|
|
}
|
|
|
|
std::vector<std::pair<std::string, std::vector<int>>> getProfiles(const std::string& path){
|
|
std::vector<std::pair<std::string, std::vector<int>>> res;
|
|
bool properData;
|
|
std::fstream profilesFile;
|
|
json profilesJson = fs::parseJsonFile(path);
|
|
if(profilesJson.empty()){
|
|
profilesJson = {{
|
|
{"L_BTN", "0A1E0A"},
|
|
{"L_JC", "82FF96"},
|
|
{"R_BTN", "0A1E28"},
|
|
{"R_JC", "96F5F5"},
|
|
{"name", "Animal Crossing: New Horizons"}
|
|
}};
|
|
}
|
|
for (const auto& x : profilesJson.items()){
|
|
std::string name = x.value()["name"];
|
|
std::vector<std::string> values = {
|
|
std::string(x.value()["L_JC"]),
|
|
std::string(x.value()["L_BTN"]),
|
|
std::string(x.value()["R_JC"]),
|
|
std::string(x.value()["R_BTN"])
|
|
};
|
|
properData = true;
|
|
for(auto& str : values){
|
|
if(!ColorSwapper::isHexaAnd3Bytes(str)){
|
|
properData = false;
|
|
}
|
|
}
|
|
if(properData){
|
|
if(name == "") name = "Unamed";
|
|
res.push_back(std::make_pair(name, (std::vector<int>){
|
|
ColorSwapper::hexToBGR(values[0]),
|
|
ColorSwapper::hexToBGR(values[1]),
|
|
ColorSwapper::hexToBGR(values[2]),
|
|
ColorSwapper::hexToBGR(values[3])
|
|
}));
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
void changeJCColor(const std::vector<int>& values){
|
|
hiddbgInitialize();
|
|
hidsysInitialize();
|
|
ProgressEvent::instance().reset();
|
|
ProgressEvent::instance().setStep(1);
|
|
int res = setColor(values);
|
|
if(res != 0){
|
|
util::showDialogBox("Could not change the Joy-Cons color. Make sure they're docked and try again.\nError :" + std::to_string(res), "Ok");
|
|
}
|
|
hiddbgExit();
|
|
hidsysExit();
|
|
ProgressEvent::instance().setStep(ProgressEvent::instance().getMax());
|
|
}
|
|
|
|
void backupJCColor(const std::string& path){
|
|
hiddbgInitialize();
|
|
hidsysInitialize();
|
|
ProgressEvent::instance().reset();
|
|
ProgressEvent::instance().setStep(1);
|
|
json backup;
|
|
json profiles;
|
|
std::fstream profilesFile;
|
|
if(std::filesystem::exists(path)){
|
|
profilesFile.open(path, std::fstream::in);
|
|
profilesFile >> profiles;
|
|
profilesFile.close();
|
|
}
|
|
|
|
std::vector<int> oldBackups;
|
|
int i = 0;
|
|
for (const auto& x : profiles.items()){
|
|
if(x.value()["name"] == "_backup") {
|
|
oldBackups.push_back(i);
|
|
}
|
|
i++;
|
|
}
|
|
for (auto &k : oldBackups){
|
|
profiles.erase(profiles.begin() + k);
|
|
}
|
|
|
|
while(backup.empty()){
|
|
backup = backupProfile();
|
|
}
|
|
|
|
profiles.push_back(backup);
|
|
//backup.push_back(profiles);
|
|
fs::writeJsonToFile(profiles, path);
|
|
hiddbgExit();
|
|
hidsysExit();
|
|
ProgressEvent::instance().setStep(ProgressEvent::instance().getMax());
|
|
}
|
|
|
|
}
|
|
|
|
namespace PC {
|
|
|
|
int setColor(const std::vector<int>& colors){
|
|
Result pads, pc;
|
|
int res = 0;
|
|
s32 nbEntries;
|
|
HidsysUniquePadId UniquePadIds[1] = {};
|
|
|
|
pads = hidsysGetUniquePadsFromNpad(HidNpadIdType_No1, UniquePadIds, 1 ,&nbEntries);
|
|
if(R_SUCCEEDED(pads)){
|
|
pc = hiddbgUpdateControllerColor(colors[0], colors[1], UniquePadIds[0]);
|
|
if (R_FAILED(pc)) res +=1;
|
|
}
|
|
else{
|
|
res +=4;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
int backupToJSON(json &profiles, const std::string& path){
|
|
HidNpadControllerColor color;
|
|
Result res = hidGetNpadControllerColorSingle(HidNpadIdType_No1, &color);
|
|
std::vector<int> oldBackups;
|
|
if (R_SUCCEEDED(res)) {
|
|
int i = 0;
|
|
for (const auto& x : profiles.items()){
|
|
if(x.value()["name"] == "_backup") {
|
|
oldBackups.push_back(i);
|
|
}
|
|
i++;
|
|
}
|
|
for (auto &k : oldBackups){
|
|
profiles.erase(profiles.begin() + k);
|
|
}
|
|
json newBackup = json::object({
|
|
{"name", "_backup"},
|
|
{"BODY", ColorSwapper::BGRToHex(color.main)},
|
|
{"BTN", ColorSwapper::BGRToHex(color.sub)}
|
|
});
|
|
profiles.push_back(newBackup);
|
|
fs::writeJsonToFile(profiles, path);
|
|
return 0;
|
|
}
|
|
else{
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
json backupProfile(){
|
|
json newBackup;
|
|
HidNpadControllerColor color;
|
|
Result res = hidGetNpadControllerColorSingle(HidNpadIdType_No1, &color);
|
|
if (R_SUCCEEDED(res)) {
|
|
newBackup = json::object({
|
|
{"name", "_backup"},
|
|
{"BODY", ColorSwapper::BGRToHex(color.main)},
|
|
{"BTN", ColorSwapper::BGRToHex(color.sub)}
|
|
});
|
|
}
|
|
return newBackup;
|
|
|
|
}
|
|
|
|
std::vector<std::pair<std::string, std::vector<int>>> getProfiles(const std::string& path){
|
|
std::vector<std::pair<std::string, std::vector<int>>> res;
|
|
bool properData;
|
|
std::fstream profilesFile;
|
|
json profilesJson = fs::parseJsonFile(path);
|
|
if(profilesJson.empty()){
|
|
profilesJson = {{
|
|
{"BTN", "2d2d2d"},
|
|
{"BODY", "e6e6e6"},
|
|
{"name", "Default black"}
|
|
}};
|
|
}
|
|
for (const auto& x : profilesJson.items()){
|
|
std::string name = x.value()["name"];
|
|
std::vector<std::string> values = {
|
|
std::string(x.value()["BODY"]),
|
|
std::string(x.value()["BTN"])
|
|
};
|
|
properData = true;
|
|
for(auto& str : values){
|
|
if(!ColorSwapper::isHexaAnd3Bytes(str)){
|
|
properData = false;
|
|
}
|
|
}
|
|
if(properData){
|
|
if(name == "") name = "Unamed";
|
|
res.push_back(std::make_pair(name, (std::vector<int>){
|
|
ColorSwapper::hexToBGR(values[0]),
|
|
ColorSwapper::hexToBGR(values[1])
|
|
}));
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
void changePCColor(const std::vector<int>& values){
|
|
hiddbgInitialize();
|
|
hidsysInitialize();
|
|
ProgressEvent::instance().reset();
|
|
ProgressEvent::instance().setStep(1);
|
|
int res = setColor(values);
|
|
if(res != 0){
|
|
util::showDialogBox("Could not change the Pro-Con color. Make they're connected to P1. This feature may not work on unofficial controllers. \nError :" + std::to_string(res), "Ok");
|
|
}
|
|
hiddbgExit();
|
|
hidsysExit();
|
|
ProgressEvent::instance().setStep(ProgressEvent::instance().getMax());
|
|
}
|
|
|
|
void backupPCColor(const std::string& path){
|
|
hiddbgInitialize();
|
|
hidsysInitialize();
|
|
ProgressEvent::instance().reset();
|
|
ProgressEvent::instance().setStep(1);
|
|
json backup;
|
|
json profiles;
|
|
std::fstream profilesFile;
|
|
if(std::filesystem::exists(path)){
|
|
profilesFile.open(path, std::fstream::in);
|
|
profilesFile >> profiles;
|
|
profilesFile.close();
|
|
}
|
|
|
|
std::vector<int> oldBackups;
|
|
int i = 0;
|
|
for (const auto& x : profiles.items()){
|
|
if(x.value()["name"] == "_backup") {
|
|
oldBackups.push_back(i);
|
|
}
|
|
i++;
|
|
}
|
|
for (auto &k : oldBackups){
|
|
profiles.erase(profiles.begin() + k);
|
|
}
|
|
|
|
while(backup.empty()){
|
|
backup = backupProfile();
|
|
}
|
|
|
|
profiles.push_back(backup);
|
|
//backup.push_back(profiles);
|
|
fs::writeJsonToFile(profiles, path);
|
|
hiddbgExit();
|
|
hidsysExit();
|
|
ProgressEvent::instance().setStep(ProgressEvent::instance().getMax());
|
|
}
|
|
|
|
} |