/* ** disposable-binding.h ** ** This file is part of mkxp. ** ** Copyright (C) 2013 Jonas Kulla ** ** 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 . */ #ifndef DISPOSABLEBINDING_H #define DISPOSABLEBINDING_H #include "disposable.h" #include "binding-util.h" /* 'Children' are disposables that are disposed together * with their parent. Currently this is only used by Viewport * in RGSS1. */ inline void disposableAddChild(VALUE disp, VALUE child) { if (NIL_P(disp) || NIL_P(child)) { return; } VALUE children = rb_iv_get(disp, "children"); bool exists = false; if (NIL_P(children)) { children = rb_ary_new(); rb_iv_set(disp, "children", children); } else { exists = RTEST(rb_funcall(children, rb_intern("include?"), 1, child)); } if (!exists) rb_ary_push(children, child); } inline void disposableRemoveChild(VALUE disp, VALUE child) { if (NIL_P(disp) || NIL_P(child)) { return; } VALUE children = rb_iv_get(disp, "children"); if (NIL_P(children)) return; VALUE index = rb_funcall(children, rb_intern("index"), 1, child); if (NIL_P(index)) return; rb_funcall(children, rb_intern("delete_at"), 1, index); } inline void disposableDisposeChildren(VALUE disp) { VALUE children = rb_iv_get(disp, "children"); if (NIL_P(children)) return; ID dispFun = rb_intern("_mkxp_dispose_alias"); for (long i = 0; i < RARRAY_LEN(children); ++i) rb_funcall2(rb_ary_entry(children, i), dispFun, 0, 0); } template RB_METHOD(disposableDispose) { RB_UNUSED_PARAM; C *d = getPrivateData(self); if (!d) return Qnil; /* Nothing to do if already disposed */ if (d->isDisposed()) return Qnil; if (rgssVer == 1) disposableDisposeChildren(self); d->dispose(); return Qnil; } template RB_METHOD(disposableIsDisposed) { RB_UNUSED_PARAM; C *d = getPrivateData(self); if (!d) return Qtrue; return rb_bool_new(d->isDisposed()); } template static void disposableBindingInit(VALUE klass) { _rb_define_method(klass, "dispose", disposableDispose); _rb_define_method(klass, "disposed?", disposableIsDisposed); /* Make sure we always have access to the original method, even * if it is overridden by user scripts */ if (rgssVer == 1) rb_define_alias(klass, "_mkxp_dispose_alias", "dispose"); } template inline void checkDisposed(VALUE self) { if (disposableIsDisposed(0, 0, self) == Qtrue) raiseDisposedAccess(self); } #endif // DISPOSABLEBINDING_H