2024-11-01 14:10:00 +01:00
|
|
|
#ifndef _UTILS_
|
|
|
|
#define _UTILS_
|
2024-11-01 15:04:43 +01:00
|
|
|
#include <curl/include/curl/curl.h>
|
2024-11-02 17:17:12 +01:00
|
|
|
#include <wx/mstream.h>
|
|
|
|
#include <wx/wx.h>
|
2024-11-01 15:04:43 +01:00
|
|
|
|
2024-11-01 14:10:00 +01:00
|
|
|
#include <filesystem>
|
|
|
|
#include <fstream>
|
|
|
|
#include <nlohmann-json/single_include/nlohmann/json.hpp>
|
|
|
|
#include <sstream>
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
#define DEFAULT_CLIENT_ID "1301849203378622545"
|
|
|
|
#define DEFAULT_APP_NAME "Music"
|
|
|
|
#define CONFIG_FILENAME "known.json"
|
|
|
|
|
|
|
|
namespace utils {
|
2024-11-02 17:17:12 +01:00
|
|
|
inline wxIcon loadIconFromMemory(const unsigned char* data, size_t size) {
|
|
|
|
wxMemoryInputStream stream(data, size);
|
|
|
|
wxImage img(stream, wxBITMAP_TYPE_PNG);
|
|
|
|
if (img.IsOk()) {
|
|
|
|
wxBitmap bmp(img);
|
|
|
|
wxIcon icon;
|
|
|
|
icon.CopyFromBitmap(bmp);
|
|
|
|
return icon;
|
|
|
|
}
|
|
|
|
return wxNullIcon;
|
|
|
|
}
|
2024-11-01 16:49:48 +01:00
|
|
|
inline std::string ltrim(std::string& s) {
|
|
|
|
s.erase(s.begin(), std::find_if(s.begin(), s.end(), [](unsigned char ch) { return !std::isspace(ch); }));
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline std::string rtrim(std::string& s) {
|
|
|
|
s.erase(std::find_if(s.rbegin(), s.rend(), [](unsigned char ch) { return !std::isspace(ch); }).base(), s.end());
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
inline std::string trim(std::string& s) {
|
|
|
|
ltrim(s);
|
|
|
|
rtrim(s);
|
|
|
|
return s;
|
|
|
|
}
|
2024-11-01 15:04:43 +01:00
|
|
|
inline std::string urlEncode(std::string str) {
|
|
|
|
std::string new_str = "";
|
|
|
|
char c;
|
|
|
|
int ic;
|
|
|
|
const char* chars = str.c_str();
|
|
|
|
char bufHex[10];
|
|
|
|
int len = strlen(chars);
|
|
|
|
|
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
c = chars[i];
|
|
|
|
ic = c;
|
|
|
|
if (c == ' ')
|
|
|
|
new_str += '+';
|
|
|
|
else if (isalnum(c) || c == '-' || c == '_' || c == '.' || c == '~')
|
|
|
|
new_str += c;
|
|
|
|
else {
|
2024-11-02 12:26:37 +01:00
|
|
|
snprintf(bufHex, sizeof(bufHex), "%X", c);
|
2024-11-01 15:04:43 +01:00
|
|
|
if (ic < 16)
|
|
|
|
new_str += "%0";
|
|
|
|
else
|
|
|
|
new_str += "%";
|
|
|
|
new_str += bufHex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return new_str;
|
|
|
|
}
|
2024-11-01 16:49:48 +01:00
|
|
|
|
2024-11-01 15:04:43 +01:00
|
|
|
inline size_t curlWriteCallback(char* contents, size_t size, size_t nmemb, void* userp) {
|
|
|
|
((std::string*)userp)->append((char*)contents, size * nmemb);
|
|
|
|
return size * nmemb;
|
|
|
|
}
|
2024-11-01 16:49:48 +01:00
|
|
|
|
2024-11-01 15:04:43 +01:00
|
|
|
inline std::string getRequest(std::string url) {
|
|
|
|
CURL* curl;
|
|
|
|
CURLcode res;
|
|
|
|
std::string buf;
|
|
|
|
curl_global_init(CURL_GLOBAL_ALL);
|
|
|
|
curl = curl_easy_init();
|
|
|
|
if (curl) {
|
|
|
|
curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
|
|
|
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, curlWriteCallback);
|
2024-11-01 16:49:48 +01:00
|
|
|
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);
|
2024-11-01 15:04:43 +01:00
|
|
|
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &buf);
|
|
|
|
res = curl_easy_perform(curl);
|
|
|
|
curl_easy_cleanup(curl);
|
|
|
|
}
|
|
|
|
curl_global_cleanup();
|
|
|
|
return buf;
|
|
|
|
}
|
2024-11-01 16:49:48 +01:00
|
|
|
|
2024-11-01 15:04:43 +01:00
|
|
|
inline std::string getArtworkURL(std::string query) {
|
|
|
|
std::string response =
|
|
|
|
getRequest("https://itunes.apple.com/search?media=music&entity=song&term=" + urlEncode(query));
|
|
|
|
nlohmann::json j = nlohmann::json::parse(response);
|
|
|
|
auto results = j["results"];
|
|
|
|
if (results.size() > 0) {
|
|
|
|
return results[0]["artworkUrl100"].get<std::string>();
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
2024-11-01 16:49:48 +01:00
|
|
|
|
2024-11-01 14:10:00 +01:00
|
|
|
inline nlohmann::json getApp(std::string processName) {
|
|
|
|
std::ifstream i("known.json");
|
|
|
|
std::stringstream s;
|
|
|
|
s << i.rdbuf();
|
|
|
|
i.close();
|
|
|
|
|
|
|
|
try {
|
|
|
|
nlohmann::json j = nlohmann::json::parse(s.str());
|
|
|
|
auto apps = j["apps"];
|
|
|
|
for (auto app : apps) {
|
|
|
|
auto processNames = app["process_names"];
|
|
|
|
for (auto process : processNames) {
|
|
|
|
if (process.get<std::string>() == processName)
|
|
|
|
return app;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (nlohmann::json::parse_error& ex) {
|
|
|
|
} // TODO: handle parse errors
|
|
|
|
return nlohmann::json();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline std::string getClientID(std::string processName) {
|
|
|
|
if (!std::filesystem::exists(CONFIG_FILENAME))
|
|
|
|
return DEFAULT_CLIENT_ID;
|
|
|
|
auto app = getApp(processName);
|
2024-11-01 16:49:48 +01:00
|
|
|
return app.contains("client_id") ? app["client_id"].get<std::string>() : DEFAULT_CLIENT_ID;
|
2024-11-01 14:10:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
inline std::string getAppName(std::string processName) {
|
|
|
|
if (!std::filesystem::exists(CONFIG_FILENAME))
|
|
|
|
return DEFAULT_APP_NAME;
|
|
|
|
auto app = getApp(processName);
|
2024-11-01 16:49:48 +01:00
|
|
|
return app.contains("name") ? app["name"].get<std::string>() : DEFAULT_APP_NAME;
|
2024-11-01 14:10:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
inline std::string getSearchEndpoint(std::string processName) {
|
|
|
|
if (!std::filesystem::exists(CONFIG_FILENAME))
|
|
|
|
return "";
|
|
|
|
auto app = getApp(processName);
|
2024-11-01 16:49:48 +01:00
|
|
|
return app.contains("search_endpoint") ? app["search_endpoint"].get<std::string>() : "";
|
2024-11-01 14:10:00 +01:00
|
|
|
}
|
|
|
|
} // namespace utils
|
|
|
|
|
|
|
|
#undef DEFAULT_APP_NAME
|
|
|
|
#undef CONFIG_FILENAME
|
|
|
|
#undef DEFAULT_CLIENT_ID
|
|
|
|
#endif
|