2013-02-01 21:52:27 +02:00
|
|
|
#include "lua/internal.hpp"
|
|
|
|
#include "fonts/wrapper.hpp"
|
2013-02-28 10:36:57 +02:00
|
|
|
#include "core/framebuffer.hpp"
|
2013-02-01 21:52:27 +02:00
|
|
|
#include "library/framebuffer.hpp"
|
|
|
|
#include "library/customfont.hpp"
|
|
|
|
#include "library/utf8.hpp"
|
2013-03-13 01:02:10 +02:00
|
|
|
#include <algorithm>
|
2013-02-01 21:52:27 +02:00
|
|
|
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
struct lua_customfont
|
|
|
|
{
|
|
|
|
public:
|
2013-02-01 22:26:22 +02:00
|
|
|
lua_customfont(lua_state* L, const std::string& filename);
|
2013-02-01 21:52:27 +02:00
|
|
|
~lua_customfont() throw();
|
2013-02-01 22:26:22 +02:00
|
|
|
int draw(lua_state& L);
|
2013-02-01 21:52:27 +02:00
|
|
|
const custom_font& get_font() { return font; }
|
|
|
|
private:
|
|
|
|
custom_font font;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
DECLARE_LUACLASS(lua_customfont, "CUSTOMFONT");
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
struct render_object_text_cf : public render_object
|
|
|
|
{
|
|
|
|
render_object_text_cf(int32_t _x, int32_t _y, const std::string& _text, premultiplied_color _fg,
|
|
|
|
premultiplied_color _bg, lua_obj_pin<lua_customfont>* _font) throw()
|
|
|
|
: x(_x), y(_y), text(_text), fg(_fg), bg(_bg), font(_font) {}
|
2013-02-28 10:36:57 +02:00
|
|
|
~render_object_text_cf() throw()
|
|
|
|
{
|
|
|
|
delete font;
|
|
|
|
}
|
2013-02-01 21:52:27 +02:00
|
|
|
template<bool X> void op(struct framebuffer<X>& scr) throw()
|
|
|
|
{
|
|
|
|
fg.set_palette(scr);
|
|
|
|
bg.set_palette(scr);
|
|
|
|
const custom_font& fdata = font->object()->get_font();
|
2013-03-13 01:02:10 +02:00
|
|
|
std::vector<uint32_t> _text;
|
|
|
|
copy_from_utf8(text.begin(), text.end(), std::back_inserter(_text));
|
2013-02-01 21:52:27 +02:00
|
|
|
int32_t orig_x = x;
|
|
|
|
int32_t drawx = x;
|
|
|
|
int32_t drawy = y;
|
|
|
|
for(size_t i = 0; i < _text.size();) {
|
|
|
|
uint32_t cp = _text[i];
|
|
|
|
ligature_key k = fdata.best_ligature_match(_text, i);
|
|
|
|
const font_glyph_data& glyph = fdata.lookup_glyph(k);
|
|
|
|
if(k.length())
|
|
|
|
i += k.length();
|
|
|
|
else
|
|
|
|
i++;
|
|
|
|
if(cp == 9) {
|
|
|
|
drawx = (drawx + 64) >> 6 << 6;
|
|
|
|
} else if(cp == 10) {
|
|
|
|
drawx = orig_x;
|
|
|
|
drawy += fdata.get_rowadvance();
|
|
|
|
} else {
|
|
|
|
glyph.render(scr, drawx, drawy, fg, bg);
|
|
|
|
drawx += glyph.width;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-02-28 10:36:57 +02:00
|
|
|
bool kill_request(void* obj) throw()
|
|
|
|
{
|
|
|
|
return kill_request_ifeq(unbox_any_pin(font), obj);
|
|
|
|
}
|
2013-02-01 21:52:27 +02:00
|
|
|
void operator()(struct framebuffer<true>& scr) throw() { op(scr); }
|
|
|
|
void operator()(struct framebuffer<false>& scr) throw() { op(scr); }
|
2013-02-01 22:26:22 +02:00
|
|
|
void clone(render_queue& q) const throw(std::bad_alloc) { q.clone_helper(this); }
|
2013-02-01 21:52:27 +02:00
|
|
|
private:
|
|
|
|
int32_t x;
|
|
|
|
int32_t y;
|
2013-02-10 15:00:12 +02:00
|
|
|
std::string text;
|
2013-02-01 21:52:27 +02:00
|
|
|
premultiplied_color fg;
|
|
|
|
premultiplied_color bg;
|
|
|
|
lua_obj_pin<lua_customfont>* font;
|
|
|
|
};
|
|
|
|
|
2013-02-01 22:26:22 +02:00
|
|
|
lua_customfont::lua_customfont(lua_state* L, const std::string& filename)
|
2013-02-01 21:52:27 +02:00
|
|
|
: font(filename)
|
|
|
|
{
|
2013-02-01 22:10:01 +02:00
|
|
|
static char done_key;
|
2013-02-01 22:26:22 +02:00
|
|
|
if(L->do_once(&done_key)) {
|
|
|
|
objclass<lua_customfont>().bind(*L, "__call", &lua_customfont::draw);
|
2013-02-01 22:10:01 +02:00
|
|
|
}
|
2013-02-01 21:52:27 +02:00
|
|
|
}
|
|
|
|
|
2013-02-28 10:36:57 +02:00
|
|
|
lua_customfont::~lua_customfont() throw()
|
|
|
|
{
|
|
|
|
render_kill_request(this);
|
|
|
|
}
|
2013-02-01 21:52:27 +02:00
|
|
|
|
2013-02-01 22:26:22 +02:00
|
|
|
int lua_customfont::draw(lua_state& L)
|
2013-02-01 21:52:27 +02:00
|
|
|
{
|
|
|
|
std::string fname = "CUSTOMFONT::__call";
|
|
|
|
if(!lua_render_ctx)
|
|
|
|
return 0;
|
|
|
|
int64_t fgc = 0xFFFFFFU;
|
|
|
|
int64_t bgc = -1;
|
2013-02-01 22:26:22 +02:00
|
|
|
int32_t _x = L.get_numeric_argument<int32_t>(2, fname.c_str());
|
|
|
|
int32_t _y = L.get_numeric_argument<int32_t>(3, fname.c_str());
|
|
|
|
L.get_numeric_argument<int64_t>(5, fgc, fname.c_str());
|
|
|
|
L.get_numeric_argument<int64_t>(6, bgc, fname.c_str());
|
|
|
|
std::string text = L.get_string(4, fname.c_str());
|
|
|
|
auto f = lua_class<lua_customfont>::pin(L, 1, fname.c_str());
|
2013-02-01 21:52:27 +02:00
|
|
|
premultiplied_color fg(fgc);
|
|
|
|
premultiplied_color bg(bgc);
|
|
|
|
lua_render_ctx->queue->create_add<render_object_text_cf>(_x, _y, text, fg, bg, f);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-02-01 22:26:22 +02:00
|
|
|
function_ptr_luafun gui_text_cf(LS, "gui.loadfont", [](lua_state& L, const std::string& fname) -> int {
|
|
|
|
std::string filename = L.get_string(1, fname.c_str());
|
2013-02-01 21:52:27 +02:00
|
|
|
try {
|
2013-02-10 15:00:12 +02:00
|
|
|
lua_class<lua_customfont>::create(L, &L, filename);
|
2013-02-01 21:52:27 +02:00
|
|
|
return 1;
|
|
|
|
} catch(std::exception& e) {
|
2013-02-01 22:26:22 +02:00
|
|
|
L.pushstring(e.what());
|
|
|
|
L.error();
|
2013-02-01 21:52:27 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|