/* ** graphics-binding.h ** ** This file is part of mkxp. ** ** Copyright (C) 2013 - 2021 Amaryllis 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 MKXPZ_SANDBOX_GRAPHICS_BINDING_H #define MKXPZ_SANDBOX_GRAPHICS_BINDING_H #include "sandbox.h" #include "binding-util.h" #include "bitmap-binding.h" #include "sharedstate.h" #include "graphics.h" #include "bitmap.h" namespace mkxp_sandbox { static VALUE graphics_module; SANDBOX_COROUTINE(graphics_binding_init, static VALUE delta(VALUE self) { SANDBOX_COROUTINE(coro, VALUE value; VALUE operator()(VALUE self) { BOOST_ASIO_CORO_REENTER (this) { GFX_LOCK; SANDBOX_AWAIT_AND_SET(value, rb_ll2inum, shState->graphics().getDelta()); GFX_UNLOCK; } return value; } ) return sb()->bind()()(self); } static VALUE update(VALUE self) { SANDBOX_COROUTINE(coro, VALUE operator()(VALUE self) { BOOST_ASIO_CORO_REENTER (this) { if (shState->graphics().update()) { SANDBOX_YIELD; } } return SANDBOX_NIL; } ) return sb()->bind()()(self); } static VALUE freeze(VALUE self) { shState->graphics().freeze(); return SANDBOX_NIL; } static VALUE transition(int32_t argc, wasm_ptr_t argv, VALUE self) { SANDBOX_COROUTINE(coro, Bitmap *trans_map; wasm_ptr_t str; int32_t duration; int32_t vague; int32_t i; VALUE operator()(int32_t argc, wasm_ptr_t argv, VALUE self) { BOOST_ASIO_CORO_REENTER (this) { trans_map = NULL; duration = 8; vague = 40; sb().transitioning = true; if (!shState->graphics().frozen()) { return SANDBOX_NIL; } if (argc >= 1) { SANDBOX_AWAIT_AND_SET(duration, rb_num2int, ((VALUE *)(**sb() + argv))[0]); if (argc >= 2) { SANDBOX_AWAIT_AND_SET(str, rb_string_value_cstr, &((VALUE *)(**sb() + argv))[1]); if (*(const char *)(**sb() + str)) { trans_map = new Bitmap((const char *)(**sb() + str)); } if (argc >= 3) { SANDBOX_AWAIT_AND_SET(vague, rb_num2int, ((VALUE *)(**sb() + argv))[2]); } } } if (duration <= 0) { shState->graphics().transition(duration, trans_map, vague, 0, 0); } else { for (i = 0; i < duration; ++i) { shState->graphics().transition(duration, trans_map, vague, i, i); SANDBOX_YIELD; } } } return SANDBOX_NIL; } ~coro() { sb().transitioning = false; if (trans_map != NULL) { delete trans_map; } } ) return sb()->bind()()(argc, argv, self); } static VALUE frame_reset(VALUE self) { GFX_LOCK; shState->graphics().frameReset(); GFX_UNLOCK; return SANDBOX_NIL; } static VALUE screenshot(VALUE self, VALUE value) { SANDBOX_COROUTINE(coro, wasm_ptr_t str; VALUE operator()(VALUE self, VALUE value) { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(str, rb_string_value_cstr, &value); GFX_GUARD_EXC(shState->graphics().screenshot((const char *)(**sb() + str));); } return SANDBOX_NIL; } ) return sb()->bind()()(self, value); } static VALUE reset(VALUE self) { GFX_GUARD_EXC(shState->graphics().reset();); return SANDBOX_NIL; } static VALUE get_frame_count(VALUE self) { return sb()->bind()()(shState->graphics().getFrameCount()); } static VALUE set_frame_count(VALUE self, VALUE value) { SANDBOX_COROUTINE(coro, int frame_count; VALUE operator()(VALUE self, VALUE value) { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(frame_count, rb_num2int, value); GFX_LOCK; shState->graphics().setFrameCount(frame_count); GFX_UNLOCK; } return value; } ) return sb()->bind()()(self, value); } static VALUE get_frame_rate(VALUE self) { return sb()->bind()()(shState->graphics().getFrameRate()); } static VALUE set_frame_rate(VALUE self, VALUE value) { SANDBOX_COROUTINE(coro, int frame_rate; VALUE operator()(VALUE self, VALUE value) { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(frame_rate, rb_num2int, value); GFX_LOCK; shState->graphics().setFrameRate(frame_rate); GFX_UNLOCK; } return value; } ) return sb()->bind()()(self, value); } static VALUE average_frame_rate(VALUE self) { return sb()->bind()()(shState->graphics().averageFrameRate()); } static VALUE width(VALUE self) { return sb()->bind()()(shState->graphics().width()); } static VALUE height(VALUE self) { return sb()->bind()()(shState->graphics().height()); } static VALUE display_width(VALUE self) { return sb()->bind()()(shState->graphics().displayWidth()); } static VALUE display_height(VALUE self) { return sb()->bind()()(shState->graphics().displayHeight()); } static VALUE wait(VALUE self, VALUE value) { SANDBOX_COROUTINE(coro, int32_t duration; int32_t i; VALUE operator()(VALUE self, VALUE value) { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(duration, rb_num2int, value); for (i = 0; i < duration; ++i) { shState->graphics().wait(duration, i, i); SANDBOX_YIELD; } } return SANDBOX_NIL; } ) return sb()->bind()()(self, value); } static VALUE fadeout(VALUE self, VALUE value) { SANDBOX_COROUTINE(coro, int32_t duration; int32_t i; int32_t brightness; VALUE operator()(VALUE self, VALUE value) { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(duration, rb_num2int, value); brightness = shState->graphics().getBrightness(); for (i = 0; i < duration; ++i) { shState->graphics().fadeout(duration, i, i, brightness); SANDBOX_YIELD; } } return SANDBOX_NIL; } ) return sb()->bind()()(self, value); } static VALUE fadein(VALUE self, VALUE value) { SANDBOX_COROUTINE(coro, int32_t duration; int32_t i; int32_t brightness; VALUE operator()(VALUE self, VALUE value) { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(duration, rb_num2int, value); brightness = shState->graphics().getBrightness(); for (i = 0; i < duration; ++i) { shState->graphics().fadein(duration, i, i, brightness); SANDBOX_YIELD; } } return SANDBOX_NIL; } ) return sb()->bind()()(self, value); } static VALUE snap_to_bitmap(VALUE self) { SANDBOX_COROUTINE(coro, Bitmap *bitmap; VALUE obj; VALUE operator()(VALUE self) { BOOST_ASIO_CORO_REENTER (this) { GFX_GUARD_EXC(bitmap = shState->graphics().snapToBitmap();); SANDBOX_AWAIT_AND_SET(obj, rb_obj_alloc, bitmap_class); set_private_data(obj, bitmap); SANDBOX_AWAIT(bitmap_init_props, bitmap, obj); } return obj; } ) return sb()->bind()()(self); } static VALUE resize_screen(VALUE self, VALUE width, VALUE height) { SANDBOX_COROUTINE(coro, int32_t w; int32_t h; VALUE operator()(VALUE self, VALUE width, VALUE height) { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(w, rb_num2int, width); SANDBOX_AWAIT_AND_SET(h, rb_num2int, height); GFX_LOCK; shState->graphics().resizeScreen(w, h); GFX_UNLOCK; } return SANDBOX_NIL; } ) return sb()->bind()()(self, width, height); } static VALUE resize_window(int32_t argc, wasm_ptr_t argv, VALUE self) { SANDBOX_COROUTINE(coro, int32_t w; int32_t h; VALUE operator()(int32_t argc, wasm_ptr_t argv, VALUE self) { BOOST_ASIO_CORO_REENTER (this) { // TODO: require at least 2 arguments SANDBOX_AWAIT_AND_SET(w, rb_num2int, ((VALUE *)(**sb() + argv))[0]); SANDBOX_AWAIT_AND_SET(h, rb_num2int, ((VALUE *)(**sb() + argv))[1]); GFX_LOCK; if (argc >= 3) { shState->graphics().resizeWindow(w, h, SANDBOX_VALUE_TO_BOOL(((VALUE *)(**sb() + argv))[2])); } else { shState->graphics().resizeWindow(w, h); } GFX_UNLOCK; } return SANDBOX_NIL; } ) return sb()->bind()()(argc, argv, self); } static VALUE center(VALUE self) { shState->graphics().center(); return SANDBOX_NIL; } static VALUE get_brightness(VALUE self) { return sb()->bind()()(shState->graphics().getBrightness()); } static VALUE set_brightness(VALUE self, VALUE value) { SANDBOX_COROUTINE(coro, int32_t brightness; VALUE operator()(VALUE self, VALUE value) { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(brightness, rb_num2int, value); GFX_LOCK; shState->graphics().setBrightness(brightness); GFX_UNLOCK; } return value; } ) return sb()->bind()()(self, value); } static VALUE play_movie(int32_t argc, wasm_ptr_t argv, VALUE self) { SANDBOX_COROUTINE(coro, wasm_ptr_t str; int32_t volume; VALUE operator()(int32_t argc, wasm_ptr_t argv, VALUE self) { BOOST_ASIO_CORO_REENTER (this) { // TODO: require at least 1 argument SANDBOX_AWAIT_AND_SET(str, rb_string_value_cstr, &((VALUE *)(**sb() + argv))[0]); if (argc >= 2) { SANDBOX_AWAIT_AND_SET(volume, rb_num2int, ((VALUE *)(**sb() + argv))[1]); } else { volume = 100; } bool skippable = argc >= 3 ? SANDBOX_VALUE_TO_BOOL(((VALUE *)(**sb() + argv))[2]) : false; // TODO: use SANDBOX_YIELD as required GFX_GUARD_EXC(shState->graphics().playMovie((const char *)(**sb() + str), volume, skippable)); } return SANDBOX_NIL; } ) return sb()->bind()()(argc, argv, self); } static VALUE get_fullscreen(VALUE self) { return SANDBOX_BOOL_TO_VALUE(shState->graphics().getFullscreen()); } static VALUE set_fullscreen(VALUE self, VALUE value) { GFX_LOCK; shState->graphics().setFullscreen(SANDBOX_VALUE_TO_BOOL(value)); GFX_UNLOCK; return value; } static VALUE get_show_cursor(VALUE self) { return SANDBOX_BOOL_TO_VALUE(shState->graphics().getShowCursor()); } static VALUE set_show_cursor(VALUE self, VALUE value) { GFX_LOCK; shState->graphics().setShowCursor(SANDBOX_VALUE_TO_BOOL(value)); GFX_UNLOCK; return value; } static VALUE get_scale(VALUE self) { return sb()->bind()()(shState->graphics().getScale()); } static VALUE set_scale(VALUE self, VALUE value) { SANDBOX_COROUTINE(coro, double scale; VALUE operator()(VALUE self, VALUE value) { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(scale, rb_num2dbl, value); GFX_LOCK; shState->graphics().setScale(scale); GFX_UNLOCK; } return value; } ) return sb()->bind()()(self, value); } static VALUE get_frameskip(VALUE self) { return SANDBOX_BOOL_TO_VALUE(shState->graphics().getFrameskip()); } static VALUE set_frameskip(VALUE self, VALUE value) { GFX_LOCK; shState->graphics().setFrameskip(SANDBOX_VALUE_TO_BOOL(value)); GFX_UNLOCK; return value; } static VALUE get_fixed_aspect_ratio(VALUE self) { return SANDBOX_BOOL_TO_VALUE(shState->graphics().getFixedAspectRatio()); } static VALUE set_fixed_aspect_ratio(VALUE self, VALUE value) { GFX_LOCK; shState->graphics().setFixedAspectRatio(SANDBOX_VALUE_TO_BOOL(value)); GFX_UNLOCK; return value; } static VALUE get_smooth_scaling(VALUE self) { return sb()->bind()()(shState->graphics().getSmoothScaling()); } static VALUE set_smooth_scaling(VALUE self, VALUE value) { SANDBOX_COROUTINE(coro, int32_t smooth_scaling; VALUE operator()(VALUE self, VALUE value) { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(smooth_scaling, rb_num2int, value); GFX_LOCK; shState->graphics().setSmoothScaling(smooth_scaling); GFX_UNLOCK; } return value; } ) return sb()->bind()()(self, value); } static VALUE get_integer_scaling(VALUE self) { return SANDBOX_BOOL_TO_VALUE(shState->graphics().getIntegerScaling()); } static VALUE set_integer_scaling(VALUE self, VALUE value) { GFX_LOCK; shState->graphics().setIntegerScaling(SANDBOX_VALUE_TO_BOOL(value)); GFX_UNLOCK; return value; } static VALUE get_last_mile_scaling(VALUE self) { return SANDBOX_BOOL_TO_VALUE(shState->graphics().getLastMileScaling()); } static VALUE set_last_mile_scaling(VALUE self, VALUE value) { GFX_LOCK; shState->graphics().setLastMileScaling(SANDBOX_VALUE_TO_BOOL(value)); GFX_UNLOCK; return value; } static VALUE get_threadsafe(VALUE self) { return SANDBOX_BOOL_TO_VALUE(shState->graphics().getThreadsafe()); } static VALUE set_threadsafe(VALUE self, VALUE value) { GFX_LOCK; shState->graphics().setThreadsafe(SANDBOX_VALUE_TO_BOOL(value)); GFX_UNLOCK; return value; } void operator()() { BOOST_ASIO_CORO_REENTER (this) { SANDBOX_AWAIT_AND_SET(graphics_module, rb_define_module, "Graphics"); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "delta", (VALUE (*)(ANYARGS))delta, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "update", (VALUE (*)(ANYARGS))update, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "freeze", (VALUE (*)(ANYARGS))freeze, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "transition", (VALUE (*)(ANYARGS))transition, -1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "frame_reset", (VALUE (*)(ANYARGS))frame_reset, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "screenshot", (VALUE (*)(ANYARGS))screenshot, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "__reset__", (VALUE (*)(ANYARGS))reset, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "frame_count", (VALUE (*)(ANYARGS))get_frame_count, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "frame_count=", (VALUE (*)(ANYARGS))set_frame_count, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "frame_rate", (VALUE (*)(ANYARGS))get_frame_rate, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "frame_rate=", (VALUE (*)(ANYARGS))set_frame_rate, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "average_frame_rate", (VALUE (*)(ANYARGS))average_frame_rate, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "width", (VALUE (*)(ANYARGS))width, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "height", (VALUE (*)(ANYARGS))height, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "display_width", (VALUE (*)(ANYARGS))display_width, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "display_height", (VALUE (*)(ANYARGS))display_height, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "wait", (VALUE (*)(ANYARGS))wait, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "fadeout", (VALUE (*)(ANYARGS))fadeout, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "fadein", (VALUE (*)(ANYARGS))fadein, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "snap_to_bitmap", (VALUE (*)(ANYARGS))snap_to_bitmap, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "resize_screen", (VALUE (*)(ANYARGS))resize_screen, 2); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "resize_window", (VALUE (*)(ANYARGS))resize_window, -1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "center", (VALUE (*)(ANYARGS))center, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "brightness", (VALUE (*)(ANYARGS))get_brightness, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "brightness=", (VALUE (*)(ANYARGS))set_brightness, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "play_movie", (VALUE (*)(ANYARGS))play_movie, -1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "fullscreen", (VALUE (*)(ANYARGS))get_fullscreen, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "fullscreen=", (VALUE (*)(ANYARGS))set_fullscreen, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "show_cursor", (VALUE (*)(ANYARGS))get_show_cursor, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "show_cursor=", (VALUE (*)(ANYARGS))set_show_cursor, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "scale", (VALUE (*)(ANYARGS))get_scale, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "scale=", (VALUE (*)(ANYARGS))set_scale, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "frameskip", (VALUE (*)(ANYARGS))get_frameskip, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "frameskip=", (VALUE (*)(ANYARGS))set_frameskip, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "fixed_aspect_ratio", (VALUE (*)(ANYARGS))get_fixed_aspect_ratio, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "fixed_aspect_ratio=", (VALUE (*)(ANYARGS))set_fixed_aspect_ratio, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "smooth_scaling", (VALUE (*)(ANYARGS))get_smooth_scaling, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "smooth_scaling=", (VALUE (*)(ANYARGS))set_smooth_scaling, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "integer_scaling", (VALUE (*)(ANYARGS))get_integer_scaling, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "integer_scaling=", (VALUE (*)(ANYARGS))set_integer_scaling, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "last_mile_scaling", (VALUE (*)(ANYARGS))get_last_mile_scaling, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "last_mile_scaling=", (VALUE (*)(ANYARGS))set_last_mile_scaling, 1); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "threadsafe", (VALUE (*)(ANYARGS))get_threadsafe, 0); SANDBOX_AWAIT(rb_define_module_function, graphics_module, "threadsafe=", (VALUE (*)(ANYARGS))set_threadsafe, 1); } } ) } #endif // MKXPZ_SANDBOX_GRAPHICS_BINDING_H