mirror of
https://github.com/mkxp-z/mkxp-z.git
synced 2025-04-21 21:52:04 +02:00

In RGSS, uninitialized disposable objects (and Fonts, sort of) are technically disposed, and other objects (Tone, Color, Rect, and Table) are created with all values set to 0. It's also possible to reinitialize them, although reinitializing disposables leaks memory. This commit causes MiniFFI and FileInt objects to improperly raise disposed errors if used while uninitialized, but that feels acceptable to me.
311 lines
8.2 KiB
C++
311 lines
8.2 KiB
C++
/*
|
|
** font-binding.cpp
|
|
**
|
|
** This file is part of mkxp.
|
|
**
|
|
** Copyright (C) 2013 - 2021 Amaryllis Kulla <ancurio@mapleshrine.eu>
|
|
**
|
|
** mkxp is free software: you can redistribute it and/or modify
|
|
** it under the terms of the GNU General Public License as published by
|
|
** the Free Software Foundation, either version 2 of the License, or
|
|
** (at your option) any later version.
|
|
**
|
|
** mkxp is distributed in the hope that it will be useful,
|
|
** but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
** GNU General Public License for more details.
|
|
**
|
|
** You should have received a copy of the GNU General Public License
|
|
** along with mkxp. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "binding-types.h"
|
|
#include "binding-util.h"
|
|
#include "exception.h"
|
|
#include "font.h"
|
|
#include "sharedstate.h"
|
|
|
|
#include <string.h>
|
|
|
|
static void collectStrings(VALUE obj, std::vector<std::string> &out) {
|
|
if (RB_TYPE_P(obj, RUBY_T_STRING)) {
|
|
out.push_back(RSTRING_PTR(obj));
|
|
} else if (RB_TYPE_P(obj, RUBY_T_ARRAY)) {
|
|
for (long i = 0; i < RARRAY_LEN(obj); ++i) {
|
|
VALUE str = rb_ary_entry(obj, i);
|
|
|
|
/* Non-string objects are tolerated (ignored) */
|
|
if (!RB_TYPE_P(str, RUBY_T_STRING))
|
|
continue;
|
|
|
|
out.push_back(RSTRING_PTR(str));
|
|
}
|
|
}
|
|
}
|
|
|
|
#if RAPI_FULL > 187
|
|
DEF_TYPE(Font);
|
|
#else
|
|
DEF_ALLOCFUNC(Font);
|
|
#endif
|
|
|
|
RB_METHOD(fontDoesExist) {
|
|
RB_UNUSED_PARAM;
|
|
|
|
const char *name = 0;
|
|
VALUE nameObj;
|
|
|
|
rb_get_args(argc, argv, "o", &nameObj RB_ARG_END);
|
|
|
|
if (RB_TYPE_P(nameObj, RUBY_T_STRING))
|
|
name = rb_string_value_cstr(&nameObj);
|
|
|
|
return rb_bool_new(Font::doesExist(name));
|
|
}
|
|
|
|
RB_METHOD(FontSetName);
|
|
|
|
RB_METHOD(fontInitialize) {
|
|
VALUE namesObj = Qnil;
|
|
int size = 0;
|
|
|
|
rb_get_args(argc, argv, "|oi", &namesObj, &size RB_ARG_END);
|
|
|
|
Font *f;
|
|
|
|
if (NIL_P(namesObj)) {
|
|
namesObj = rb_iv_get(rb_obj_class(self), "default_name");
|
|
f = new Font(0, size);
|
|
} else {
|
|
std::vector<std::string> names;
|
|
collectStrings(namesObj, names);
|
|
|
|
f = new Font(&names, size);
|
|
}
|
|
|
|
/* This is semantically wrong; the new Font object should take
|
|
* a dup'ed object here in case of an array. Ditto for the setters.
|
|
* However the same bug/behavior exists in all RM versions. */
|
|
rb_iv_set(self, "name", namesObj);
|
|
|
|
Font *orig = getPrivateDataNoRaise<Font>(self);
|
|
if (orig)
|
|
{
|
|
*orig = *f;
|
|
delete f;
|
|
f = orig;
|
|
} else {
|
|
setPrivateData(self, f);
|
|
}
|
|
|
|
/* Wrap property objects */
|
|
f->initDynAttribs();
|
|
|
|
wrapProperty(self, &f->getColor(), "color", ColorType);
|
|
|
|
if (rgssVer >= 3)
|
|
wrapProperty(self, &f->getOutColor(), "out_color", ColorType);
|
|
|
|
return self;
|
|
}
|
|
|
|
RB_METHOD(fontInitializeCopy) {
|
|
VALUE origObj;
|
|
rb_get_args(argc, argv, "o", &origObj RB_ARG_END);
|
|
|
|
if (!OBJ_INIT_COPY(self, origObj))
|
|
return self;
|
|
|
|
Font *orig = getPrivateData<Font>(origObj);
|
|
Font *f = new Font(*orig);
|
|
setPrivateData(self, f);
|
|
|
|
/* Wrap property objects */
|
|
f->initDynAttribs();
|
|
|
|
wrapProperty(self, &f->getColor(), "color", ColorType);
|
|
|
|
if (rgssVer >= 3)
|
|
wrapProperty(self, &f->getOutColor(), "out_color", ColorType);
|
|
|
|
return self;
|
|
}
|
|
|
|
RB_METHOD(FontGetName) {
|
|
RB_UNUSED_PARAM;
|
|
|
|
return rb_iv_get(self, "name");
|
|
}
|
|
|
|
RB_METHOD(FontSetName) {
|
|
Font *f = getPrivateData<Font>(self);
|
|
|
|
rb_check_argc(argc, 1);
|
|
|
|
std::vector<std::string> namesObj;
|
|
collectStrings(argv[0], namesObj);
|
|
|
|
f->setName(namesObj);
|
|
rb_iv_set(self, "name", argv[0]);
|
|
|
|
return argv[0];
|
|
}
|
|
|
|
template <class C> static void checkDisposed(VALUE) {}
|
|
|
|
DEF_PROP_OBJ_VAL(Font, Color, Color, "color")
|
|
DEF_PROP_OBJ_VAL(Font, Color, OutColor, "out_color")
|
|
|
|
DEF_PROP_I(Font, Size)
|
|
|
|
DEF_PROP_B(Font, Bold)
|
|
DEF_PROP_B(Font, Italic)
|
|
DEF_PROP_B(Font, Shadow)
|
|
DEF_PROP_B(Font, Outline)
|
|
|
|
#define DEF_KLASS_PROP(Klass, type, PropName, param_t_s, value_fun) \
|
|
RB_METHOD(Klass##Get##PropName) { \
|
|
RB_UNUSED_PARAM; \
|
|
return value_fun(Klass::get##PropName()); \
|
|
} \
|
|
RB_METHOD(Klass##Set##PropName) { \
|
|
RB_UNUSED_PARAM; \
|
|
type value; \
|
|
rb_get_args(argc, argv, param_t_s, &value RB_ARG_END); \
|
|
Klass::set##PropName(value); \
|
|
return value_fun(value); \
|
|
}
|
|
|
|
DEF_KLASS_PROP(Font, int, DefaultSize, "i", rb_fix_new)
|
|
DEF_KLASS_PROP(Font, bool, DefaultBold, "b", rb_bool_new)
|
|
DEF_KLASS_PROP(Font, bool, DefaultItalic, "b", rb_bool_new)
|
|
DEF_KLASS_PROP(Font, bool, DefaultShadow, "b", rb_bool_new)
|
|
DEF_KLASS_PROP(Font, bool, DefaultOutline, "b", rb_bool_new)
|
|
|
|
RB_METHOD(FontGetDefaultOutColor) {
|
|
RB_UNUSED_PARAM;
|
|
return rb_iv_get(self, "default_out_color");
|
|
}
|
|
|
|
RB_METHOD(FontSetDefaultOutColor) {
|
|
RB_UNUSED_PARAM;
|
|
|
|
VALUE colorObj;
|
|
rb_get_args(argc, argv, "o", &colorObj RB_ARG_END);
|
|
|
|
Color *c = getPrivateDataCheck<Color>(colorObj, ColorType);
|
|
|
|
Font::setDefaultOutColor(*c);
|
|
|
|
return colorObj;
|
|
}
|
|
|
|
RB_METHOD(FontGetDefaultName) {
|
|
RB_UNUSED_PARAM;
|
|
|
|
return rb_iv_get(self, "default_name");
|
|
}
|
|
|
|
RB_METHOD(FontSetDefaultName) {
|
|
RB_UNUSED_PARAM;
|
|
|
|
rb_check_argc(argc, 1);
|
|
|
|
std::vector<std::string> namesObj;
|
|
collectStrings(argv[0], namesObj);
|
|
|
|
Font::setDefaultName(namesObj, shState->fontState());
|
|
rb_iv_set(self, "default_name", argv[0]);
|
|
|
|
return argv[0];
|
|
}
|
|
|
|
RB_METHOD(FontGetDefaultColor) {
|
|
RB_UNUSED_PARAM;
|
|
return rb_iv_get(self, "default_color");
|
|
}
|
|
|
|
RB_METHOD(FontSetDefaultColor) {
|
|
RB_UNUSED_PARAM;
|
|
|
|
VALUE colorObj;
|
|
rb_get_args(argc, argv, "o", &colorObj RB_ARG_END);
|
|
|
|
Color *c = getPrivateDataCheck<Color>(colorObj, ColorType);
|
|
|
|
Font::setDefaultColor(*c);
|
|
|
|
return colorObj;
|
|
}
|
|
|
|
#define INIT_KLASS_PROP_BIND(Klass, PropName, prop_name_s) \
|
|
{ \
|
|
rb_define_class_method(klass, prop_name_s, Klass##Get##PropName); \
|
|
rb_define_class_method(klass, prop_name_s "=", Klass##Set##PropName); \
|
|
}
|
|
|
|
void fontBindingInit() {
|
|
VALUE klass = rb_define_class("Font", rb_cObject);
|
|
#if RAPI_FULL > 187
|
|
rb_define_alloc_func(klass, classAllocate<&FontType>);
|
|
#else
|
|
rb_define_alloc_func(klass, FontAllocate);
|
|
#endif
|
|
|
|
Font::initDefaultDynAttribs();
|
|
wrapProperty(klass, &Font::getDefaultColor(), "default_color", ColorType);
|
|
|
|
/* Initialize default names */
|
|
const std::vector<std::string> &defNames = Font::getInitialDefaultNames();
|
|
VALUE defNamesObj;
|
|
|
|
if (defNames.size() == 1) {
|
|
defNamesObj = rb_utf8_str_new_cstr(defNames[0].c_str());
|
|
} else {
|
|
defNamesObj = rb_ary_new2(defNames.size());
|
|
|
|
for (size_t i = 0; i < defNames.size(); ++i)
|
|
rb_ary_push(defNamesObj, rb_utf8_str_new_cstr(defNames[i].c_str()));
|
|
}
|
|
|
|
rb_iv_set(klass, "default_name", defNamesObj);
|
|
|
|
if (rgssVer >= 3)
|
|
wrapProperty(klass, &Font::getDefaultOutColor(), "default_out_color",
|
|
ColorType);
|
|
|
|
INIT_KLASS_PROP_BIND(Font, DefaultName, "default_name");
|
|
INIT_KLASS_PROP_BIND(Font, DefaultSize, "default_size");
|
|
INIT_KLASS_PROP_BIND(Font, DefaultBold, "default_bold");
|
|
INIT_KLASS_PROP_BIND(Font, DefaultItalic, "default_italic");
|
|
INIT_KLASS_PROP_BIND(Font, DefaultColor, "default_color");
|
|
|
|
if (rgssVer >= 2) {
|
|
INIT_KLASS_PROP_BIND(Font, DefaultShadow, "default_shadow");
|
|
}
|
|
|
|
if (rgssVer >= 3) {
|
|
INIT_KLASS_PROP_BIND(Font, DefaultOutline, "default_outline");
|
|
INIT_KLASS_PROP_BIND(Font, DefaultOutColor, "default_out_color");
|
|
}
|
|
|
|
rb_define_class_method(klass, "exist?", fontDoesExist);
|
|
|
|
_rb_define_method(klass, "initialize", fontInitialize);
|
|
_rb_define_method(klass, "initialize_copy", fontInitializeCopy);
|
|
|
|
INIT_PROP_BIND(Font, Name, "name");
|
|
INIT_PROP_BIND(Font, Size, "size");
|
|
INIT_PROP_BIND(Font, Bold, "bold");
|
|
INIT_PROP_BIND(Font, Italic, "italic");
|
|
INIT_PROP_BIND(Font, Color, "color");
|
|
|
|
if (rgssVer >= 2) {
|
|
INIT_PROP_BIND(Font, Shadow, "shadow");
|
|
}
|
|
|
|
if (rgssVer >= 3) {
|
|
INIT_PROP_BIND(Font, Outline, "outline");
|
|
INIT_PROP_BIND(Font, OutColor, "out_color");
|
|
}
|
|
}
|