|
14 | 14 | #include <iterator>
|
15 | 15 | #include <optional>
|
16 | 16 | #include <span>
|
| 17 | +#include <string> |
| 18 | +#include <unordered_set> |
17 | 19 |
|
18 | 20 | #include <SDL_version.h>
|
19 | 21 | #include <expected.hpp>
|
@@ -58,6 +60,49 @@ namespace devilution {
|
58 | 60 |
|
59 | 61 | namespace {
|
60 | 62 |
|
| 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 | + |
61 | 106 | std::optional<Ini> ini;
|
62 | 107 |
|
63 | 108 | #if defined(__ANDROID__) || (defined(TARGET_OS_IPHONE) && TARGET_OS_IPHONE == 1)
|
@@ -158,6 +203,7 @@ bool HardwareCursorSupported()
|
158 | 203 | void LoadOptions()
|
159 | 204 | {
|
160 | 205 | LoadIni();
|
| 206 | + DiscoverMods(); |
161 | 207 | Options &options = GetOptions();
|
162 | 208 | for (OptionCategoryBase *pCategory : options.GetCategories()) {
|
163 | 209 | for (OptionEntryBase *pEntry : pCategory->GetEntries()) {
|
@@ -1478,20 +1524,31 @@ std::vector<OptionEntryBase *> ModOptions::GetEntries()
|
1478 | 1524 | return optionEntries;
|
1479 | 1525 | }
|
1480 | 1526 |
|
| 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 | + |
1481 | 1545 | std::forward_list<ModOptions::ModEntry> &ModOptions::GetModEntries()
|
1482 | 1546 | {
|
1483 | 1547 | if (modEntries)
|
1484 | 1548 | return *modEntries;
|
1485 | 1549 |
|
1486 | 1550 | std::vector<std::string> modNames = ini->getKeys(key);
|
1487 | 1551 |
|
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 |
| - |
1495 | 1552 | std::forward_list<ModOptions::ModEntry> &newModEntries = modEntries.emplace();
|
1496 | 1553 | for (auto &modName : modNames) {
|
1497 | 1554 | newModEntries.emplace_front(modName);
|
|
0 commit comments