#ifndef OPENMW_COMPONENTS_FX_WIDGETS_H #define OPENMW_COMPONENTS_FX_WIDGETS_H #include #include #include #include #include #include #include #include "types.hpp" namespace Gui { class AutoSizedTextBox; class AutoSizedButton; } namespace fx { namespace Widgets { enum Index { None = -1, Zero = 0, One = 1, Two = 2, Three = 3 }; class EditBase { public: virtual ~EditBase() = default; void setData(const std::shared_ptr& uniform, Index index = None) { mUniform = uniform; mIndex = index; } virtual void setValueFromUniform() = 0; virtual void toDefault() = 0; protected: std::weak_ptr mUniform; Index mIndex; }; class EditBool : public EditBase, public MyGUI::Widget { MYGUI_RTTI_DERIVED(EditBool) public: void setValue(bool value); void setValueFromUniform() override; void toDefault() override; private: void initialiseOverride() override; void notifyMouseButtonClick(MyGUI::Widget* sender); MyGUI::Button* mCheckbutton{nullptr}; MyGUI::Widget* mFill{nullptr}; }; template class EditNumber : public EditBase, public MyGUI::Widget { MYGUI_RTTI_DERIVED(EditNumber) public: void setValue(T value) { mValue = value; if constexpr (std::is_floating_point_v) mValueLabel->setCaption(Misc::StringUtils::format("%.3f", mValue)); else mValueLabel->setCaption(std::to_string(mValue)); float range = 0.f; float min = 0.f; if (auto uniform = mUniform.lock()) { if constexpr (std::is_fundamental_v) { uniform->template setValue(mValue); range = uniform->template getMax() - uniform->template getMin(); min = uniform->template getMin(); } else { UType uvalue = uniform->template getValue(); uvalue[mIndex] = mValue; uniform->template setValue(uvalue); range = uniform->template getMax()[mIndex] - uniform->template getMin()[mIndex]; min = uniform->template getMin()[mIndex]; } } float fill = (range == 0.f) ? 1.f : (mValue - min) / range; mFill->setRealSize(fill, 1.0); } void setValueFromUniform() override { if (auto uniform = mUniform.lock()) { T value; if constexpr (std::is_fundamental_v) value = uniform->template getValue(); else value = uniform->template getValue()[mIndex]; setValue(value); } } void toDefault() override { if (auto uniform = mUniform.lock()) { if constexpr (std::is_fundamental_v) setValue(uniform->template getDefault()); else setValue(uniform->template getDefault()[mIndex]); } } private: void initialiseOverride() override { Base::initialiseOverride(); assignWidget(mDragger, "Dragger"); assignWidget(mValueLabel, "Value"); assignWidget(mButtonIncrease, "ButtonIncrease"); assignWidget(mButtonDecrease, "ButtonDecrease"); assignWidget(mFill, "Fill"); mButtonIncrease->eventMouseButtonClick += MyGUI::newDelegate(this, &EditNumber::notifyButtonClicked); mButtonDecrease->eventMouseButtonClick += MyGUI::newDelegate(this, &EditNumber::notifyButtonClicked); mDragger->eventMouseButtonPressed += MyGUI::newDelegate(this, &EditNumber::notifyMouseButtonPressed); mDragger->eventMouseDrag += MyGUI::newDelegate(this, &EditNumber::notifyMouseButtonDragged); mDragger->eventMouseWheel += MyGUI::newDelegate(this, &EditNumber::notifyMouseWheel); } void notifyMouseWheel(MyGUI::Widget* sender, int rel) { auto uniform = mUniform.lock(); if (!uniform) return; if (rel > 0) increment(uniform->mStep); else increment(-uniform->mStep); } void notifyMouseButtonDragged(MyGUI::Widget* sender, int left, int top, MyGUI::MouseButton id) { if (id != MyGUI::MouseButton::Left) return; auto uniform = mUniform.lock(); if (!uniform) return; int delta = left - mLastPointerX; // allow finer tuning when shift is pressed constexpr double scaling = 20.0; T step = MyGUI::InputManager::getInstance().isShiftPressed() ? uniform->mStep / scaling : uniform->mStep; if (step == 0) { if constexpr (std::is_integral_v) step = 1; else step = uniform->mStep; } if (delta > 0) increment(step); else if (delta < 0) increment(-step); mLastPointerX = left; } void notifyMouseButtonPressed(MyGUI::Widget* sender, int left, int top, MyGUI::MouseButton id) { if (id != MyGUI::MouseButton::Left) return; mLastPointerX = left; } void increment(T step) { auto uniform = mUniform.lock(); if (!uniform) return; if constexpr (std::is_fundamental_v) setValue(std::clamp(uniform->template getValue() + step, uniform->template getMin(), uniform->template getMax())); else setValue(std::clamp(uniform->template getValue()[mIndex] + step, uniform->template getMin()[mIndex], uniform->template getMax()[mIndex])); } void notifyButtonClicked(MyGUI::Widget* sender) { auto uniform = mUniform.lock(); if (!uniform) return; if (sender == mButtonDecrease) increment(-uniform->mStep); else if (sender == mButtonIncrease) increment(uniform->mStep); } MyGUI::Button* mButtonDecrease{nullptr}; MyGUI::Button* mButtonIncrease{nullptr}; MyGUI::Widget* mDragger{nullptr}; MyGUI::Widget* mFill{nullptr}; MyGUI::TextBox* mValueLabel{nullptr}; T mValue{}; int mLastPointerX{0}; }; class EditNumberFloat4 : public EditNumber { MYGUI_RTTI_DERIVED(EditNumberFloat4) }; class EditNumberFloat3 : public EditNumber { MYGUI_RTTI_DERIVED(EditNumberFloat3) }; class EditNumberFloat2 : public EditNumber { MYGUI_RTTI_DERIVED(EditNumberFloat2) }; class EditNumberFloat : public EditNumber { MYGUI_RTTI_DERIVED(EditNumberFloat) }; class EditNumberInt : public EditNumber { MYGUI_RTTI_DERIVED(EditNumberInt) }; class UniformBase final : public MyGUI::Widget { MYGUI_RTTI_DERIVED(UniformBase) public: void init(const std::shared_ptr& uniform); void toDefault(); void addItem(EditBase* item); Gui::AutoSizedTextBox* getLabel() { return mLabel; } private: void notifyResetClicked(MyGUI::Widget* sender); void initialiseOverride() override; Gui::AutoSizedButton* mReset{nullptr}; Gui::AutoSizedTextBox* mLabel{nullptr}; MyGUI::Widget* mClient{nullptr}; std::vector mBases; }; } } #endif