-
-
Notifications
You must be signed in to change notification settings - Fork 42
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
open-source the upcoming free features
- Loading branch information
Showing
8 changed files
with
888 additions
and
5 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,44 @@ | ||
#include <Geode/modify/EditorUI.hpp> | ||
#include <Geode/utils/general.hpp> | ||
#include <Geode/ui/Notification.hpp> | ||
|
||
using namespace geode::prelude; | ||
|
||
static bool isProbablyObjectString(std::string_view str) { | ||
// check if it starts with [0-9]+, | ||
size_t ix = 0; | ||
for (auto c : str) { | ||
if (!(c == ',' && ix > 0) && !(c >= '0' && c <= '9')) { | ||
return false; | ||
} | ||
if (c == ',' && ix > 0) { | ||
return true; | ||
} | ||
ix += 1; | ||
} | ||
return false; | ||
} | ||
|
||
class $modify(EditorUI) { | ||
// Hook these instead of copyObjects and pasteObjects so something like | ||
// onDuplicate doesn't get overwritten | ||
void doCopyObjects(bool idk) { | ||
EditorUI::doCopyObjects(idk); | ||
if (Mod::get()->template getSettingValue<bool>("copy-paste-from-clipboard")) { | ||
clipboard::write(GameManager::get()->m_editorClipboard); | ||
} | ||
} | ||
void doPasteObjects(bool idk) { | ||
if ( | ||
Mod::get()->template getSettingValue<bool>("copy-paste-from-clipboard") && | ||
isProbablyObjectString(clipboard::read()) | ||
) { | ||
GameManager::get()->m_editorClipboard = clipboard::read(); | ||
EditorUI::doPasteObjects(idk); | ||
Notification::create("Pasted Objects from Clipboard", NotificationIcon::Info)->show(); | ||
} | ||
else { | ||
EditorUI::doPasteObjects(idk); | ||
} | ||
} | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,270 @@ | ||
#include <Geode/modify/SetGroupIDLayer.hpp> | ||
#include <Geode/utils/cocos.hpp> | ||
|
||
using namespace geode::prelude; | ||
|
||
template <class T> | ||
class MixedValuesHandler final { | ||
protected: | ||
std::vector<typename T::GameObjectType*> m_targets; | ||
typename T::Type m_mixedSource; | ||
CCTextInputNode* m_input; | ||
|
||
public: | ||
void setup(GameObject* obj, CCArray* objs, CCTextInputNode* input) { | ||
m_input = input; | ||
m_targets.clear(); | ||
if (obj) { | ||
if constexpr (std::is_same_v<typename T::GameObjectType, GameObject>) { | ||
m_targets.push_back(obj); | ||
} | ||
else { | ||
if (auto o = typeinfo_cast<typename T::GameObjectType*>(obj)) { | ||
m_targets.push_back(o); | ||
} | ||
} | ||
} | ||
for (auto o : CCArrayExt<GameObject*>(objs)) { | ||
if constexpr (std::is_same_v<typename T::GameObjectType, GameObject>) { | ||
m_targets.push_back(o); | ||
} | ||
else { | ||
if (auto e = typeinfo_cast<typename T::GameObjectType*>(o)) { | ||
m_targets.push_back(e); | ||
} | ||
} | ||
} | ||
if (!m_targets.empty()) { | ||
m_mixedSource = T::get(m_targets.front()); | ||
} | ||
// Disable input until explicitly unmixed via button | ||
// This is because A) i'm too lazy to parse "Mix+N" strings and | ||
// B) this makes it clear how to unmix | ||
if (this->isMixed()) { | ||
input->setMouseEnabled(false); | ||
input->setTouchEnabled(false); | ||
|
||
auto unmixSpr = ButtonSprite::create("Unmix", "goldFont.fnt", "GJ_button_05.png", .8f); | ||
unmixSpr->setScale(.3f); | ||
auto unmixBtn = CCMenuItemExt::createSpriteExtra( | ||
unmixSpr, [this](auto) { | ||
this->override(T::DEFAULT_VALUE); | ||
} | ||
); | ||
auto menu = CCMenu::create(); | ||
menu->setID("unmix-menu"_spr); | ||
menu->ignoreAnchorPointForPosition(false); | ||
menu->setContentSize({ 25, 15 }); | ||
menu->addChildAtPosition(unmixBtn, Anchor::Center); | ||
input->getParent()->addChildAtPosition(menu, Anchor::Bottom, ccp(0, 0), false); | ||
} | ||
this->updateLabel(); | ||
} | ||
bool isMixed() const { | ||
if (m_targets.empty()) { | ||
return false; | ||
} | ||
auto value = T::get(m_targets.front()); | ||
for (auto obj : m_targets) { | ||
if (T::get(obj) != value) { | ||
return true; | ||
} | ||
} | ||
return false; | ||
} | ||
void override(typename T::Type value) const { | ||
m_input->setMouseEnabled(true); | ||
m_input->setTouchEnabled(true); | ||
m_input->getParent()->removeChildByID("unmix-menu"_spr); | ||
|
||
for (auto obj : m_targets) { | ||
T::set(obj, clamp(value, T::MIN_VALUE, T::MAX_VALUE)); | ||
} | ||
this->updateLabel(); | ||
} | ||
void parse(std::string const& value) const { | ||
this->override(numFromString<typename T::Type>(value).unwrapOr(0)); | ||
} | ||
void increment(typename T::Type value) const { | ||
for (auto obj : m_targets) { | ||
auto val = clamp(T::get(obj) + value, T::MIN_VALUE, T::MAX_VALUE); | ||
if (T::SKIP_ZERO && val == 0) { | ||
val = value > 0 ? 1 : -1; | ||
} | ||
T::set(obj, val); | ||
} | ||
this->updateLabel(); | ||
} | ||
void nextFree() const { | ||
std::set<typename T::Type> usedLayers; | ||
for (auto obj : m_targets) { | ||
usedLayers.insert(T::get(obj)); | ||
} | ||
typename T::Type nextFree; | ||
for (nextFree = T::MIN_VALUE; nextFree < T::MAX_VALUE; nextFree += 1) { | ||
if (!usedLayers.contains(nextFree)) { | ||
break; | ||
} | ||
} | ||
this->override(nextFree); | ||
} | ||
void updateLabel() const { | ||
if (m_targets.empty()) return; | ||
if (this->isMixed()) { | ||
auto value = T::get(m_targets.front()) - m_mixedSource; | ||
m_input->setString(value == 0 ? "Mixed" : fmt::format("Mix{:+}", value)); | ||
} | ||
else { | ||
m_input->setString(fmt::format("{}", T::get(m_targets.front()))); | ||
} | ||
} | ||
}; | ||
|
||
#define MIXABLE_GAME_OBJECT_PROP(name_, gty_, ty_, val_, min_, max_, def_, skip_zero_, prop_) \ | ||
struct name_ final { \ | ||
using Type = ty_; \ | ||
using GameObjectType = gty_; \ | ||
static constexpr Type MIXED_VALUE = val_; \ | ||
static constexpr Type MIN_VALUE = min_; \ | ||
static constexpr Type MAX_VALUE = max_; \ | ||
static constexpr Type DEFAULT_VALUE = def_; \ | ||
static constexpr bool SKIP_ZERO = skip_zero_; \ | ||
static Type get(GameObjectType* obj) { return obj->prop_; } \ | ||
static void set(GameObjectType* obj, Type value) { obj->prop_ = value; } \ | ||
}; | ||
|
||
MIXABLE_GAME_OBJECT_PROP(GOEL, GameObject, short, -1, 0, std::numeric_limits<short>::max(), 0, false, m_editorLayer); | ||
MIXABLE_GAME_OBJECT_PROP(GOEL2, GameObject, short, -1, 0, std::numeric_limits<short>::max(), 0, false, m_editorLayer2); | ||
MIXABLE_GAME_OBJECT_PROP(GOZO, GameObject, int, -1000, -999, 999, 2, true, m_zOrder); | ||
MIXABLE_GAME_OBJECT_PROP(EGOOV, EffectGameObject, int, -1, 0, std::numeric_limits<short>::max(), 0, false, m_ordValue); | ||
MIXABLE_GAME_OBJECT_PROP(EGOCV, EffectGameObject, int, -1, 0, std::numeric_limits<short>::max(), 0, false, m_channelValue); | ||
|
||
class $modify(SetGroupIDLayer) { | ||
struct Fields { | ||
MixedValuesHandler<GOEL> editorLayerHandler; | ||
MixedValuesHandler<GOEL2> editorLayer2Handler; | ||
MixedValuesHandler<GOZO> zOrderHandler; | ||
MixedValuesHandler<EGOOV> channelOrderHandler; | ||
MixedValuesHandler<EGOCV> channelHandler; | ||
}; | ||
|
||
$override | ||
bool init(GameObject* obj, CCArray* objs) { | ||
if (!SetGroupIDLayer::init(obj, objs)) | ||
return false; | ||
|
||
m_fields->editorLayerHandler.setup(obj, objs, m_editorLayerInput); | ||
m_fields->editorLayer2Handler.setup(obj, objs, m_editorLayer2Input); | ||
m_fields->zOrderHandler.setup(obj, objs, m_zOrderInput); | ||
if (m_orderInput) m_fields->channelOrderHandler.setup(obj, objs, m_orderInput); | ||
if (m_channelInput) m_fields->channelHandler.setup(obj, objs, m_channelInput); | ||
|
||
return true; | ||
} | ||
|
||
$override | ||
void onArrow(int tag, int increment) { | ||
if (tag == m_editorLayerInput->getTag()) { | ||
m_fields->editorLayerHandler.increment(increment); | ||
} | ||
else if (tag == m_editorLayer2Input->getTag()) { | ||
m_fields->editorLayer2Handler.increment(increment); | ||
} | ||
else if (tag == m_zOrderInput->getTag()) { | ||
m_fields->zOrderHandler.increment(increment); | ||
} | ||
else if (m_orderInput && tag == m_orderInput->getTag()) { | ||
m_fields->channelOrderHandler.increment(increment); | ||
} | ||
else if (m_channelInput && tag == m_channelInput->getTag()) { | ||
m_fields->channelHandler.increment(increment); | ||
} | ||
else { | ||
SetGroupIDLayer::onArrow(tag, increment); | ||
} | ||
} | ||
|
||
$override | ||
void onNextFreeEditorLayer1(CCObject* sender) { | ||
if (0) SetGroupIDLayer::onNextFreeEditorLayer1(sender); | ||
m_fields->editorLayerHandler.nextFree(); | ||
} | ||
$override | ||
void onNextFreeEditorLayer2(CCObject* sender) { | ||
if (0) SetGroupIDLayer::onNextFreeEditorLayer2(sender); | ||
m_fields->editorLayerHandler.nextFree(); | ||
} | ||
$override | ||
void onNextFreeOrderChannel(CCObject* sender) { | ||
if (0) SetGroupIDLayer::onNextFreeOrderChannel(sender); | ||
m_fields->channelOrderHandler.nextFree(); | ||
} | ||
|
||
$override | ||
void updateEditorLayerID() { | ||
if (0) SetGroupIDLayer::updateEditorLayerID(); | ||
} | ||
$override | ||
void updateEditorLabel() { | ||
if (0) SetGroupIDLayer::updateEditorLabel(); | ||
} | ||
$override | ||
void updateEditorLayerID2() { | ||
if (0) SetGroupIDLayer::updateEditorLayerID2(); | ||
} | ||
$override | ||
void updateEditorLabel2() { | ||
if (0) SetGroupIDLayer::updateEditorLabel2(); | ||
} | ||
$override | ||
void updateZOrder() { | ||
if (0) SetGroupIDLayer::updateZOrder(); | ||
} | ||
$override | ||
void updateZOrderLabel() { | ||
if (0) SetGroupIDLayer::updateZOrderLabel(); | ||
} | ||
$override | ||
void updateOrderChannel() { | ||
if (0) SetGroupIDLayer::updateOrderChannel(); | ||
} | ||
$override | ||
void updateOrderChannelLabel() { | ||
if (0) SetGroupIDLayer::updateOrderChannelLabel(); | ||
} | ||
$override | ||
void updateEditorOrder() { | ||
if (0) SetGroupIDLayer::updateEditorOrder(); | ||
} | ||
$override | ||
void updateEditorOrderLabel() { | ||
if (0) SetGroupIDLayer::updateEditorOrderLabel(); | ||
} | ||
|
||
// Skip textChanged resetting the value to 0 if the input string is 'Mixed' | ||
$override | ||
virtual void textChanged(CCTextInputNode* input) { | ||
auto str = std::string(input->getString()); | ||
if (str.size() && (str[0] == 'm' || str[0] == 'M')) { | ||
return; | ||
} | ||
if (input == m_editorLayerInput) { | ||
m_fields->editorLayerHandler.parse(input->getString()); | ||
} | ||
else if (input == m_editorLayer2Input) { | ||
m_fields->editorLayer2Handler.parse(input->getString()); | ||
} | ||
else if (input == m_zOrderInput) { | ||
m_fields->zOrderHandler.parse(input->getString()); | ||
} | ||
else if (input == m_orderInput) { | ||
m_fields->channelOrderHandler.parse(input->getString()); | ||
} | ||
else if (input == m_channelInput) { | ||
m_fields->channelHandler.parse(input->getString()); | ||
} | ||
else { | ||
SetGroupIDLayer::textChanged(input); | ||
} | ||
} | ||
}; |
Oops, something went wrong.