Commit 5884e5a2 authored by nanahira's avatar nanahira

merge new subdir

parents 8de09587 a6f86739
...@@ -71,6 +71,7 @@ inline int myswprintf(wchar_t(&buf)[N], const wchar_t* fmt, TR... args) { ...@@ -71,6 +71,7 @@ inline int myswprintf(wchar_t(&buf)[N], const wchar_t* fmt, TR... args) {
#include <memory.h> #include <memory.h>
#include <time.h> #include <time.h>
#include "bufferio.h" #include "bufferio.h"
#include "myfilesystem.h"
#include "mymutex.h" #include "mymutex.h"
#include "mysignal.h" #include "mysignal.h"
#include "mythread.h" #include "mythread.h"
......
...@@ -65,7 +65,13 @@ bool DataManager::LoadDB(const char* file) { ...@@ -65,7 +65,13 @@ bool DataManager::LoadDB(const char* file) {
return true; return true;
} }
bool DataManager::LoadStrings(const char* file) { bool DataManager::LoadStrings(const char* file) {
#ifdef _WIN32
wchar_t fname[1024];
BufferIO::DecodeUTF8(file, fname);
FILE* fp = _wfopen(fname, L"r");
#else
FILE* fp = fopen(file, "r"); FILE* fp = fopen(file, "r");
#endif // _WIN32
if(!fp) if(!fp)
return false; return false;
char linebuf[256]; char linebuf[256];
......
...@@ -13,7 +13,13 @@ DeckManager deckManager; ...@@ -13,7 +13,13 @@ DeckManager deckManager;
void DeckManager::LoadLFListSingle(const char* path) { void DeckManager::LoadLFListSingle(const char* path) {
LFList* cur = NULL; LFList* cur = NULL;
#ifdef _WIN32
wchar_t fname[1024];
BufferIO::DecodeUTF8(path, fname);
FILE* fp = _wfopen(fname, L"r");
#else
FILE* fp = fopen(path, "r"); FILE* fp = fopen(path, "r");
#endif // _WIN32
char linebuf[256]; char linebuf[256];
wchar_t strBuffer[256]; wchar_t strBuffer[256];
if(fp) { if(fp) {
...@@ -52,42 +58,15 @@ void DeckManager::LoadLFListSingle(const char* path) { ...@@ -52,42 +58,15 @@ void DeckManager::LoadLFListSingle(const char* path) {
fclose(fp); fclose(fp);
} }
} }
void DeckManager::LoadLFListDirectry(const char* path) {
char fpath[1000];
sprintf(fpath, "%s/lflist.conf", path);
LoadLFListSingle(fpath);
}
void DeckManager::LoadLFList() { void DeckManager::LoadLFList() {
LoadLFListSingle("expansions/lflist.conf"); LoadLFListSingle("expansions/lflist.conf");
#ifdef _WIN32 FileSystem::TraversalDir("./expansions", [this](const char* name, bool isdir) {
char fpath[1000]; if(isdir && strcmp(name, ".") && strcmp(name, "..")) {
WIN32_FIND_DATAW fdataw; char fpath[1024];
HANDLE fh = FindFirstFileW(L"./expansions/*", &fdataw); sprintf(fpath, "./expansions/%s/lflist.conf", name);
if(fh != INVALID_HANDLE_VALUE) { LoadLFListSingle(fpath);
do {
if(wcscmp(L".",fdataw.cFileName) != 0 && wcscmp(L"..",fdataw.cFileName) != 0 && fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
char fname[780];
BufferIO::EncodeUTF8(fdataw.cFileName, fname);
sprintf(fpath, "./expansions/%s", fname);
LoadLFListDirectry(fpath);
}
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
}
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./expansions/")) != NULL) {
while((dirp = readdir(dir)) != NULL) {
if (strcmp(".", dirp->d_name) == 0 || strcmp("..", dirp->d_name) == 0 || dirp->d_type != DT_DIR)
continue;
char filepath[1000];
sprintf(filepath, "./expansions/%s", dirp->d_name);
LoadLFListDirectry(filepath);
} }
closedir(dir); });
}
#endif
LoadLFListSingle("lflist.conf"); LoadLFListSingle("lflist.conf");
LFList nolimit; LFList nolimit;
myswprintf(nolimit.listName, L"N/A"); myswprintf(nolimit.listName, L"N/A");
...@@ -288,6 +267,8 @@ bool DeckManager::LoadDeck(const wchar_t* file) { ...@@ -288,6 +267,8 @@ bool DeckManager::LoadDeck(const wchar_t* file) {
return true; return true;
} }
bool DeckManager::SaveDeck(Deck& deck, const wchar_t* name) { bool DeckManager::SaveDeck(Deck& deck, const wchar_t* name) {
if(!FileSystem::IsDirExists(L"./deck") && !FileSystem::MakeDir(L"./deck"))
return false;
wchar_t file[64]; wchar_t file[64];
myswprintf(file, L"./deck/%ls.ydk", name); myswprintf(file, L"./deck/%ls.ydk", name);
FILE* fp = OpenDeckFile(file, "w"); FILE* fp = OpenDeckFile(file, "w");
......
...@@ -36,7 +36,6 @@ public: ...@@ -36,7 +36,6 @@ public:
std::vector<LFList> _lfList; std::vector<LFList> _lfList;
void LoadLFListSingle(const char* path); void LoadLFListSingle(const char* path);
void LoadLFListDirectry(const char* path);
void LoadLFList(); void LoadLFList();
wchar_t* GetLFListName(int lfhash); wchar_t* GetLFListName(int lfhash);
int CheckDeck(Deck& deck, int lfhash, bool allow_ocg, bool allow_tcg); int CheckDeck(Deck& deck, int lfhash, bool allow_ocg, bool allow_tcg);
......
...@@ -11,16 +11,6 @@ ...@@ -11,16 +11,6 @@
#include "single_mode.h" #include "single_mode.h"
#include <sstream> #include <sstream>
#ifndef _WIN32
#include <sys/types.h>
#include <sys/stat.h>
#include <dirent.h>
#include <unistd.h>
#else
#include <direct.h>
#include <io.h>
#endif
unsigned short PRO_VERSION = 0x1346; unsigned short PRO_VERSION = 0x1346;
namespace ygo { namespace ygo {
...@@ -29,7 +19,6 @@ Game* mainGame; ...@@ -29,7 +19,6 @@ Game* mainGame;
bool Game::Initialize() { bool Game::Initialize() {
srand(time(0)); srand(time(0));
initUtils();
LoadConfig(); LoadConfig();
irr::SIrrlichtCreationParameters params = irr::SIrrlichtCreationParameters(); irr::SIrrlichtCreationParameters params = irr::SIrrlichtCreationParameters();
params.AntiAlias = gameConf.antialias; params.AntiAlias = gameConf.antialias;
...@@ -983,142 +972,44 @@ void Game::SetStaticText(irr::gui::IGUIStaticText* pControl, u32 cWidth, irr::gu ...@@ -983,142 +972,44 @@ void Game::SetStaticText(irr::gui::IGUIStaticText* pControl, u32 cWidth, irr::gu
} }
void Game::LoadExpansionDB() { void Game::LoadExpansionDB() {
LoadExpansionDBDirectry("./expansions"); LoadExpansionDBDirectry("./expansions");
#ifdef _WIN32 FileSystem::TraversalDir("./expansions", [this](const char* name, bool isdir) {
char fpath[1000]; if(isdir && strcmp(name, ".") && strcmp(name, "..")) {
WIN32_FIND_DATAW fdataw; char subdir[1024];
HANDLE fh = FindFirstFileW(L"./expansions/*", &fdataw); sprintf(subdir, "./expansions/%s", name);
if(fh != INVALID_HANDLE_VALUE) { LoadExpansionDBDirectry(subdir);
do {
if(wcscmp(L".",fdataw.cFileName) != 0 && wcscmp(L"..",fdataw.cFileName) != 0 && fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
char fname[780];
BufferIO::EncodeUTF8(fdataw.cFileName, fname);
sprintf(fpath, "./expansions/%s", fname);
LoadExpansionDBDirectry(fpath);
}
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
}
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./expansions/")) != NULL) {
while((dirp = readdir(dir)) != NULL) {
if (strcmp(".", dirp->d_name) == 0 || strcmp("..", dirp->d_name) == 0 || dirp->d_type != DT_DIR)
continue;
char filepath[1000];
sprintf(filepath, "./expansions/%s/", dirp->d_name);
LoadExpansionDBDirectry(filepath);
} }
closedir(dir); });
}
#endif
} }
void Game::LoadExpansionDBDirectry(const char* path) { void Game::LoadExpansionDBDirectry(const char* path) {
#ifdef _WIN32 FileSystem::TraversalDir(path, [path](const char* name, bool isdir) {
char fpath[1000]; if(!isdir && !mystrncasecmp(strrchr(name, '.'), ".cdb", 4)) {
wchar_t wpath1[1000]; char fpath[1024];
wchar_t wpath2[1000]; sprintf(fpath, "%s/%s", path, name);
BufferIO::DecodeUTF8(path, wpath1); dataManager.LoadDB(fpath);
myswprintf(wpath2, L"%ls/*.cdb", wpath1);
WIN32_FIND_DATAW fdataw;
HANDLE fh = FindFirstFileW(wpath2, &fdataw);
if(fh != INVALID_HANDLE_VALUE) {
do {
if(!(fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) {
char fname[780];
BufferIO::EncodeUTF8(fdataw.cFileName, fname);
sprintf(fpath, "%s/%s", path, fname);
dataManager.LoadDB(fpath);
}
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
}
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir(path)) != NULL) {
while((dirp = readdir(dir)) != NULL) {
size_t len = strlen(dirp->d_name);
if(len < 5 || strcasecmp(dirp->d_name + len - 4, ".cdb") != 0)
continue;
char filepath[1000];
sprintf(filepath, "%s/%s", path, dirp->d_name);
dataManager.LoadDB(filepath);
} }
closedir(dir); });
}
#endif
} }
void Game::LoadExpansionStrings() { void Game::LoadExpansionStrings() {
LoadExpansionStringsDirectry("./expansions"); dataManager.LoadStrings("./expansions/strings.conf");
#ifdef _WIN32 FileSystem::TraversalDir("./expansions", [](const char* name, bool isdir) {
char fpath[1000]; if(isdir && strcmp(name, ".") && strcmp(name, "..")) {
WIN32_FIND_DATAW fdataw; char fpath[1024];
HANDLE fh = FindFirstFileW(L"./expansions/*", &fdataw); sprintf(fpath, "./expansions/%s/strings.conf", name);
if(fh != INVALID_HANDLE_VALUE) { dataManager.LoadStrings(fpath);
do {
if(wcscmp(L".",fdataw.cFileName) != 0 && wcscmp(L"..",fdataw.cFileName) != 0 && fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
char fname[780];
BufferIO::EncodeUTF8(fdataw.cFileName, fname);
sprintf(fpath, "./expansions/%s", fname);
LoadExpansionStringsDirectry(fpath);
}
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
}
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./expansions/")) != NULL) {
while((dirp = readdir(dir)) != NULL) {
if (strcmp(".", dirp->d_name) == 0 || strcmp("..", dirp->d_name) == 0 || dirp->d_type != DT_DIR)
continue;
char filepath[1000];
sprintf(filepath, "./expansions/%s/", dirp->d_name);
LoadExpansionStringsDirectry(filepath);
} }
closedir(dir); });
}
#endif
}
void Game::LoadExpansionStringsDirectry(const char* path) {
char fpath[1000];
sprintf(fpath, "%s/strings.conf", path);
dataManager.LoadStrings(fpath);
} }
void Game::RefreshDeck(irr::gui::IGUIComboBox* cbDeck) { void Game::RefreshDeck(irr::gui::IGUIComboBox* cbDeck) {
cbDeck->clear(); cbDeck->clear();
#ifdef _WIN32 FileSystem::TraversalDir(L"./deck", [cbDeck](const wchar_t* name, bool isdir) {
WIN32_FIND_DATAW fdataw; if(!isdir && !mywcsncasecmp(wcsrchr(name, '.'), L".ydk", 4)) {
HANDLE fh = FindFirstFileW(L"./deck/*.ydk", &fdataw); size_t len = wcslen(name);
if(fh == INVALID_HANDLE_VALUE) wchar_t deckname[256];
return; wcsncpy(deckname, name, len - 4);
do { deckname[len - 4] = 0;
if(!(fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) { cbDeck->addItem(deckname);
wchar_t* pf = fdataw.cFileName;
while(*pf) pf++;
while(*pf != L'.') pf--;
*pf = 0;
cbDeck->addItem(fdataw.cFileName);
} }
} while(FindNextFileW(fh, &fdataw)); });
FindClose(fh);
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./deck/")) == NULL)
return;
while((dirp = readdir(dir)) != NULL) {
size_t len = strlen(dirp->d_name);
if(len < 5 || strcasecmp(dirp->d_name + len - 4, ".ydk") != 0)
continue;
dirp->d_name[len - 4] = 0;
wchar_t wname[256];
BufferIO::DecodeUTF8(dirp->d_name, wname);
cbDeck->addItem(wname);
}
closedir(dir);
#endif
for(size_t i = 0; i < cbDeck->getItemCount(); ++i) { for(size_t i = 0; i < cbDeck->getItemCount(); ++i) {
if(!wcscmp(cbDeck->getItem(i), gameConf.lastdeck)) { if(!wcscmp(cbDeck->getItem(i), gameConf.lastdeck)) {
cbDeck->setSelected(i); cbDeck->setSelected(i);
...@@ -1128,61 +1019,17 @@ void Game::RefreshDeck(irr::gui::IGUIComboBox* cbDeck) { ...@@ -1128,61 +1019,17 @@ void Game::RefreshDeck(irr::gui::IGUIComboBox* cbDeck) {
} }
void Game::RefreshReplay() { void Game::RefreshReplay() {
lstReplayList->clear(); lstReplayList->clear();
#ifdef _WIN32 FileSystem::TraversalDir(L"./replay", [this](const wchar_t* name, bool isdir) {
WIN32_FIND_DATAW fdataw; if(!isdir && !mywcsncasecmp(wcsrchr(name, '.'), L".yrp", 4) && Replay::CheckReplay(name))
HANDLE fh = FindFirstFileW(L"./replay/*.yrp", &fdataw); lstReplayList->addItem(name);
if(fh == INVALID_HANDLE_VALUE) });
return;
do {
if(!(fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) && Replay::CheckReplay(fdataw.cFileName)) {
lstReplayList->addItem(fdataw.cFileName);
}
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./replay/")) == NULL)
return;
while((dirp = readdir(dir)) != NULL) {
size_t len = strlen(dirp->d_name);
if(len < 5 || strcasecmp(dirp->d_name + len - 4, ".yrp") != 0)
continue;
wchar_t wname[256];
BufferIO::DecodeUTF8(dirp->d_name, wname);
if(Replay::CheckReplay(wname))
lstReplayList->addItem(wname);
}
closedir(dir);
#endif
} }
void Game::RefreshSingleplay() { void Game::RefreshSingleplay() {
lstSinglePlayList->clear(); lstSinglePlayList->clear();
#ifdef _WIN32 FileSystem::TraversalDir(L"./single", [this](const wchar_t* name, bool isdir) {
WIN32_FIND_DATAW fdataw; if(!isdir && !mywcsncasecmp(wcsrchr(name, '.'), L".lua", 4))
HANDLE fh = FindFirstFileW(L"./single/*.lua", &fdataw); lstSinglePlayList->addItem(name);
if(fh == INVALID_HANDLE_VALUE) });
return;
do {
if(!(fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY))
lstSinglePlayList->addItem(fdataw.cFileName);
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./single/")) == NULL)
return;
while((dirp = readdir(dir)) != NULL) {
size_t len = strlen(dirp->d_name);
if(len < 5 || strcasecmp(dirp->d_name + len - 4, ".lua") != 0)
continue;
wchar_t wname[256];
BufferIO::DecodeUTF8(dirp->d_name, wname);
lstSinglePlayList->addItem(wname);
}
closedir(dir);
#endif
} }
void Game::RefreshLocales() { void Game::RefreshLocales() {
cbLocale->clear(); cbLocale->clear();
...@@ -1742,7 +1589,7 @@ void Game::ClearCardInfo(int player) { ...@@ -1742,7 +1589,7 @@ void Game::ClearCardInfo(int player) {
stText->setText(L""); stText->setText(L"");
scrCardText->setVisible(false); scrCardText->setVisible(false);
} }
void Game::AddChatMsg(wchar_t* msg, int player) { void Game::AddChatMsg(const wchar_t* msg, int player) {
for(int i = 7; i > 0; --i) { for(int i = 7; i > 0; --i) {
chatMsg[i] = chatMsg[i - 1]; chatMsg[i] = chatMsg[i - 1];
chatTiming[i] = chatTiming[i - 1]; chatTiming[i] = chatTiming[i - 1];
...@@ -1797,8 +1644,7 @@ void Game::ClearChatMsg() { ...@@ -1797,8 +1644,7 @@ void Game::ClearChatMsg() {
chatTiming[i] = 0; chatTiming[i] = 0;
} }
} }
void Game::AddDebugMsg(char* msg) void Game::AddDebugMsg(const char* msg) {
{
if (enable_log & 0x1) { if (enable_log & 0x1) {
wchar_t wbuf[1024]; wchar_t wbuf[1024];
BufferIO::DecodeUTF8(msg, wbuf); BufferIO::DecodeUTF8(msg, wbuf);
...@@ -1810,7 +1656,7 @@ void Game::AddDebugMsg(char* msg) ...@@ -1810,7 +1656,7 @@ void Game::AddDebugMsg(char* msg)
ErrorLog(msgbuf); ErrorLog(msgbuf);
} }
} }
void Game::ErrorLog(char* msg) { void Game::ErrorLog(const char* msg) {
FILE* fp = fopen("error.log", "at"); FILE* fp = fopen("error.log", "at");
if(!fp) if(!fp)
return; return;
...@@ -1821,69 +1667,48 @@ void Game::ErrorLog(char* msg) { ...@@ -1821,69 +1667,48 @@ void Game::ErrorLog(char* msg) {
fprintf(fp, "[%s]%s\n", timebuf, msg); fprintf(fp, "[%s]%s\n", timebuf, msg);
fclose(fp); fclose(fp);
} }
bool Game::MakeDirectory(const std::string folder) {
std::string folder_builder;
std::string sub;
sub.reserve(folder.size());
for(auto it = folder.begin(); it != folder.end(); ++it) {
const char c = *it;
sub.push_back(c);
if(c == '/' || it == folder.end() - 1) {
folder_builder.append(sub);
if(access(folder_builder.c_str(), 0) != 0)
#ifdef _WIN32
if(mkdir(folder_builder.c_str()) != 0)
#else
if(mkdir(folder_builder.c_str(), 0777) != 0)
#endif
return false;
sub.clear();
}
}
return true;
}
void Game::initUtils() { void Game::initUtils() {
//user files //user files
MakeDirectory("replay"); FileSystem::MakeDir("replay");
MakeDirectory("screenshots"); FileSystem::MakeDir("screenshots");
//cards from extra pack //cards from extra pack
MakeDirectory("expansions"); FileSystem::MakeDir("expansions");
//files in ygopro-starter-pack //files in ygopro-starter-pack
MakeDirectory("deck"); FileSystem::MakeDir("deck");
MakeDirectory("single"); FileSystem::MakeDir("single");
//original files //original files
MakeDirectory("script"); FileSystem::MakeDir("script");
MakeDirectory("skin"); FileSystem::MakeDir("skin");
MakeDirectory("textures"); FileSystem::MakeDir("textures");
//subdirs in textures //subdirs in textures
MakeDirectory("textures/act"); FileSystem::MakeDir("textures/act");
MakeDirectory("textures/attack"); FileSystem::MakeDir("textures/attack");
MakeDirectory("textures/bg"); FileSystem::MakeDir("textures/bg");
MakeDirectory("textures/bg_deck"); FileSystem::MakeDir("textures/bg_deck");
MakeDirectory("textures/bg_menu"); FileSystem::MakeDir("textures/bg_menu");
MakeDirectory("textures/cover"); FileSystem::MakeDir("textures/cover");
MakeDirectory("textures/cover2"); FileSystem::MakeDir("textures/cover2");
MakeDirectory("textures/pscale"); FileSystem::MakeDir("textures/pscale");
//sound //sound
#ifdef YGOPRO_USE_IRRKLANG #ifdef YGOPRO_USE_IRRKLANG
MakeDirectory("sound"); FileSystem::MakeDir("sound");
MakeDirectory("sound/BGM"); FileSystem::MakeDir("sound/BGM");
MakeDirectory("sound/BGM/advantage"); FileSystem::MakeDir("sound/BGM/advantage");
MakeDirectory("sound/BGM/deck"); FileSystem::MakeDir("sound/BGM/deck");
MakeDirectory("sound/BGM/disadvantage"); FileSystem::MakeDir("sound/BGM/disadvantage");
MakeDirectory("sound/BGM/duel"); FileSystem::MakeDir("sound/BGM/duel");
MakeDirectory("sound/BGM/lose"); FileSystem::MakeDir("sound/BGM/lose");
MakeDirectory("sound/BGM/menu"); FileSystem::MakeDir("sound/BGM/menu");
MakeDirectory("sound/BGM/win"); FileSystem::MakeDir("sound/BGM/win");
//custom sound //custom sound
MakeDirectory("sound/custom"); FileSystem::MakeDir("sound/custom");
MakeDirectory("sound/BGM/custom"); FileSystem::MakeDir("sound/BGM/custom");
#endif #endif
//locales //locales
MakeDirectory("locales"); FileSystem::MakeDir("locales");
//pics //pics
MakeDirectory("pics"); FileSystem::MakeDir("pics");
MakeDirectory("pics/field"); FileSystem::MakeDir("pics/field");
} }
void Game::ClearTextures() { void Game::ClearTextures() {
matManager.mCard.setTexture(0, 0); matManager.mCard.setTexture(0, 0);
......
...@@ -127,7 +127,6 @@ public: ...@@ -127,7 +127,6 @@ public:
void LoadExpansionDB(); void LoadExpansionDB();
void LoadExpansionDBDirectry(const char* path); void LoadExpansionDBDirectry(const char* path);
void LoadExpansionStrings(); void LoadExpansionStrings();
void LoadExpansionStringsDirectry(const char* path);
void RefreshDeck(irr::gui::IGUIComboBox* cbDeck); void RefreshDeck(irr::gui::IGUIComboBox* cbDeck);
void RefreshReplay(); void RefreshReplay();
void RefreshSingleplay(); void RefreshSingleplay();
...@@ -157,12 +156,10 @@ public: ...@@ -157,12 +156,10 @@ public:
void SaveConfig(); void SaveConfig();
void ShowCardInfo(int code, bool resize = false); void ShowCardInfo(int code, bool resize = false);
void ClearCardInfo(int player = 0); void ClearCardInfo(int player = 0);
void AddChatMsg(wchar_t* msg, int player); void AddChatMsg(const wchar_t* msg, int player);
void ClearChatMsg(); void ClearChatMsg();
void AddDebugMsg(char* msgbuf); void AddDebugMsg(const char* msgbuf);
void ErrorLog(char* msgbuf); void ErrorLog(const char* msgbuf);
bool MakeDirectory(const std::string folder);
void initUtils();
void ClearTextures(); void ClearTextures();
void CloseDuelWindow(); void CloseDuelWindow();
......
...@@ -301,23 +301,30 @@ void imageScaleNNAA(irr::video::IImage *src, irr::video::IImage *dest) { ...@@ -301,23 +301,30 @@ void imageScaleNNAA(irr::video::IImage *src, irr::video::IImage *dest) {
} }
} }
irr::video::ITexture* ImageManager::GetTextureFromFile(char* file, s32 width, s32 height) { irr::video::ITexture* ImageManager::GetTextureFromFile(char* file, s32 width, s32 height) {
#ifdef _WIN32
wchar_t name[1024];
BufferIO::DecodeUTF8(file, name);
#else
char* name = file;
#endif // _WIN32
if(mainGame->gameConf.use_image_scale) { if(mainGame->gameConf.use_image_scale) {
irr::video::ITexture* texture; irr::video::ITexture* texture;
irr::video::IImage* srcimg = driver->createImageFromFile(file); irr::video::IImage* srcimg = driver->createImageFromFile(name);
if(srcimg == NULL) if(srcimg == NULL)
return NULL; return NULL;
if(srcimg->getDimension() == irr::core::dimension2d<u32>(width, height)) { if(srcimg->getDimension() == irr::core::dimension2d<u32>(width, height)) {
texture = driver->addTexture(file, srcimg); texture = driver->addTexture(name, srcimg);
} else { } else {
video::IImage *destimg = driver->createImage(srcimg->getColorFormat(), irr::core::dimension2d<u32>(width, height)); video::IImage *destimg = driver->createImage(srcimg->getColorFormat(), irr::core::dimension2d<u32>(width, height));
imageScaleNNAA(srcimg, destimg); imageScaleNNAA(srcimg, destimg);
texture = driver->addTexture(file, destimg); texture = driver->addTexture(name, destimg);
destimg->drop(); destimg->drop();
} }
srcimg->drop(); srcimg->drop();
return texture; return texture;
} else { } else {
return driver->getTexture(file); return driver->getTexture(name);
} }
} }
irr::video::ITexture* ImageManager::GetTextureUnknown(s32 width, s32 height, int index) { irr::video::ITexture* ImageManager::GetTextureUnknown(s32 width, s32 height, int index) {
...@@ -329,39 +336,18 @@ irr::video::ITexture* ImageManager::GetTextureExpansions(char* file, s32 width, ...@@ -329,39 +336,18 @@ irr::video::ITexture* ImageManager::GetTextureExpansions(char* file, s32 width,
irr::video::ITexture* img = GetTextureExpansionsDirectry("./expansions", file, width, height); irr::video::ITexture* img = GetTextureExpansionsDirectry("./expansions", file, width, height);
if(img != NULL) if(img != NULL)
return img; return img;
#ifdef _WIN32 bool find = false;
char fpath[1000]; FileSystem::TraversalDir("./expansions", [this, file, width, height, &img, &find](const char* name, bool isdir) {
WIN32_FIND_DATAW fdataw; if(!find && isdir && strcmp(name, ".") && strcmp(name, "..")) {
HANDLE fh = FindFirstFileW(L"./expansions/*", &fdataw); char subdir[1024];
if(fh != INVALID_HANDLE_VALUE) { sprintf(subdir, "./expansions/%s", name);
do { img = GetTextureExpansionsDirectry(subdir, file, width, height);
if(wcscmp(L".",fdataw.cFileName) != 0 && wcscmp(L"..",fdataw.cFileName) != 0 && fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) { if(img)
char fname[780]; find = true;
BufferIO::EncodeUTF8(fdataw.cFileName, fname);
sprintf(fpath, "./expansions/%s", fname);
img = GetTextureExpansionsDirectry(fpath, file, width, height);
if(img != NULL)
return img;
}
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
}
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./expansions/")) != NULL) {
while((dirp = readdir(dir)) != NULL) {
if (strcmp(".", dirp->d_name) == 0 || strcmp("..", dirp->d_name) == 0 || dirp->d_type != DT_DIR)
continue;
char filepath[1000];
sprintf(filepath, "./expansions/%s/", dirp->d_name);
img = GetTextureExpansionsDirectry(filepath, file, width, height);
if(img != NULL)
return img;
} }
closedir(dir); });
} if(find)
#endif return img;
return img; return img;
} }
irr::video::ITexture* ImageManager::GetTextureExpansionsDirectry(const char* path, char* file, s32 width, s32 height) { irr::video::ITexture* ImageManager::GetTextureExpansionsDirectry(const char* path, char* file, s32 width, s32 height) {
......
#ifndef FILESYSTEM_H
#define FILESYSTEM_H
#include <string.h>
#include <functional>
#include "bufferio.h"
#ifdef _WIN32
#include <direct.h>
#include <sys/stat.h>
#else
#include <dirent.h>
#include <sys/stat.h>
#endif
#ifdef _WIN32
#include <Windows.h>
class FileSystem {
public:
static bool IsFileExists(const wchar_t* wfile) {
struct _stat fileStat;
return (_wstat(wfile, &fileStat) == 0) && !(fileStat.st_mode & _S_IFDIR);
}
static bool IsFileExists(const char* file) {
wchar_t wfile[1024];
BufferIO::DecodeUTF8(file, wfile);
return IsFileExists(wfile);
}
static bool IsDirExists(const wchar_t* wdir) {
struct _stat fileStat;
return (_wstat(wdir, &fileStat) == 0) && (fileStat.st_mode & _S_IFDIR);
}
static bool IsDirExists(const char* dir) {
wchar_t wdir[1024];
BufferIO::DecodeUTF8(dir, wdir);
return IsDirExists(wdir);
}
static bool MakeDir(const wchar_t* wdir) {
return _wmkdir(wdir) == 0;
}
static bool MakeDir(const char* dir) {
wchar_t wdir[1024];
BufferIO::DecodeUTF8(dir, wdir);
return MakeDir(wdir);
}
static void TraversalDir(const wchar_t* wpath, const std::function<void(const wchar_t*, bool)>& cb) {
wchar_t findstr[1024];
wcscpy(findstr, wpath);
wcscat(findstr, L"/*");
WIN32_FIND_DATAW fdataw;
HANDLE fh = FindFirstFileW(findstr, &fdataw);
if(fh == INVALID_HANDLE_VALUE)
return;
do {
cb(fdataw.cFileName, (fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY));
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
}
static void TraversalDir(const char* path, const std::function<void(const char*, bool)>& cb) {
wchar_t wpath[1024];
BufferIO::DecodeUTF8(path, wpath);
TraversalDir(wpath, [&cb](const wchar_t* wname, bool isdir) {
char name[1024];
BufferIO::EncodeUTF8(wname, name);
cb(name, isdir);
});
}
};
#else
class FileSystem {
public:
static bool IsFileExists(const char* file) {
struct stat fileStat;
return (stat(file, &fileStat) == 0) && !S_ISDIR(fileStat.st_mode);
}
static bool IsFileExists(const wchar_t* wfile) {
char file[1024];
BufferIO::EncodeUTF8(wfile, file);
return IsFileExists(file);
}
static bool IsDirExists(const char* dir) {
struct stat fileStat;
return (stat(dir, &fileStat) == 0) && S_ISDIR(fileStat.st_mode);
}
static bool IsDirExists(const wchar_t* wdir) {
char dir[1024];
BufferIO::EncodeUTF8(wdir, dir);
return IsDirExists(dir);
}
static bool MakeDir(const char* dir) {
return mkdir(dir, 0775) == 0;
}
static bool MakeDir(const wchar_t* wdir) {
char dir[1024];
BufferIO::EncodeUTF8(wdir, dir);
return MakeDir(dir);
}
static void TraversalDir(const char* path, const std::function<void(const char*, bool)>& cb) {
DIR* dir = nullptr;
struct dirent* dirp = nullptr;
if((dir = opendir(path)) == nullptr)
return;
struct stat fileStat;
while((dirp = readdir(dir)) != nullptr) {
char fname[1024];
strcpy(fname, path);
strcat(fname, "/");
strcat(fname, dirp->d_name);
stat(fname, &fileStat);
cb(dirp->d_name, S_ISDIR(fileStat.st_mode));
}
closedir(dir);
}
static void TraversalDir(const wchar_t* wpath, const std::function<void(const wchar_t*, bool)>& cb) {
char path[1024];
BufferIO::EncodeUTF8(wpath, path);
TraversalDir(path, [&cb](const char* name, bool isdir) {
wchar_t wname[1024];
BufferIO::DecodeUTF8(name, wname);
cb(wname, isdir);
});
}
};
#endif // _WIN32
#endif //FILESYSTEM_H
...@@ -17,6 +17,8 @@ Replay::~Replay() { ...@@ -17,6 +17,8 @@ Replay::~Replay() {
delete[] comp_data; delete[] comp_data;
} }
void Replay::BeginRecord() { void Replay::BeginRecord() {
if(!FileSystem::IsDirExists(L"./replay") && !FileSystem::MakeDir(L"./replay"))
return;
#ifdef _WIN32 #ifdef _WIN32
if(is_recording) if(is_recording)
CloseHandle(recording_fp); CloseHandle(recording_fp);
...@@ -123,6 +125,8 @@ void Replay::EndRecord() { ...@@ -123,6 +125,8 @@ void Replay::EndRecord() {
is_recording = false; is_recording = false;
} }
void Replay::SaveReplay(const wchar_t* name) { void Replay::SaveReplay(const wchar_t* name) {
if(!FileSystem::IsDirExists(L"./replay") && !FileSystem::MakeDir(L"./replay"))
return;
wchar_t fname[256]; wchar_t fname[256];
myswprintf(fname, L"./replay/%ls.yrp", name); myswprintf(fname, L"./replay/%ls.yrp", name);
#ifdef WIN32 #ifdef WIN32
......
...@@ -952,39 +952,18 @@ byte* ReplayMode::ScriptReaderEx(const char* script_name, int* slen) { ...@@ -952,39 +952,18 @@ byte* ReplayMode::ScriptReaderEx(const char* script_name, int* slen) {
buffer = ScriptReaderExDirectry("./beta", script_name, slen); buffer = ScriptReaderExDirectry("./beta", script_name, slen);
if(buffer) if(buffer)
return buffer; return buffer;
#ifdef _WIN32 bool find = false;
char fpath[1000]; FileSystem::TraversalDir("./expansions", [script_name, slen, &buffer, &find](const char* name, bool isdir) {
WIN32_FIND_DATAW fdataw; if(!find && isdir && strcmp(name, ".") && strcmp(name, "..")) {
HANDLE fh = FindFirstFileW(L"./expansions/*", &fdataw); char subdir[1024];
if(fh != INVALID_HANDLE_VALUE) { sprintf(subdir, "./expansions/%s", name);
do { buffer = ScriptReaderExDirectry(subdir, script_name, slen);
if(wcscmp(L".",fdataw.cFileName) != 0 && wcscmp(L"..",fdataw.cFileName) != 0 && fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
char fname[780];
BufferIO::EncodeUTF8(fdataw.cFileName, fname);
sprintf(fpath, "./expansions/%s", fname);
buffer = ScriptReaderExDirectry(fpath, script_name, slen);
if(buffer)
return buffer;
}
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
}
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./expansions/")) != NULL) {
while((dirp = readdir(dir)) != NULL) {
if (strcmp(".", dirp->d_name) == 0 || strcmp("..", dirp->d_name) == 0 || dirp->d_type != DT_DIR)
continue;
char filepath[1000];
sprintf(filepath, "./expansions/%s/", dirp->d_name);
buffer = ScriptReaderExDirectry(filepath, script_name, slen);
if(buffer) if(buffer)
return buffer; find = true;
} }
closedir(dir); });
} if(find)
#endif return buffer;
return default_script_reader(script_name, slen); return default_script_reader(script_name, slen);
} }
byte* ReplayMode::ScriptReaderExDirectry(const char* path, const char* script_name, int* slen, int pre_len) { byte* ReplayMode::ScriptReaderExDirectry(const char* path, const char* script_name, int* slen, int pre_len) {
......
...@@ -1565,39 +1565,18 @@ byte* SingleDuel::ScriptReaderEx(const char* script_name, int* slen) { ...@@ -1565,39 +1565,18 @@ byte* SingleDuel::ScriptReaderEx(const char* script_name, int* slen) {
buffer = ScriptReaderExDirectry("./beta", script_name, slen); buffer = ScriptReaderExDirectry("./beta", script_name, slen);
if(buffer) if(buffer)
return buffer; return buffer;
#ifdef _WIN32 bool find = false;
char fpath[1000]; FileSystem::TraversalDir("./expansions", [script_name, slen, &buffer, &find](const char* name, bool isdir) {
WIN32_FIND_DATAW fdataw; if(!find && isdir && strcmp(name, ".") && strcmp(name, "..")) {
HANDLE fh = FindFirstFileW(L"./expansions/*", &fdataw); char subdir[1024];
if(fh != INVALID_HANDLE_VALUE) { sprintf(subdir, "./expansions/%s", name);
do { buffer = ScriptReaderExDirectry(subdir, script_name, slen);
if(wcscmp(L".",fdataw.cFileName) != 0 && wcscmp(L"..",fdataw.cFileName) != 0 && fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
char fname[780];
BufferIO::EncodeUTF8(fdataw.cFileName, fname);
sprintf(fpath, "./expansions/%s", fname);
buffer = ScriptReaderExDirectry(fpath, script_name, slen);
if(buffer)
return buffer;
}
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
}
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./expansions/")) != NULL) {
while((dirp = readdir(dir)) != NULL) {
if (strcmp(".", dirp->d_name) == 0 || strcmp("..", dirp->d_name) == 0 || dirp->d_type != DT_DIR)
continue;
char filepath[1000];
sprintf(filepath, "./expansions/%s/", dirp->d_name);
buffer = ScriptReaderExDirectry(filepath, script_name, slen);
if(buffer) if(buffer)
return buffer; find = true;
} }
closedir(dir); });
} if(find)
#endif return buffer;
return default_script_reader(script_name, slen); return default_script_reader(script_name, slen);
} }
byte* SingleDuel::ScriptReaderExDirectry(const char* path, const char* script_name, int* slen, int pre_len) { byte* SingleDuel::ScriptReaderExDirectry(const char* path, const char* script_name, int* slen, int pre_len) {
......
...@@ -873,45 +873,23 @@ byte* SingleMode::ScriptReaderEx(const char* script_name, int* slen) { ...@@ -873,45 +873,23 @@ byte* SingleMode::ScriptReaderEx(const char* script_name, int* slen) {
buffer = ScriptReaderExDirectry("./beta", script_name, slen); buffer = ScriptReaderExDirectry("./beta", script_name, slen);
if(buffer) if(buffer)
return buffer; return buffer;
#ifdef _WIN32 bool find = false;
char fpath[1000]; FileSystem::TraversalDir("./expansions", [script_name, slen, &buffer, &find](const char* name, bool isdir) {
WIN32_FIND_DATAW fdataw; if(!find && isdir && strcmp(name, ".") && strcmp(name, "..")) {
HANDLE fh = FindFirstFileW(L"./expansions/*", &fdataw); char subdir[1024];
if(fh != INVALID_HANDLE_VALUE) { sprintf(subdir, "./expansions/%s", name);
do { buffer = ScriptReaderExDirectry(subdir, script_name, slen);
if(wcscmp(L".",fdataw.cFileName) != 0 && wcscmp(L"..",fdataw.cFileName) != 0 && fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
char fname[780];
BufferIO::EncodeUTF8(fdataw.cFileName, fname);
sprintf(fpath, "./expansions/%s", fname);
buffer = ScriptReaderExDirectry(fpath, script_name, slen);
if(buffer)
return buffer;
}
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
}
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./expansions/")) != NULL) {
while((dirp = readdir(dir)) != NULL) {
if (strcmp(".", dirp->d_name) == 0 || strcmp("..", dirp->d_name) == 0 || dirp->d_type != DT_DIR)
continue;
char filepath[1000];
sprintf(filepath, "./expansions/%s/", dirp->d_name);
buffer = ScriptReaderExDirectry(filepath, script_name, slen);
if(buffer) if(buffer)
return buffer; find = true;
} }
closedir(dir); });
} if(find)
#endif return buffer;
return ScriptReader(script_name, slen); return ScriptReader(script_name, slen);
} }
byte* SingleMode::ScriptReaderExDirectry(const char* path, const char* script_name, int* slen, int pre_len) { byte* SingleMode::ScriptReaderExDirectry(const char* path, const char* script_name, int* slen, int pre_len) {
char sname[256]; char sname[256];
strcpy(sname, path); sprintf(sname, "%s%s", path, script_name + pre_len); //default script name: ./script/c%d.lua
strcat(sname, script_name + pre_len);//default script name: ./script/c%d.lua
return ScriptReader(sname, slen); return ScriptReader(sname, slen);
} }
byte* SingleMode::ScriptReader(const char* script_name, int* slen) { byte* SingleMode::ScriptReader(const char* script_name, int* slen) {
......
#include "sound_manager.h" #include "sound_manager.h"
#ifndef _WIN32
#include <dirent.h>
#endif
#ifdef YGOPRO_USE_IRRKLANG #ifdef YGOPRO_USE_IRRKLANG
#include "../ikpmp3/ikpMP3.h" #include "../ikpmp3/ikpMP3.h"
#endif #endif
...@@ -31,55 +28,25 @@ bool SoundManager::Init() { ...@@ -31,55 +28,25 @@ bool SoundManager::Init() {
void SoundManager::RefreshBGMList() { void SoundManager::RefreshBGMList() {
#ifdef YGOPRO_USE_IRRKLANG #ifdef YGOPRO_USE_IRRKLANG
RefershBGMDir(L"", BGM_DUEL); RefershBGMDir(L"", BGM_DUEL);
RefershBGMDir(L"duel/", BGM_DUEL); RefershBGMDir(L"duel", BGM_DUEL);
RefershBGMDir(L"menu/", BGM_MENU); RefershBGMDir(L"menu", BGM_MENU);
RefershBGMDir(L"deck/", BGM_DECK); RefershBGMDir(L"deck", BGM_DECK);
RefershBGMDir(L"advantage/", BGM_ADVANTAGE); RefershBGMDir(L"advantage", BGM_ADVANTAGE);
RefershBGMDir(L"disadvantage/", BGM_DISADVANTAGE); RefershBGMDir(L"disadvantage", BGM_DISADVANTAGE);
RefershBGMDir(L"win/", BGM_WIN); RefershBGMDir(L"win", BGM_WIN);
RefershBGMDir(L"lose/", BGM_LOSE); RefershBGMDir(L"lose", BGM_LOSE);
RefershBGMDir(L"custom/", BGM_CUSTOM); RefershBGMDir(L"custom/", BGM_CUSTOM);
#endif #endif
} }
void SoundManager::RefershBGMDir(std::wstring path, int scene) { void SoundManager::RefershBGMDir(std::wstring path, int scene) {
#ifdef YGOPRO_USE_IRRKLANG std::wstring search = L"./sound/BGM/" + path;
#ifdef _WIN32 FileSystem::TraversalDir(search.c_str(), [this, &path, scene](const wchar_t* name, bool isdir) {
WIN32_FIND_DATAW fdataw; if(!isdir && (!mywcsncasecmp(wcsrchr(name, '.'), L".mp3", 4) || !mywcsncasecmp(wcsrchr(name, '.'), L".ogg", 4))) {
std::wstring search = L"./sound/BGM/" + path + L"*.*"; std::wstring filename = path + L"/" + name;
HANDLE fh = FindFirstFileW(search.c_str(), &fdataw); BGMList[BGM_ALL].push_back(filename);
if(fh == INVALID_HANDLE_VALUE) BGMList[scene].push_back(filename);
return; }
do { });
size_t len = wcslen(fdataw.cFileName);
if((fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) || len < 5
|| !(_wcsicmp(fdataw.cFileName + len - 4, L".mp3") == 0 || _wcsicmp(fdataw.cFileName + len - 4, L".ogg") == 0))
continue;
std::wstring filename = path + (std::wstring)fdataw.cFileName;
BGMList[BGM_ALL].push_back(filename);
BGMList[scene].push_back(filename);
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
#else
DIR * dir;
struct dirent * dirp;
std::wstring wsearchpath = L"./sound/BGM/" + path;
char searchpath[256];
BufferIO::EncodeUTF8(wsearchpath.c_str(), searchpath);
if((dir = opendir(searchpath)) == NULL)
return;
while((dirp = readdir(dir)) != NULL) {
size_t len = strlen(dirp->d_name);
if(len < 5 || !(strcasecmp(dirp->d_name + len - 4, ".mp3") == 0 || strcasecmp(dirp->d_name + len - 4, ".ogg")))
continue;
wchar_t wname[256];
BufferIO::DecodeUTF8(dirp->d_name, wname);
std::wstring filename = path + (std::wstring)wname;
BGMList[BGM_ALL].push_back(filename);
BGMList[scene].push_back(filename);
}
closedir(dir);
#endif
#endif
} }
void SoundManager::PlaySoundEffect(int sound) { void SoundManager::PlaySoundEffect(int sound) {
#ifdef YGOPRO_USE_IRRKLANG #ifdef YGOPRO_USE_IRRKLANG
......
...@@ -1706,39 +1706,18 @@ byte* TagDuel::ScriptReaderEx(const char* script_name, int* slen) { ...@@ -1706,39 +1706,18 @@ byte* TagDuel::ScriptReaderEx(const char* script_name, int* slen) {
buffer = ScriptReaderExDirectry("./beta", script_name, slen); buffer = ScriptReaderExDirectry("./beta", script_name, slen);
if(buffer) if(buffer)
return buffer; return buffer;
#ifdef _WIN32 bool find = false;
char fpath[1000]; FileSystem::TraversalDir("./expansions", [script_name, slen, &buffer, &find](const char* name, bool isdir) {
WIN32_FIND_DATAW fdataw; if(!find && isdir && strcmp(name, ".") && strcmp(name, "..")) {
HANDLE fh = FindFirstFileW(L"./expansions/*", &fdataw); char subdir[1024];
if(fh != INVALID_HANDLE_VALUE) { sprintf(subdir, "./expansions/%s", name);
do { buffer = ScriptReaderExDirectry(subdir, script_name, slen);
if(wcscmp(L".",fdataw.cFileName) != 0 && wcscmp(L"..",fdataw.cFileName) != 0 && fdataw.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
char fname[780];
BufferIO::EncodeUTF8(fdataw.cFileName, fname);
sprintf(fpath, "./expansions/%s", fname);
buffer = ScriptReaderExDirectry(fpath, script_name, slen);
if(buffer)
return buffer;
}
} while(FindNextFileW(fh, &fdataw));
FindClose(fh);
}
#else
DIR * dir;
struct dirent * dirp;
if((dir = opendir("./expansions/")) != NULL) {
while((dirp = readdir(dir)) != NULL) {
if (strcmp(".", dirp->d_name) == 0 || strcmp("..", dirp->d_name) == 0 || dirp->d_type != DT_DIR)
continue;
char filepath[1000];
sprintf(filepath, "./expansions/%s/", dirp->d_name);
buffer = ScriptReaderExDirectry(filepath, script_name, slen);
if(buffer) if(buffer)
return buffer; find = true;
} }
closedir(dir); });
} if(find)
#endif return buffer;
return default_script_reader(script_name, slen); return default_script_reader(script_name, slen);
} }
byte* TagDuel::ScriptReaderExDirectry(const char* path, const char* script_name, int* slen, int pre_len) { byte* TagDuel::ScriptReaderExDirectry(const char* path, const char* script_name, int* slen, int pre_len) {
......
Subproject commit 17c2dfaf1b157fb65e81ab05883106a3cc38e2ae Subproject commit a85b90e10c0a93763cc8650f743f924aae5a33b2
Subproject commit aaef0a01369bd09bf0e4dd812483df1d0f856f3a Subproject commit 74c6c3d87e622eaf8f80344d4b1578c69333cd16
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment