fix: Various achievements issues (Edit the Hex, ROM Hacks) (#1985)
### Problem description
As described in #1846:
- the `Edit the Hex` achievement doesn't unlock when it should
- the `ROM Hacks` achievement is not using event-driven architecture
(the functions call `unlockAchievement` themselves)
### Implementation description
Firstly, for the `Edit the Hex` achievement:
- replaced the old event listener on `EventPatchCreated` with a listener
on `EventProviderDataModified`, which picks up bytes changes
- ensured the provider data change comes from a File provider, else
unlocking the achievement wouldn't make sense
- *Note*: a discovered side effect is that the "Fill" function modifies
the provider byte per byte (with a for loop)
- there is no use in testing the size of the data change, as it is
always 1 byte
- the Fill function could probably be reworked to fill in whole regions
at a time?
About the `ROM Hacks` achievement:
- implemented the new, still unused `EventPatchCreated` event.
- signal signature is `const unsigned char *, u64, const IPSKind`:
buffer pointer, buffer size, and IPS kind (IPS/IPS32)
- make use of the `::post` and `::subscribe` methods on said event to
unlock the achievement
- **WARNING::behaviour change**: the event's `post` signal has been
moved in the success branch of the IPS generation condition, meaning
that achievement will only unlock if IPS patch export has worked. I felt
it would make more sense than unlocking an achievement on an error, if
there was any to raise.
---------
Signed-off-by: BioTheWolff <47079795+BioTheWolff@users.noreply.github.com>
2024-12-06 00:36:42 +01:00
|
|
|
#include <iostream>
|
2023-08-06 21:33:15 +02:00
|
|
|
#include <hex/api/achievement_manager.hpp>
|
|
|
|
#include <hex/api/project_file_manager.hpp>
|
2023-11-30 11:23:12 +01:00
|
|
|
#include <hex/api/event_manager.hpp>
|
2023-08-06 21:33:15 +02:00
|
|
|
|
|
|
|
#include <hex/helpers/crypto.hpp>
|
fix: Various achievements issues (Edit the Hex, ROM Hacks) (#1985)
### Problem description
As described in #1846:
- the `Edit the Hex` achievement doesn't unlock when it should
- the `ROM Hacks` achievement is not using event-driven architecture
(the functions call `unlockAchievement` themselves)
### Implementation description
Firstly, for the `Edit the Hex` achievement:
- replaced the old event listener on `EventPatchCreated` with a listener
on `EventProviderDataModified`, which picks up bytes changes
- ensured the provider data change comes from a File provider, else
unlocking the achievement wouldn't make sense
- *Note*: a discovered side effect is that the "Fill" function modifies
the provider byte per byte (with a for loop)
- there is no use in testing the size of the data change, as it is
always 1 byte
- the Fill function could probably be reworked to fill in whole regions
at a time?
About the `ROM Hacks` achievement:
- implemented the new, still unused `EventPatchCreated` event.
- signal signature is `const unsigned char *, u64, const IPSKind`:
buffer pointer, buffer size, and IPS kind (IPS/IPS32)
- make use of the `::post` and `::subscribe` methods on said event to
unlock the achievement
- **WARNING::behaviour change**: the event's `post` signal has been
moved in the success branch of the IPS generation condition, meaning
that achievement will only unlock if IPS patch export has worked. I felt
it would make more sense than unlocking an achievement on an error, if
there was any to raise.
---------
Signed-off-by: BioTheWolff <47079795+BioTheWolff@users.noreply.github.com>
2024-12-06 00:36:42 +01:00
|
|
|
#include <hex/providers/provider.hpp>
|
2023-08-06 21:33:15 +02:00
|
|
|
|
2023-12-26 00:22:47 +01:00
|
|
|
#include <toasts/toast_notification.hpp>
|
2023-12-23 21:09:41 +01:00
|
|
|
#include <popups/popup_notification.hpp>
|
|
|
|
#include <popups/popup_text_input.hpp>
|
2023-08-06 21:33:15 +02:00
|
|
|
|
|
|
|
#include <nlohmann/json.hpp>
|
|
|
|
#include <romfs/romfs.hpp>
|
|
|
|
|
|
|
|
namespace hex::plugin::builtin {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class AchievementStartingOut : public Achievement {
|
|
|
|
public:
|
2023-12-19 12:22:28 +01:00
|
|
|
explicit AchievementStartingOut(UnlocalizedString unlocalizedName) : Achievement("hex.builtin.achievement.starting_out", std::move(unlocalizedName)) { }
|
2023-08-06 21:33:15 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class AchievementHexEditor : public Achievement {
|
|
|
|
public:
|
2023-12-19 12:22:28 +01:00
|
|
|
explicit AchievementHexEditor(UnlocalizedString unlocalizedName) : Achievement("hex.builtin.achievement.hex_editor", std::move(unlocalizedName)) { }
|
2023-08-06 21:33:15 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class AchievementPatterns : public Achievement {
|
|
|
|
public:
|
2023-12-19 12:22:28 +01:00
|
|
|
explicit AchievementPatterns(UnlocalizedString unlocalizedName) : Achievement("hex.builtin.achievement.patterns", std::move(unlocalizedName)) { }
|
2023-08-06 21:33:15 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class AchievementDataProcessor : public Achievement {
|
|
|
|
public:
|
2023-12-19 12:22:28 +01:00
|
|
|
explicit AchievementDataProcessor(UnlocalizedString unlocalizedName) : Achievement("hex.builtin.achievement.data_processor", std::move(unlocalizedName)) { }
|
2023-08-06 21:33:15 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class AchievementFind : public Achievement {
|
|
|
|
public:
|
2023-12-19 12:22:28 +01:00
|
|
|
explicit AchievementFind(UnlocalizedString unlocalizedName) : Achievement("hex.builtin.achievement.find", std::move(unlocalizedName)) { }
|
2023-08-06 21:33:15 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class AchievementMisc : public Achievement {
|
|
|
|
public:
|
2023-12-19 12:22:28 +01:00
|
|
|
explicit AchievementMisc(UnlocalizedString unlocalizedName) : Achievement("hex.builtin.achievement.misc", std::move(unlocalizedName)) { }
|
2023-08-06 21:33:15 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
void registerGettingStartedAchievements() {
|
|
|
|
AchievementManager::addAchievement<AchievementStartingOut>("hex.builtin.achievement.starting_out.docs.name")
|
|
|
|
.setDescription("hex.builtin.achievement.starting_out.docs.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/open-book.png").span());
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementStartingOut>("hex.builtin.achievement.starting_out.open_file.name")
|
|
|
|
.setDescription("hex.builtin.achievement.starting_out.open_file.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/page-facing-up.png").span());
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementStartingOut>("hex.builtin.achievement.starting_out.save_project.name")
|
|
|
|
.setDescription("hex.builtin.achievement.starting_out.save_project.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/card-index-dividers.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.starting_out.open_file.name");
|
|
|
|
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementStartingOut>("hex.builtin.achievement.starting_out.crash.name")
|
|
|
|
.setDescription("hex.builtin.achievement.starting_out.crash.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/collision-symbol.png").span())
|
|
|
|
.setInvisible();
|
|
|
|
}
|
|
|
|
|
|
|
|
void registerHexEditorAchievements() {
|
|
|
|
AchievementManager::addAchievement<AchievementHexEditor>("hex.builtin.achievement.hex_editor.select_byte.name")
|
|
|
|
.setDescription("hex.builtin.achievement.hex_editor.select_byte.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/bookmark-tabs.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.starting_out.open_file.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementHexEditor>("hex.builtin.achievement.hex_editor.open_new_view.name")
|
|
|
|
.setDescription("hex.builtin.achievement.hex_editor.open_new_view.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/frame-with-picture.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.hex_editor.create_bookmark.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementHexEditor>("hex.builtin.achievement.hex_editor.modify_byte.name")
|
|
|
|
.setDescription("hex.builtin.achievement.hex_editor.modify_byte.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/pencil.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.hex_editor.select_byte.name")
|
|
|
|
.addVisibilityRequirement("hex.builtin.achievement.hex_editor.select_byte.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementHexEditor>("hex.builtin.achievement.hex_editor.copy_as.name")
|
|
|
|
.setDescription("hex.builtin.achievement.hex_editor.copy_as.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/copy.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.hex_editor.modify_byte.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementHexEditor>("hex.builtin.achievement.hex_editor.create_patch.name")
|
|
|
|
.setDescription("hex.builtin.achievement.hex_editor.create_patch.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/adhesive-bandage.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.hex_editor.modify_byte.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementHexEditor>("hex.builtin.achievement.hex_editor.fill.name")
|
|
|
|
.setDescription("hex.builtin.achievement.hex_editor.fill.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/water-wave.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.hex_editor.select_byte.name")
|
|
|
|
.addVisibilityRequirement("hex.builtin.achievement.hex_editor.select_byte.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementHexEditor>("hex.builtin.achievement.hex_editor.create_bookmark.name")
|
|
|
|
.setDescription("hex.builtin.achievement.hex_editor.create_bookmark.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/bookmark.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.hex_editor.select_byte.name")
|
|
|
|
.addVisibilityRequirement("hex.builtin.achievement.hex_editor.select_byte.name");
|
|
|
|
}
|
|
|
|
|
|
|
|
void registerPatternsAchievements() {
|
|
|
|
AchievementManager::addAchievement<AchievementPatterns>("hex.builtin.achievement.patterns.place_menu.name")
|
|
|
|
.setDescription("hex.builtin.achievement.patterns.place_menu.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/clipboard.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.hex_editor.select_byte.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementPatterns>("hex.builtin.achievement.patterns.load_existing.name")
|
|
|
|
.setDescription("hex.builtin.achievement.patterns.load_existing.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/hourglass.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.patterns.place_menu.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementPatterns>("hex.builtin.achievement.patterns.modify_data.name")
|
|
|
|
.setDescription("hex.builtin.achievement.patterns.modify_data.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/hammer.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.patterns.place_menu.name");
|
|
|
|
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementPatterns>("hex.builtin.achievement.patterns.data_inspector.name")
|
|
|
|
.setDescription("hex.builtin.achievement.patterns.data_inspector.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/eye-in-speech-bubble.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.hex_editor.select_byte.name");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void registerFindAchievements() {
|
|
|
|
AchievementManager::addAchievement<AchievementFind>("hex.builtin.achievement.find.find_strings.name")
|
|
|
|
.setDescription("hex.builtin.achievement.find.find_strings.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/ring.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.starting_out.open_file.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementFind>("hex.builtin.achievement.find.find_specific_string.name")
|
|
|
|
.setDescription("hex.builtin.achievement.find.find_specific_string.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/right-pointing-magnifying-glass.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.find.find_strings.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementFind>("hex.builtin.achievement.find.find_numeric.name")
|
|
|
|
.setDescription("hex.builtin.achievement.find.find_numeric.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/abacus.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.find.find_strings.name");
|
|
|
|
}
|
|
|
|
|
|
|
|
void registerDataProcessorAchievements() {
|
|
|
|
AchievementManager::addAchievement<AchievementDataProcessor>("hex.builtin.achievement.data_processor.place_node.name")
|
|
|
|
.setDescription("hex.builtin.achievement.data_processor.place_node.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/cloud.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.starting_out.open_file.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementDataProcessor>("hex.builtin.achievement.data_processor.create_connection.name")
|
|
|
|
.setDescription("hex.builtin.achievement.data_processor.create_connection.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/linked-paperclips.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.data_processor.place_node.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementDataProcessor>("hex.builtin.achievement.data_processor.modify_data.name")
|
|
|
|
.setDescription("hex.builtin.achievement.data_processor.modify_data.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/hammer-and-pick.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.data_processor.create_connection.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementDataProcessor>("hex.builtin.achievement.data_processor.custom_node.name")
|
|
|
|
.setDescription("hex.builtin.achievement.data_processor.custom_node.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/wrench.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.data_processor.create_connection.name");
|
|
|
|
}
|
|
|
|
|
|
|
|
void registerMiscAchievements() {
|
|
|
|
AchievementManager::addAchievement<AchievementMisc>("hex.builtin.achievement.misc.analyze_file.name")
|
|
|
|
.setDescription("hex.builtin.achievement.misc.analyze_file.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/brain.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.starting_out.open_file.name");
|
|
|
|
|
|
|
|
AchievementManager::addAchievement<AchievementMisc>("hex.builtin.achievement.misc.download_from_store.name")
|
|
|
|
.setDescription("hex.builtin.achievement.misc.download_from_store.desc")
|
|
|
|
.setIcon(romfs::get("assets/achievements/package.png").span())
|
|
|
|
.addRequirement("hex.builtin.achievement.starting_out.open_file.name");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void registerEvents() {
|
2023-12-08 10:29:44 +01:00
|
|
|
EventRegionSelected::subscribe([](const auto ®ion) {
|
2023-08-06 21:33:15 +02:00
|
|
|
if (region.getSize() > 1)
|
|
|
|
AchievementManager::unlockAchievement("hex.builtin.achievement.hex_editor", "hex.builtin.achievement.hex_editor.select_byte.name");
|
|
|
|
});
|
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
EventBookmarkCreated::subscribe([](const auto&) {
|
2023-08-06 21:33:15 +02:00
|
|
|
AchievementManager::unlockAchievement("hex.builtin.achievement.hex_editor", "hex.builtin.achievement.hex_editor.create_bookmark.name");
|
|
|
|
});
|
|
|
|
|
fix: Various achievements issues (Edit the Hex, ROM Hacks) (#1985)
### Problem description
As described in #1846:
- the `Edit the Hex` achievement doesn't unlock when it should
- the `ROM Hacks` achievement is not using event-driven architecture
(the functions call `unlockAchievement` themselves)
### Implementation description
Firstly, for the `Edit the Hex` achievement:
- replaced the old event listener on `EventPatchCreated` with a listener
on `EventProviderDataModified`, which picks up bytes changes
- ensured the provider data change comes from a File provider, else
unlocking the achievement wouldn't make sense
- *Note*: a discovered side effect is that the "Fill" function modifies
the provider byte per byte (with a for loop)
- there is no use in testing the size of the data change, as it is
always 1 byte
- the Fill function could probably be reworked to fill in whole regions
at a time?
About the `ROM Hacks` achievement:
- implemented the new, still unused `EventPatchCreated` event.
- signal signature is `const unsigned char *, u64, const IPSKind`:
buffer pointer, buffer size, and IPS kind (IPS/IPS32)
- make use of the `::post` and `::subscribe` methods on said event to
unlock the achievement
- **WARNING::behaviour change**: the event's `post` signal has been
moved in the success branch of the IPS generation condition, meaning
that achievement will only unlock if IPS patch export has worked. I felt
it would make more sense than unlocking an achievement on an error, if
there was any to raise.
---------
Signed-off-by: BioTheWolff <47079795+BioTheWolff@users.noreply.github.com>
2024-12-06 00:36:42 +01:00
|
|
|
EventProviderDataModified::subscribe([](const prv::Provider *, u64, const u64, const u8*) {
|
|
|
|
// Warning: overlaps with the "Flood fill" achievement, since "Fill" works by writing to bytes one-by-one.
|
|
|
|
// Thus, we do not check for size, that will always be equal to 1 even during a fill operation.
|
2023-08-06 21:33:15 +02:00
|
|
|
AchievementManager::unlockAchievement("hex.builtin.achievement.hex_editor", "hex.builtin.achievement.hex_editor.modify_byte.name");
|
|
|
|
});
|
|
|
|
|
fix: Various achievements issues (Edit the Hex, ROM Hacks) (#1985)
### Problem description
As described in #1846:
- the `Edit the Hex` achievement doesn't unlock when it should
- the `ROM Hacks` achievement is not using event-driven architecture
(the functions call `unlockAchievement` themselves)
### Implementation description
Firstly, for the `Edit the Hex` achievement:
- replaced the old event listener on `EventPatchCreated` with a listener
on `EventProviderDataModified`, which picks up bytes changes
- ensured the provider data change comes from a File provider, else
unlocking the achievement wouldn't make sense
- *Note*: a discovered side effect is that the "Fill" function modifies
the provider byte per byte (with a for loop)
- there is no use in testing the size of the data change, as it is
always 1 byte
- the Fill function could probably be reworked to fill in whole regions
at a time?
About the `ROM Hacks` achievement:
- implemented the new, still unused `EventPatchCreated` event.
- signal signature is `const unsigned char *, u64, const IPSKind`:
buffer pointer, buffer size, and IPS kind (IPS/IPS32)
- make use of the `::post` and `::subscribe` methods on said event to
unlock the achievement
- **WARNING::behaviour change**: the event's `post` signal has been
moved in the success branch of the IPS generation condition, meaning
that achievement will only unlock if IPS patch export has worked. I felt
it would make more sense than unlocking an achievement on an error, if
there was any to raise.
---------
Signed-off-by: BioTheWolff <47079795+BioTheWolff@users.noreply.github.com>
2024-12-06 00:36:42 +01:00
|
|
|
EventPatchCreated::subscribe([](const u8 *, u8, PatchKind) {
|
|
|
|
AchievementManager::unlockAchievement("hex.builtin.achievement.hex_editor", "hex.builtin.achievement.hex_editor.create_patch.name");
|
|
|
|
});
|
|
|
|
|
2023-08-06 21:33:15 +02:00
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
EventImHexStartupFinished::subscribe(AchievementManager::loadProgress);
|
|
|
|
EventAchievementUnlocked::subscribe([](const Achievement &) {
|
2023-10-20 13:34:45 +02:00
|
|
|
AchievementManager::storeProgress();
|
|
|
|
});
|
2023-08-06 21:33:15 +02:00
|
|
|
|
2023-10-20 13:34:45 +02:00
|
|
|
// Clear temporary achievements when the last provider is closed
|
2023-12-08 10:29:44 +01:00
|
|
|
EventProviderChanged::subscribe([](hex::prv::Provider *oldProvider, const hex::prv::Provider *newProvider) {
|
2024-12-14 21:35:54 +01:00
|
|
|
std::ignore = oldProvider;
|
2023-08-06 21:33:15 +02:00
|
|
|
if (newProvider == nullptr) {
|
|
|
|
AchievementManager::clearTemporary();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void registerChallengeAchievementHandlers() {
|
|
|
|
static std::string challengeAchievement;
|
|
|
|
static std::string challengeDescription;
|
|
|
|
|
|
|
|
static std::map<std::string, std::vector<u8>> icons;
|
|
|
|
|
|
|
|
ProjectFile::registerHandler({
|
|
|
|
.basePath = "challenge",
|
|
|
|
.required = false,
|
2023-12-27 16:33:49 +01:00
|
|
|
.load = [](const std::fs::path &basePath, const Tar &tar) {
|
2023-08-06 21:33:15 +02:00
|
|
|
if (!tar.contains(basePath / "achievements.json") || !tar.contains(basePath / "description.txt"))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
challengeAchievement = tar.readString(basePath / "achievements.json");
|
|
|
|
challengeDescription = tar.readString(basePath / "description.txt");
|
|
|
|
|
|
|
|
nlohmann::json unlockedJson;
|
|
|
|
if (tar.contains(basePath / "unlocked.json")) {
|
|
|
|
unlockedJson = nlohmann::json::parse(tar.readString(basePath / "unlocked.json"));
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
auto json = nlohmann::json::parse(challengeAchievement);
|
|
|
|
|
|
|
|
if (json.contains("achievements")) {
|
|
|
|
for (const auto &achievement : json["achievements"]) {
|
2024-01-21 14:31:19 +01:00
|
|
|
auto &newAchievement = AchievementManager::addTemporaryAchievement<Achievement>("hex.builtin.achievement.challenge", achievement["name"].get<std::string>())
|
2023-08-06 21:33:15 +02:00
|
|
|
.setDescription(achievement["description"]);
|
|
|
|
|
|
|
|
if (achievement.contains("icon")) {
|
|
|
|
if (const auto &icon = achievement["icon"]; icon.is_string() && !icon.is_null()) {
|
|
|
|
auto iconPath = icon.get<std::string>();
|
|
|
|
|
|
|
|
auto data = tar.readVector(basePath / iconPath);
|
|
|
|
newAchievement.setIcon(data);
|
|
|
|
icons[iconPath] = std::move(data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (achievement.contains("requirements")) {
|
|
|
|
if (const auto &requirements = achievement["requirements"]; requirements.is_array()) {
|
|
|
|
for (const auto &requirement : requirements) {
|
|
|
|
newAchievement.addRequirement(requirement.get<std::string>());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (achievement.contains("visibility_requirements")) {
|
|
|
|
if (const auto &requirements = achievement["visibility_requirements"]; requirements.is_array()) {
|
|
|
|
for (const auto &requirement : requirements) {
|
|
|
|
newAchievement.addVisibilityRequirement(requirement.get<std::string>());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (achievement.contains("password")) {
|
|
|
|
if (const auto &password = achievement["password"]; password.is_string() && !password.is_null()) {
|
|
|
|
newAchievement.setClickCallback([password = password.get<std::string>()](Achievement &achievement) {
|
|
|
|
if (password.empty())
|
|
|
|
achievement.setUnlocked(true);
|
|
|
|
else
|
2023-12-23 21:09:41 +01:00
|
|
|
ui::PopupTextInput::open("Enter Password", "Enter the password to unlock this achievement", [password, &achievement](const std::string &input) {
|
2023-08-06 21:33:15 +02:00
|
|
|
if (input == password)
|
|
|
|
achievement.setUnlocked(true);
|
|
|
|
else
|
2023-12-26 00:22:47 +01:00
|
|
|
ui::ToastWarning::open("The password you entered was incorrect.");
|
2023-08-06 21:33:15 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
if (unlockedJson.contains("achievements") && unlockedJson["achievements"].is_array()) {
|
|
|
|
for (const auto &unlockedAchievement : unlockedJson["achievements"]) {
|
|
|
|
if (unlockedAchievement.is_string() && unlockedAchievement.get<std::string>() == achievement["name"].get<std::string>()) {
|
|
|
|
newAchievement.setUnlocked(true);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (const nlohmann::json::exception &e) {
|
|
|
|
log::error("Failed to load challenge project: {}", e.what());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-12-23 21:09:41 +01:00
|
|
|
ui::PopupInfo::open(challengeDescription);
|
2023-08-06 21:33:15 +02:00
|
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
},
|
2023-12-27 16:33:49 +01:00
|
|
|
.store = [](const std::fs::path &basePath, const Tar &tar) {
|
2023-08-06 21:33:15 +02:00
|
|
|
if (!challengeAchievement.empty())
|
|
|
|
tar.writeString(basePath / "achievements.json", challengeAchievement);
|
|
|
|
if (!challengeDescription.empty())
|
|
|
|
tar.writeString(basePath / "description.txt", challengeDescription);
|
|
|
|
|
|
|
|
for (const auto &[iconPath, data] : icons) {
|
|
|
|
tar.writeVector(basePath / iconPath, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
nlohmann::json unlockedJson;
|
|
|
|
|
|
|
|
unlockedJson["achievements"] = nlohmann::json::array();
|
|
|
|
for (const auto &[categoryName, achievements] : AchievementManager::getAchievements()) {
|
|
|
|
for (const auto &[achievementName, achievement] : achievements) {
|
|
|
|
if (achievement->isTemporary() && achievement->isUnlocked()) {
|
|
|
|
unlockedJson["achievements"].push_back(achievementName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tar.writeString(basePath / "unlocked.json", unlockedJson.dump(4));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void registerAchievements() {
|
|
|
|
registerGettingStartedAchievements();
|
|
|
|
registerHexEditorAchievements();
|
|
|
|
registerPatternsAchievements();
|
|
|
|
registerFindAchievements();
|
|
|
|
registerDataProcessorAchievements();
|
|
|
|
registerMiscAchievements();
|
|
|
|
|
|
|
|
registerEvents();
|
|
|
|
registerChallengeAchievementHandlers();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|