// Copyright 2021 Jean Pierre Cimalando // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // // SPDX-License-Identifier: Apache-2.0 // #include "ysfx.hpp" #include "ysfx_config.hpp" #include "ysfx_api_gfx.hpp" #include "ysfx_eel_utils.hpp" #if !defined(YSFX_NO_GFX) # include "lice_stb/lice_stb_loaders.hpp" # define WDL_NO_DEFINE_MINMAX # include "WDL/swell/swell.h" # include "WDL/lice/lice.h" # include "WDL/lice/lice_text.h" # include "WDL/wdlstring.h" #endif #include #include #include #include #include #include #if !defined(YSFX_NO_GFX) #define GFX_GET_CONTEXT(opaque) (((opaque)) ? ysfx_gfx_get_context((ysfx_t *)(opaque)) : nullptr) enum { ysfx_gfx_max_images = 1024, ysfx_gfx_max_fonts = 128, ysfx_gfx_max_input = 1024, }; class eel_lice_state; struct ysfx_gfx_state_t { ysfx_gfx_state_t(ysfx_t *fx); ~ysfx_gfx_state_t(); std::unique_ptr lice; std::queue input_queue; std::unordered_set keys_pressed; ysfx_real scale = 0.0; void *callback_data = nullptr; int (*show_menu)(void *, const char *, int32_t, int32_t) = nullptr; void (*set_cursor)(void *, int32_t) = nullptr; const char *(*get_drop_file)(void *user_data, int32_t index) = nullptr; }; #endif //------------------------------------------------------------------------------ #if !defined(YSFX_NO_GFX) static bool eel_lice_get_filename_for_string(void *opaque, EEL_F idx, WDL_FastString *fs, int iswrite) { if (iswrite) return false; // this is neither supported nor used ysfx_t *fx = (ysfx_t *)opaque; std::string filepath; if (!ysfx_find_data_file(fx, &idx, filepath)) return false; if (fs) fs->Set(filepath.data(), (uint32_t)filepath.size()); return true; } #define EEL_LICE_GET_FILENAME_FOR_STRING(idx, fs, p) \ eel_lice_get_filename_for_string(opaque, (idx), (fs), (p)) #endif //------------------------------------------------------------------------------ #if !defined(YSFX_NO_GFX) # include "ysfx_api_gfx_lice.hpp" #else # include "ysfx_api_gfx_dummy.hpp" #endif //------------------------------------------------------------------------------ #if !defined(YSFX_NO_GFX) static bool translate_special_key(uint32_t uni_key, uint32_t &jsfx_key) { auto key_c = [](uint8_t a, uint8_t b, uint8_t c, uint8_t d) -> uint32_t { return a | (b << 8) | (c << 16) | (d << 24); }; switch (uni_key) { default: return false; case ysfx_key_delete: jsfx_key = key_c('d', 'e', 'l', 0); break; case ysfx_key_f1: jsfx_key = key_c('f', '1', 0, 0); break; case ysfx_key_f2: jsfx_key = key_c('f', '2', 0, 0); break; case ysfx_key_f3: jsfx_key = key_c('f', '3', 0, 0); break; case ysfx_key_f4: jsfx_key = key_c('f', '4', 0, 0); break; case ysfx_key_f5: jsfx_key = key_c('f', '5', 0, 0); break; case ysfx_key_f6: jsfx_key = key_c('f', '6', 0, 0); break; case ysfx_key_f7: jsfx_key = key_c('f', '7', 0, 0); break; case ysfx_key_f8: jsfx_key = key_c('f', '8', 0, 0); break; case ysfx_key_f9: jsfx_key = key_c('f', '9', 0, 0); break; case ysfx_key_f10: jsfx_key = key_c('f', '1', '0', 0); break; case ysfx_key_f11: jsfx_key = key_c('f', '1', '1', 0); break; case ysfx_key_f12: jsfx_key = key_c('f', '1', '2', 0); break; case ysfx_key_left: jsfx_key = key_c('l', 'e', 'f', 't'); break; case ysfx_key_up: jsfx_key = key_c('u', 'p', 0, 0); break; case ysfx_key_right: jsfx_key = key_c('r', 'g', 'h', 't'); break; case ysfx_key_down: jsfx_key = key_c('d', 'o', 'w', 'n'); break; case ysfx_key_page_up: jsfx_key = key_c('p', 'g', 'u', 'p'); break; case ysfx_key_page_down: jsfx_key = key_c('p', 'g', 'd', 'n'); break; case ysfx_key_home: jsfx_key = key_c('h', 'o', 'm', 'e'); break; case ysfx_key_end: jsfx_key = key_c('e', 'n', 'd', 0); break; case ysfx_key_insert: jsfx_key = key_c('i', 'n', 's', 0); break; } return true; } static EEL_F NSEEL_CGEN_CALL ysfx_api_gfx_getchar(void *opaque, EEL_F *p) { ysfx_gfx_state_t *state = GFX_GET_CONTEXT(opaque); if (!state) return 0; if (*p >= 1/*2*/) { // NOTE(jpc) this is 2.0 originally, which seems wrong if (*p == 65536) { // TODO implement window flags return 0; } // current key down status uint32_t key = (uint32_t)*p; uint32_t key_id; if (translate_special_key(key, key)) key_id = key; else if (key < 256) key_id = ysfx::latin1_tolower(key); else // support the Latin-1 character set only return 0; return (EEL_F)(state->keys_pressed.find(key_id) != state->keys_pressed.end()); } if (!state->input_queue.empty()) { uint32_t key = state->input_queue.front(); state->input_queue.pop(); return (EEL_F)key; } return 0; } static EEL_F NSEEL_CGEN_CALL ysfx_api_gfx_showmenu(void *opaque, INT_PTR nparms, EEL_F **parms) { ysfx_gfx_state_t *state = GFX_GET_CONTEXT(opaque); if (!state || !state->show_menu) return 0; ysfx_t *fx = (ysfx_t *)state->lice->m_user_ctx; std::string desc; if (!ysfx_string_get(fx, *parms[0], desc) || desc.empty()) return 0; int32_t x = (int32_t)*fx->var.gfx_x; int32_t y = (int32_t)*fx->var.gfx_y; return state->show_menu(state->callback_data, desc.c_str(), x, y); } static EEL_F NSEEL_CGEN_CALL ysfx_api_gfx_setcursor(void *opaque, INT_PTR nparms, EEL_F **parms) { ysfx_gfx_state_t *state = GFX_GET_CONTEXT(opaque); if (!state || !state->set_cursor) return 0; int32_t id = (int32_t)*parms[0]; state->set_cursor(state->callback_data, id); return 0; } static EEL_F NSEEL_CGEN_CALL ysfx_api_gfx_getdropfile(void *opaque, INT_PTR np, EEL_F **parms) { ysfx_gfx_state_t *state = GFX_GET_CONTEXT(opaque); if (!state || !state->get_drop_file) return 0; const int32_t idx = (int)*parms[0]; if (idx < 0) { state->get_drop_file(state->callback_data, -1); return 0; } const char *file = state->get_drop_file(state->callback_data, idx); if (!file) return 0; if (np > 1) { ysfx_t *fx = (ysfx_t *)state->lice->m_user_ctx; ysfx_string_set(fx, *parms[1], file); } return 1; } //------------------------------------------------------------------------------ #if !defined(YSFX_NO_GFX) ysfx_gfx_state_t::ysfx_gfx_state_t(ysfx_t *fx) : lice{new eel_lice_state{fx->vm.get(), fx, ysfx_gfx_max_images, ysfx_gfx_max_fonts}} { lice->m_framebuffer = new LICE_WrapperBitmap{nullptr, 0, 0, 0, false}; } ysfx_gfx_state_t::~ysfx_gfx_state_t() { } #endif ysfx_gfx_state_t *ysfx_gfx_state_new(ysfx_t *fx) { return new ysfx_gfx_state_t{fx}; } void ysfx_gfx_state_free(ysfx_gfx_state_t *state) { delete state; } void ysfx_gfx_state_set_bitmap(ysfx_gfx_state_t *state, uint8_t *data, uint32_t w, uint32_t h, uint32_t stride) { if (stride == 0) stride = 4 * w; eel_lice_state *lice = state->lice.get(); assert(stride % 4 == 0); *static_cast(lice->m_framebuffer) = LICE_WrapperBitmap{(LICE_pixel *)data, (int)w, (int)h, (int)(stride / 4), false}; } void ysfx_gfx_state_set_scale_factor(ysfx_gfx_state_t *state, ysfx_real scale) { state->scale = scale; } void ysfx_gfx_state_set_callback_data(ysfx_gfx_state_t *state, void *callback_data) { state->callback_data = callback_data; } void ysfx_gfx_state_set_show_menu_callback(ysfx_gfx_state_t *state, int (*callback)(void *, const char *, int32_t, int32_t)) { state->show_menu = callback; } void ysfx_gfx_state_set_set_cursor_callback(ysfx_gfx_state_t *state, void (*callback)(void *, int32_t)) { state->set_cursor = callback; } void ysfx_gfx_state_set_get_drop_file_callback(ysfx_gfx_state_t *state, const char *(*callback)(void *, int32_t)) { state->get_drop_file = callback; } bool ysfx_gfx_state_is_dirty(ysfx_gfx_state_t *state) { return state->lice->m_framebuffer_dirty; } void ysfx_gfx_state_add_key(ysfx_gfx_state_t *state, uint32_t mods, uint32_t key, bool press) { if (key < 1) return; uint32_t key_id; if (translate_special_key(key, key)) key_id = key; else if (key < 256) key_id = ysfx::latin1_tolower(key); else // support the Latin-1 character set only return; uint32_t key_with_mod = key; if (key_id >= 'a' && key_id <= 'z') { uint32_t off = (uint32_t)(key_id - 'a'); if (mods & (ysfx_mod_ctrl|ysfx_mod_alt)) key_with_mod = off + 257; else if (mods & ysfx_mod_ctrl) key_with_mod = off + 1; else if (mods & ysfx_mod_alt) key_with_mod = off + 321; } if (press && key_with_mod > 0) { while (state->input_queue.size() >= ysfx_gfx_max_input) state->input_queue.pop(); state->input_queue.push(key_with_mod); } if (press) state->keys_pressed.insert(key_id); else state->keys_pressed.erase(key_id); } //------------------------------------------------------------------------------ void ysfx_gfx_enter(ysfx_t *fx, bool doinit) { fx->gfx.mutex.lock(); ysfx_gfx_state_t *state = fx->gfx.state.get(); if (doinit) { if (fx->gfx.must_init.exchange(false, std::memory_order_acquire)) { *fx->var.gfx_r = 1.0; *fx->var.gfx_g = 1.0; *fx->var.gfx_b = 1.0; *fx->var.gfx_a = 1.0; *fx->var.gfx_a2 = 1.0; *fx->var.gfx_dest = -1.0; *fx->var.mouse_wheel = 0.0; *fx->var.mouse_hwheel = 0.0; // NOTE the above are according to eel_lice.h `resetVarsToStock` // it helps to reset a few more, especially for clearing *fx->var.gfx_mode = 0; *fx->var.gfx_clear = 0; *fx->var.gfx_texth = 0; *fx->var.mouse_cap = 0; // reset key state state->input_queue = {}; state->keys_pressed = {}; // reset lice eel_lice_state *lice = state->lice.get(); LICE_WrapperBitmap framebuffer = *static_cast(lice->m_framebuffer); state->lice.reset(); lice = new eel_lice_state{fx->vm.get(), fx, ysfx_gfx_max_images, ysfx_gfx_max_fonts}; state->lice.reset(lice); lice->m_framebuffer = new LICE_WrapperBitmap(framebuffer); // load images from filenames uint32_t numfiles = (uint32_t)fx->source.main->header.filenames.size(); for (uint32_t i = 0; i < numfiles; ++i) lice->gfx_loadimg(fx, (int32_t)i, (EEL_F)i); fx->gfx.ready = true; } } ysfx_set_thread_id(ysfx_thread_id_gfx); } void ysfx_gfx_leave(ysfx_t *fx) { ysfx_set_thread_id(ysfx_thread_id_none); fx->gfx.mutex.unlock(); } ysfx_gfx_state_t *ysfx_gfx_get_context(ysfx_t *fx) { if (!fx) return nullptr; // NOTE: make sure that this will be used from the @gfx thread only if (ysfx_get_thread_id() != ysfx_thread_id_gfx) return nullptr; return fx->gfx.state.get(); } void ysfx_gfx_prepare(ysfx_t *fx) { ysfx_gfx_state_t *state = ysfx_gfx_get_context(fx); eel_lice_state *lice = state->lice.get(); lice->m_framebuffer_dirty = false; // set variables `gfx_w` and `gfx_h` ysfx_real gfx_w = (ysfx_real)lice->m_framebuffer->getWidth(); ysfx_real gfx_h = (ysfx_real)lice->m_framebuffer->getHeight(); if (state->scale > 1.0) { gfx_w *= state->scale; gfx_h *= state->scale; *fx->var.gfx_ext_retina = state->scale; } *fx->var.gfx_w = gfx_w; *fx->var.gfx_h = gfx_h; } #endif //------------------------------------------------------------------------------ void ysfx_api_init_gfx() { #if !defined(YSFX_NO_GFX) lice_stb_install_loaders(); #endif NSEEL_addfunc_retptr("gfx_lineto", 3, NSEEL_PProc_THIS, &ysfx_api_gfx_lineto); NSEEL_addfunc_retptr("gfx_lineto", 2, NSEEL_PProc_THIS, &ysfx_api_gfx_lineto2); NSEEL_addfunc_retptr("gfx_rectto", 2, NSEEL_PProc_THIS, &ysfx_api_gfx_rectto); NSEEL_addfunc_varparm("gfx_rect", 4, NSEEL_PProc_THIS, &ysfx_api_gfx_rect); NSEEL_addfunc_varparm("gfx_line", 4, NSEEL_PProc_THIS, &ysfx_api_gfx_line); NSEEL_addfunc_varparm("gfx_gradrect", 8, NSEEL_PProc_THIS, &ysfx_api_gfx_gradrect); NSEEL_addfunc_varparm("gfx_muladdrect", 7, NSEEL_PProc_THIS, &ysfx_api_gfx_muladdrect); NSEEL_addfunc_varparm("gfx_deltablit", 9, NSEEL_PProc_THIS, &ysfx_api_gfx_deltablit); NSEEL_addfunc_exparms("gfx_transformblit", 8, NSEEL_PProc_THIS, &ysfx_api_gfx_transformblit); NSEEL_addfunc_varparm("gfx_circle", 3, NSEEL_PProc_THIS, &ysfx_api_gfx_circle); NSEEL_addfunc_varparm("gfx_triangle", 6, NSEEL_PProc_THIS, &ysfx_api_gfx_triangle); NSEEL_addfunc_varparm("gfx_roundrect", 5, NSEEL_PProc_THIS, &ysfx_api_gfx_roundrect); NSEEL_addfunc_varparm("gfx_arc", 5, NSEEL_PProc_THIS, &ysfx_api_gfx_arc); NSEEL_addfunc_retptr("gfx_blurto", 2, NSEEL_PProc_THIS, &ysfx_api_gfx_blurto); NSEEL_addfunc_exparms("gfx_showmenu", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_showmenu); NSEEL_addfunc_varparm("gfx_setcursor", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_setcursor); NSEEL_addfunc_retptr("gfx_drawnumber", 2, NSEEL_PProc_THIS, &ysfx_api_gfx_drawnumber); NSEEL_addfunc_retptr("gfx_drawchar", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_drawchar); NSEEL_addfunc_varparm("gfx_drawstr", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_drawstr); NSEEL_addfunc_retptr("gfx_measurestr", 3, NSEEL_PProc_THIS, &ysfx_api_gfx_measurestr); NSEEL_addfunc_retptr("gfx_measurechar", 3, NSEEL_PProc_THIS, &ysfx_api_gfx_measurechar); NSEEL_addfunc_varparm("gfx_printf", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_printf); NSEEL_addfunc_retptr("gfx_setpixel", 3, NSEEL_PProc_THIS, &ysfx_api_gfx_setpixel); NSEEL_addfunc_retptr("gfx_getpixel", 3, NSEEL_PProc_THIS, &ysfx_api_gfx_getpixel); NSEEL_addfunc_retptr("gfx_getimgdim", 3, NSEEL_PProc_THIS, &ysfx_api_gfx_getimgdim); NSEEL_addfunc_retval("gfx_setimgdim", 3, NSEEL_PProc_THIS, &ysfx_api_gfx_setimgdim); NSEEL_addfunc_retval("gfx_loadimg", 2, NSEEL_PProc_THIS, &ysfx_api_gfx_loadimg); NSEEL_addfunc_retptr("gfx_blitext", 3, NSEEL_PProc_THIS, &ysfx_api_gfx_blitext); NSEEL_addfunc_varparm("gfx_blit", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_blit2); NSEEL_addfunc_varparm("gfx_setfont", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_setfont); NSEEL_addfunc_varparm("gfx_getfont", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_getfont); NSEEL_addfunc_varparm("gfx_set", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_set); NSEEL_addfunc_varparm("gfx_getdropfile", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_getdropfile); NSEEL_addfunc_varparm("gfx_getsyscol", 0, NSEEL_PProc_THIS, &ysfx_api_gfx_getsyscol); NSEEL_addfunc_retval("gfx_getchar", 1, NSEEL_PProc_THIS, &ysfx_api_gfx_getchar); } //------------------------------------------------------------------------------ // SWELL helpers #if !defined(YSFX_NO_GFX) // implement these GL functions on SWELL targets where they are missing #if !defined(SWELL_TARGET_GDK) && !defined(SWELL_TARGET_OSX) void SWELL_SetViewGL(HWND h, char wantGL) { } bool SWELL_GetViewGL(HWND h) { return false; } bool SWELL_SetGLContextToView(HWND h) { return false; } #endif // !defined(SWELL_TARGET_GDK) && !defined(SWELL_TARGET_OSX) #endif