Skip to content

Commit 0713033

Browse files
kphoenix137AJenboStephenCWills
authored
Dynamic mod detection
Co-authored-by: Anders Jenbo <[email protected]> Co-authored-by: Stephen C. Wills <[email protected]>
1 parent 520e9e8 commit 0713033

File tree

4 files changed

+136
-10
lines changed

4 files changed

+136
-10
lines changed

Source/options.cpp

+64-7
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,8 @@
1414
#include <iterator>
1515
#include <optional>
1616
#include <span>
17+
#include <string>
18+
#include <unordered_set>
1719

1820
#include <SDL_version.h>
1921
#include <expected.hpp>
@@ -58,6 +60,49 @@ namespace devilution {
5860

5961
namespace {
6062

63+
void DiscoverMods()
64+
{
65+
// Add mods available by default:
66+
std::unordered_set<std::string> modNames = { "clock" };
67+
68+
// Check if the mods directory exists.
69+
const std::string modsPath = StrCat(paths::PrefPath(), "mods");
70+
if (DirectoryExists(modsPath.c_str())) {
71+
// Find unpacked mods
72+
for (const std::string &modFolder : ListDirectories(modsPath.c_str())) {
73+
// Only consider this folder if the init.lua file exists.
74+
std::string modScriptPath = modsPath + modFolder + DIRECTORY_SEPARATOR_STR + "init.lua";
75+
if (!FileExists(modScriptPath.c_str()))
76+
continue;
77+
78+
modNames.insert(modFolder);
79+
}
80+
81+
// Find packed mods
82+
for (const std::string &modMpq : ListFiles(modsPath.c_str())) {
83+
if (!modMpq.ends_with(".mpq"))
84+
continue;
85+
86+
modNames.insert(modMpq.substr(0, modMpq.size() - 4));
87+
}
88+
}
89+
90+
// Get the list of mods currently stored in the INI.
91+
std::vector<std::string_view> existingMods = GetOptions().Mods.GetModList();
92+
93+
// Add new mods.
94+
for (const std::string &modName : modNames) {
95+
if (std::find(existingMods.begin(), existingMods.end(), modName) == existingMods.end())
96+
GetOptions().Mods.AddModEntry(modName);
97+
}
98+
99+
// Remove mods that are no longer installed.
100+
for (const std::string_view &modName : existingMods) {
101+
if (modNames.find(std::string(modName)) == modNames.end())
102+
GetOptions().Mods.RemoveModEntry(std::string(modName));
103+
}
104+
}
105+
61106
std::optional<Ini> ini;
62107

63108
#if defined(__ANDROID__) || (defined(TARGET_OS_IPHONE) && TARGET_OS_IPHONE == 1)
@@ -158,6 +203,7 @@ bool HardwareCursorSupported()
158203
void LoadOptions()
159204
{
160205
LoadIni();
206+
DiscoverMods();
161207
Options &options = GetOptions();
162208
for (OptionCategoryBase *pCategory : options.GetCategories()) {
163209
for (OptionEntryBase *pEntry : pCategory->GetEntries()) {
@@ -1478,20 +1524,31 @@ std::vector<OptionEntryBase *> ModOptions::GetEntries()
14781524
return optionEntries;
14791525
}
14801526

1527+
void ModOptions::AddModEntry(const std::string &modName)
1528+
{
1529+
auto &entries = GetModEntries();
1530+
entries.emplace_front(modName);
1531+
}
1532+
1533+
void ModOptions::RemoveModEntry(const std::string &modName)
1534+
{
1535+
if (!modEntries) {
1536+
return;
1537+
}
1538+
1539+
auto &entries = *modEntries;
1540+
entries.remove_if([&](const ModEntry &entry) {
1541+
return entry.name == modName;
1542+
});
1543+
}
1544+
14811545
std::forward_list<ModOptions::ModEntry> &ModOptions::GetModEntries()
14821546
{
14831547
if (modEntries)
14841548
return *modEntries;
14851549

14861550
std::vector<std::string> modNames = ini->getKeys(key);
14871551

1488-
// Add mods available by default:
1489-
for (const std::string_view modName : { "clock" }) {
1490-
if (c_find(modNames, modName) != modNames.end()) continue;
1491-
ini->set(key, modName, false);
1492-
modNames.emplace_back(modName);
1493-
}
1494-
14951552
std::forward_list<ModOptions::ModEntry> &newModEntries = modEntries.emplace();
14961553
for (auto &modName : modNames) {
14971554
newModEntries.emplace_front(modName);

Source/options.h

+2
Original file line numberDiff line numberDiff line change
@@ -830,6 +830,8 @@ struct ModOptions : OptionCategoryBase {
830830
std::vector<std::string_view> GetActiveModList();
831831
std::vector<std::string_view> GetModList();
832832
std::vector<OptionEntryBase *> GetEntries() override;
833+
void AddModEntry(const std::string &modName);
834+
void RemoveModEntry(const std::string &modName);
833835

834836
private:
835837
struct ModEntry {

Source/utils/file_util.cpp

+66-3
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,6 @@
55
#include <cstdint>
66
#include <cstring>
77
#include <limits>
8-
#include <string>
9-
#include <string_view>
10-
#include <vector>
118

129
#include <SDL.h>
1310

@@ -472,4 +469,70 @@ FILE *OpenFile(const char *path, const char *mode)
472469
#endif
473470
}
474471

472+
std::vector<std::string> ListDirectories(const char *path)
473+
{
474+
std::vector<std::string> dirs;
475+
#ifdef DVL_HAS_FILESYSTEM
476+
std::error_code ec;
477+
for (const auto &entry : std::filesystem::directory_iterator(reinterpret_cast<const char8_t *>(path), ec)) {
478+
if (!entry.is_directory())
479+
continue;
480+
std::u8string filename = entry.path().filename().u8string();
481+
dirs.emplace_back(filename.begin(), filename.end());
482+
}
483+
#elif defined(_WIN32)
484+
WIN32_FIND_DATAA findData;
485+
// Construct the search path by appending the directory separator and wildcard.
486+
std::string searchPath = std::string(path) + DIRECTORY_SEPARATOR_STR + "*";
487+
HANDLE hFind = FindFirstFileA(searchPath.c_str(), &findData);
488+
if (hFind == INVALID_HANDLE_VALUE)
489+
return dirs;
490+
do {
491+
std::string folder = findData.cFileName;
492+
// Skip the special entries "." and ".."
493+
if (folder == "." || folder == "..")
494+
continue;
495+
if (findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
496+
dirs.push_back(folder);
497+
} while (FindNextFileA(hFind, &findData));
498+
FindClose(hFind);
499+
#else
500+
static_assert(false, "ListDirectories not implemented for the current platform");
501+
#endif
502+
return dirs;
503+
}
504+
505+
std::vector<std::string> ListFiles(const char *path)
506+
{
507+
std::vector<std::string> files;
508+
#ifdef DVL_HAS_FILESYSTEM
509+
std::error_code ec;
510+
for (const auto &entry : std::filesystem::directory_iterator(reinterpret_cast<const char8_t *>(path), ec)) {
511+
if (!entry.is_regular_file())
512+
continue;
513+
std::u8string filename = entry.path().filename().u8string();
514+
files.emplace_back(filename.begin(), filename.end());
515+
}
516+
#elif defined(_WIN32)
517+
WIN32_FIND_DATAA findData;
518+
// Construct the search path by appending the directory separator and wildcard.
519+
std::string searchPath = std::string(path) + DIRECTORY_SEPARATOR_STR + "*";
520+
HANDLE hFind = FindFirstFileA(searchPath.c_str(), &findData);
521+
if (hFind == INVALID_HANDLE_VALUE)
522+
return files;
523+
do {
524+
std::string file = findData.cFileName;
525+
// Skip the special entries "." and ".."
526+
if (file == "." || file == "..")
527+
continue;
528+
if (!(findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY))
529+
files.push_back(file);
530+
} while (FindNextFileA(hFind, &findData));
531+
FindClose(hFind);
532+
#else
533+
static_assert(false, "ListFiles not implemented for the current platform");
534+
#endif
535+
return files;
536+
}
537+
475538
} // namespace devilution

Source/utils/file_util.h

+4
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
#include <memory>
66
#include <string>
77
#include <string_view>
8+
#include <vector>
89

910
namespace devilution {
1011

@@ -46,4 +47,7 @@ FILE *OpenFile(const char *path, const char *mode);
4647
std::unique_ptr<wchar_t[]> ToWideChar(std::string_view path);
4748
#endif
4849

50+
std::vector<std::string> ListDirectories(const char *path);
51+
std::vector<std::string> ListFiles(const char *path);
52+
4953
} // namespace devilution

0 commit comments

Comments
 (0)