#pragma once
#include <vector>
namespace DFHack {
    #define GUI_HOOK_DECLARE(name, rtype, args) DFHACK_EXPORT extern DFHack::GuiHooks::Hook<rtype args> name
    #define GUI_HOOK_DEFINE(name, base_func) DFHack::GuiHooks::Hook<decltype(base_func)> name(base_func)
    #define GUI_HOOK_TOP(name) name.top()
    #define GUI_HOOK_CALLBACK(hook, name, callback) DFHack::GuiHooks::Hook<decltype(callback)>::Callback name(&hook, callback)
    namespace GuiHooks {
        template <typename T_func>
        class Hook {
            typedef Hook<T_func> T_hook;
            friend class Callback;
            T_func *base_func;
            std::vector<T_func*> funcs;
            void add(T_func *func)
            {
                if (std::find(funcs.begin(), funcs.end(), func) == funcs.end())
                    funcs.push_back(func);
            }
            void remove(T_func *func)
            {
                auto it = std::find(funcs.begin(), funcs.end(), func);
                if (it != funcs.end())
                    funcs.erase(it);
            }
        public:
            Hook(T_func* base) : base_func(base)
            { }
            T_func* top()
            {
                return funcs.empty() ? base_func : funcs[funcs.size() - 1];
            }
            T_func* next(T_func* cur)
            {
                if (funcs.size())
                {
                    auto it = std::find(funcs.begin(), funcs.end(), cur);
                    if (it != funcs.end() && it != funcs.begin())
                        return *(it - 1);
                }
                return base_func;
            }

            class Callback {
                T_hook *hook;
                T_func *func;
                bool enabled;
            public:
                Callback(T_hook *hook, T_func *func) : hook(hook), func(func)
                { }
                ~Callback()
                {
                    disable();
                }
                inline T_func *next() { return hook->next(func); }
                void apply (bool enable)
                {
                    if (enable)
                        hook->add(func);
                    else
                        hook->remove(func);
                    enabled = enable;
                }
                inline void enable() { apply(true); }
                inline void disable() { apply(false); }
                inline bool is_enabled() { return enabled; }
                inline void toggle() { apply(!enabled); }
            };
        };
    }
}