Add initial support for hooking into Gui-related functions
parent
928bcb6d95
commit
fff9072b07
@ -0,0 +1,67 @@
|
||||
#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_func *func;
|
||||
T_hook *hook;
|
||||
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);
|
||||
}
|
||||
inline void enable() { apply(true); }
|
||||
inline void disable() { apply(false); }
|
||||
};
|
||||
};
|
||||
}
|
||||
}
|
@ -0,0 +1,96 @@
|
||||
#include "Console.h"
|
||||
#include "Core.h"
|
||||
#include "DataDefs.h"
|
||||
#include "Export.h"
|
||||
#include "PluginManager.h"
|
||||
|
||||
#include "modules/Screen.h"
|
||||
|
||||
using namespace DFHack;
|
||||
|
||||
DFHACK_PLUGIN("color-dfhack-text");
|
||||
DFHACK_PLUGIN_IS_ENABLED(is_enabled);
|
||||
|
||||
struct {
|
||||
bool flicker;
|
||||
uint8_t color;
|
||||
short tick;
|
||||
} config;
|
||||
|
||||
void color_text_tile(const Screen::Pen &pen, int x, int y, bool map);
|
||||
GUI_HOOK_CALLBACK(Screen::Hooks::set_tile, color_text_hook, color_text_tile);
|
||||
void color_text_tile(const Screen::Pen &pen, int x, int y, bool map)
|
||||
{
|
||||
Screen::Pen pen2 = pen;
|
||||
uint8_t color = config.flicker ? config.tick % 8 : config.color;
|
||||
pen2.fg = color;
|
||||
pen2.bg = color;
|
||||
pen2.bold = true;
|
||||
return color_text_hook.next()(pen2, x, y, map);
|
||||
}
|
||||
|
||||
DFhackCExport command_result plugin_enable (color_ostream &out, bool enable)
|
||||
{
|
||||
color_text_hook.apply(enable);
|
||||
is_enabled = enable;
|
||||
return CR_OK;
|
||||
}
|
||||
|
||||
command_result color(color_ostream &out, std::vector<std::string> ¶ms)
|
||||
{
|
||||
plugin_enable(out, true);
|
||||
std::string p0 = toLower(params[0]);
|
||||
#define CHECK_COLOR(color_name) else if (p0 == toLower(std::string(#color_name))) \
|
||||
{ config.flicker = false; config.color = COLOR_##color_name % 8; }
|
||||
if (params.empty())
|
||||
return CR_OK;
|
||||
CHECK_COLOR(RED)
|
||||
CHECK_COLOR(GREEN)
|
||||
CHECK_COLOR(BLUE)
|
||||
CHECK_COLOR(YELLOW)
|
||||
CHECK_COLOR(BROWN)
|
||||
CHECK_COLOR(CYAN)
|
||||
CHECK_COLOR(MAGENTA)
|
||||
CHECK_COLOR(WHITE)
|
||||
CHECK_COLOR(GREY)
|
||||
CHECK_COLOR(BLACK)
|
||||
#undef CHECK_COLOR
|
||||
else if (p0 == "flicker")
|
||||
{
|
||||
config.flicker = true;
|
||||
}
|
||||
else if (p0 == "disable")
|
||||
{
|
||||
plugin_enable(out, false);
|
||||
}
|
||||
else if (p0 != "enable")
|
||||
{
|
||||
out.printerr("Unrecognized option: %s\n", p0.c_str());
|
||||
return CR_WRONG_USAGE;
|
||||
}
|
||||
return CR_OK;
|
||||
}
|
||||
|
||||
DFhackCExport command_result plugin_init (color_ostream &out, std::vector <PluginCommand> &commands)
|
||||
{
|
||||
commands.push_back(PluginCommand(
|
||||
"color-dfhack-text",
|
||||
"color <color>|flicker|enable|disable",
|
||||
color,
|
||||
false
|
||||
));
|
||||
config.flicker = false;
|
||||
config.color = COLOR_RED;
|
||||
return CR_OK;
|
||||
}
|
||||
|
||||
DFhackCExport command_result plugin_shutdown (color_ostream &out)
|
||||
{
|
||||
return CR_OK;
|
||||
}
|
||||
|
||||
DFhackCExport command_result plugin_onupdate (color_ostream &out)
|
||||
{
|
||||
++config.tick;
|
||||
return CR_OK;
|
||||
}
|
Loading…
Reference in New Issue