| 
							- // 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 <vector>
 - #include <queue>
 - #include <unordered_set>
 - #include <memory>
 - #include <atomic>
 - #include <cassert>
 - 
 - #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<eel_lice_state> lice;
 -     std::queue<uint32_t> input_queue;
 -     std::unordered_set<uint32_t> 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_WrapperBitmap *>(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_WrapperBitmap *>(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
 
 
  |