2020-01-23 23:04:46 +03:00
|
|
|
// Copyright (c) 2019-2020 Alexander Medvednikov. All rights reserved.
|
2020-07-06 21:29:05 +03:00
|
|
|
// Use of this source code is governed by an MIT license that can be found in the LICENSE file.
|
2019-06-22 21:20:28 +03:00
|
|
|
module gg
|
|
|
|
|
2020-04-26 12:42:44 +03:00
|
|
|
import gx
|
2020-06-04 17:05:12 +03:00
|
|
|
import sokol
|
|
|
|
import sokol.sapp
|
|
|
|
import sokol.sgl
|
|
|
|
import sokol.gfx
|
2019-06-22 21:20:28 +03:00
|
|
|
|
2020-08-02 00:40:25 +03:00
|
|
|
// import time
|
|
|
|
pub type FNCb = fn (x voidptr)
|
|
|
|
|
|
|
|
pub type FNEvent = fn (e, x voidptr)
|
|
|
|
|
|
|
|
pub type FNFail = fn (msg string, x voidptr)
|
|
|
|
|
|
|
|
pub type FNKeyDown = fn (c sapp.KeyCode, m sapp.Modifier, x voidptr)
|
|
|
|
|
|
|
|
pub type FNChar = fn (c u32, x voidptr)
|
2019-06-22 21:20:28 +03:00
|
|
|
|
2020-06-04 17:05:12 +03:00
|
|
|
pub struct Config {
|
2019-08-23 00:00:31 +03:00
|
|
|
pub:
|
2020-08-02 00:40:25 +03:00
|
|
|
width int
|
|
|
|
height int
|
|
|
|
use_ortho bool
|
|
|
|
retina bool
|
|
|
|
resizable bool
|
|
|
|
user_data voidptr
|
|
|
|
font_size int
|
|
|
|
create_window bool
|
2020-06-04 17:05:12 +03:00
|
|
|
// window_user_ptr voidptr
|
2020-08-02 00:40:25 +03:00
|
|
|
window_title string
|
2020-06-04 20:57:13 +03:00
|
|
|
borderless_window bool
|
2020-08-02 00:40:25 +03:00
|
|
|
always_on_top bool
|
|
|
|
bg_color gx.Color
|
|
|
|
init_fn FNCb = voidptr(0)
|
|
|
|
frame_fn FNCb = voidptr(0)
|
|
|
|
cleanup_fn FNCb = voidptr(0)
|
|
|
|
fail_fn FNFail = voidptr(0)
|
|
|
|
event_fn FNEvent = voidptr(0)
|
|
|
|
keydown_fn FNKeyDown = voidptr(0) // special case of event_fn
|
|
|
|
char_fn FNChar = voidptr(0) // special case of event_fn
|
|
|
|
wait_events bool // set this to true for UIs, to save power
|
|
|
|
fullscreen bool
|
|
|
|
scale f32 = 1.0 // vid needs this
|
|
|
|
// init_text bool
|
|
|
|
font_path string
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
|
2020-06-04 21:26:18 +03:00
|
|
|
pub struct Context {
|
2020-07-06 20:45:00 +03:00
|
|
|
render_text bool
|
2020-08-05 17:00:24 +03:00
|
|
|
mut:
|
|
|
|
// a cache with all images created by the user. used for sokol image init and to save space
|
|
|
|
// (so that the user can store image ids, not entire Image objects)
|
|
|
|
image_cache []Image
|
2019-08-23 00:00:31 +03:00
|
|
|
pub mut:
|
2020-08-02 00:40:25 +03:00
|
|
|
scale f32 = 1.0 // will get set to 2.0 for retina, will remain 1.0 for normal
|
|
|
|
width int
|
|
|
|
height int
|
|
|
|
clear_pass C.sg_pass_action
|
|
|
|
window C.sapp_desc
|
2020-08-03 00:30:08 +03:00
|
|
|
timage_pip C.sgl_pipeline
|
2020-08-02 00:40:25 +03:00
|
|
|
config Config
|
|
|
|
ft &FT
|
2020-07-06 20:45:00 +03:00
|
|
|
font_inited bool
|
2020-06-04 17:05:12 +03:00
|
|
|
}
|
|
|
|
|
2020-08-02 00:40:25 +03:00
|
|
|
pub struct Size {
|
|
|
|
pub:
|
|
|
|
width int
|
|
|
|
height int
|
|
|
|
}
|
2020-06-04 20:57:13 +03:00
|
|
|
|
2020-06-04 17:05:12 +03:00
|
|
|
fn gg_init_sokol_window(user_data voidptr) {
|
2020-06-04 21:26:18 +03:00
|
|
|
mut g := &Context(user_data)
|
2020-08-23 08:47:50 +03:00
|
|
|
desc := sapp.create_desc()
|
2020-08-23 08:25:30 +03:00
|
|
|
/*
|
2020-06-04 17:05:12 +03:00
|
|
|
desc := C.sg_desc{
|
|
|
|
mtl_device: sapp.metal_get_device()
|
|
|
|
mtl_renderpass_descriptor_cb: sapp.metal_get_renderpass_descriptor
|
|
|
|
mtl_drawable_cb: sapp.metal_get_drawable
|
|
|
|
d3d11_device: sapp.d3d11_get_device()
|
|
|
|
d3d11_device_context: sapp.d3d11_get_device_context()
|
|
|
|
d3d11_render_target_view_cb: sapp.d3d11_get_render_target_view
|
|
|
|
d3d11_depth_stencil_view_cb: sapp.d3d11_get_depth_stencil_view
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
2020-08-23 08:25:30 +03:00
|
|
|
*/
|
2020-06-04 17:05:12 +03:00
|
|
|
gfx.setup(&desc)
|
|
|
|
sgl_desc := C.sgl_desc_t{}
|
|
|
|
sgl.setup(&sgl_desc)
|
2020-06-04 21:09:46 +03:00
|
|
|
g.scale = sapp.dpi_scale()
|
|
|
|
// NB: on older X11, `Xft.dpi` from ~/.Xresources, that sokol uses,
|
|
|
|
// may not be set which leads to sapp.dpi_scale reporting incorrectly 0.0
|
|
|
|
if g.scale < 0.1 {
|
|
|
|
g.scale = 1.0
|
|
|
|
}
|
2020-08-02 00:40:25 +03:00
|
|
|
// is_high_dpi := sapp.high_dpi()
|
|
|
|
// fb_w := sapp.width()
|
|
|
|
// fb_h := sapp.height()
|
|
|
|
// println('g.scale=$g.scale is_high_dpi=$is_high_dpi fb_w=$fb_w fb_h=$fb_h')
|
|
|
|
// if g.config.init_text {
|
2020-07-06 20:45:00 +03:00
|
|
|
if g.config.font_path != '' {
|
2020-08-02 00:40:25 +03:00
|
|
|
// t := time.ticks()
|
|
|
|
g.ft = new_ft({
|
|
|
|
font_path: g.config.font_path
|
|
|
|
scale: sapp.dpi_scale()
|
|
|
|
}) or {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
// println('FT took ${time.ticks()-t} ms')
|
2020-07-06 20:45:00 +03:00
|
|
|
g.font_inited = true
|
|
|
|
}
|
2020-08-03 00:30:08 +03:00
|
|
|
//
|
|
|
|
mut pipdesc := C.sg_pipeline_desc{}
|
|
|
|
unsafe { C.memset(&pipdesc, 0, sizeof(pipdesc)) }
|
|
|
|
pipdesc.blend.enabled = true
|
|
|
|
pipdesc.blend.src_factor_rgb = C.SG_BLENDFACTOR_SRC_ALPHA
|
|
|
|
pipdesc.blend.dst_factor_rgb = C.SG_BLENDFACTOR_ONE_MINUS_SRC_ALPHA
|
|
|
|
g.timage_pip = sgl.make_pipeline(&pipdesc)
|
|
|
|
//
|
2020-06-04 17:05:12 +03:00
|
|
|
if g.config.init_fn != voidptr(0) {
|
2020-07-06 20:45:00 +03:00
|
|
|
g.config.init_fn(g.config.user_data)
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
2020-08-05 17:00:24 +03:00
|
|
|
// Create images now that we can do that after sg is inited
|
|
|
|
for i in 0..g.image_cache.len {
|
|
|
|
g.image_cache[i].init_sokol_image()
|
|
|
|
}
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
|
2020-06-04 17:05:12 +03:00
|
|
|
fn gg_frame_fn(user_data voidptr) {
|
2020-06-04 21:26:18 +03:00
|
|
|
mut g := &Context(user_data)
|
2020-06-04 17:05:12 +03:00
|
|
|
if g.config.frame_fn != voidptr(0) {
|
2020-07-06 20:45:00 +03:00
|
|
|
g.config.frame_fn(g.config.user_data)
|
2019-08-23 00:00:31 +03:00
|
|
|
}
|
|
|
|
}
|
2019-07-19 14:27:44 +03:00
|
|
|
|
2020-06-05 11:46:58 +03:00
|
|
|
// TODO: remove this hacky workaround...
|
|
|
|
// NB: todo_remove_this is needed to workaround a v bug,
|
|
|
|
// where it thinks that &sapp.Event(x) is a function call,
|
|
|
|
// instead of a cast, if v has not yet seen &sapp.Event used
|
|
|
|
// as a parameter type.
|
2020-08-02 00:40:25 +03:00
|
|
|
fn todo_remove_this(e &sapp.Event) {
|
|
|
|
}
|
2020-06-05 11:46:58 +03:00
|
|
|
|
2020-08-02 00:40:25 +03:00
|
|
|
fn gg_event_fn(ce &C.sapp_event, user_data voidptr) {
|
2020-06-05 11:46:58 +03:00
|
|
|
e := &sapp.Event(ce)
|
2020-06-04 21:26:18 +03:00
|
|
|
mut g := &Context(user_data)
|
2020-06-04 17:05:12 +03:00
|
|
|
if g.config.event_fn != voidptr(0) {
|
|
|
|
g.config.event_fn(e, g.config.user_data)
|
2019-08-23 00:00:31 +03:00
|
|
|
}
|
2020-06-05 11:46:58 +03:00
|
|
|
match e.typ {
|
|
|
|
.key_down {
|
|
|
|
if g.config.keydown_fn != voidptr(0) {
|
|
|
|
kdfn := g.config.keydown_fn
|
|
|
|
kdfn(e.key_code, e.modifiers, g.config.user_data)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
.char {
|
|
|
|
if g.config.char_fn != voidptr(0) {
|
|
|
|
cfn := g.config.char_fn
|
|
|
|
cfn(e.char_code, g.config.user_data)
|
|
|
|
}
|
|
|
|
}
|
2020-08-02 00:40:25 +03:00
|
|
|
else {}
|
2020-06-05 11:46:58 +03:00
|
|
|
}
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
|
2020-08-02 00:40:25 +03:00
|
|
|
fn gg_cleanup_fn(user_data voidptr) {
|
2020-06-04 21:26:18 +03:00
|
|
|
mut g := &Context(user_data)
|
2020-06-04 17:05:12 +03:00
|
|
|
if g.config.cleanup_fn != voidptr(0) {
|
|
|
|
g.config.cleanup_fn(g.config.user_data)
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-02 00:40:25 +03:00
|
|
|
fn gg_fail_fn(msg charptr, user_data voidptr) {
|
2020-06-04 21:26:18 +03:00
|
|
|
mut g := &Context(user_data)
|
2020-06-04 17:05:12 +03:00
|
|
|
vmsg := tos3(msg)
|
|
|
|
if g.config.fail_fn != voidptr(0) {
|
|
|
|
g.config.fail_fn(vmsg, g.config.user_data)
|
2020-08-02 00:40:25 +03:00
|
|
|
} else {
|
2020-06-04 17:05:12 +03:00
|
|
|
eprintln('gg error: $vmsg')
|
|
|
|
}
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
|
2020-06-04 17:05:12 +03:00
|
|
|
//
|
2020-08-02 00:40:25 +03:00
|
|
|
pub fn new_context(cfg Config) &Context {
|
2020-06-04 21:26:18 +03:00
|
|
|
mut g := &Context{
|
2020-06-04 17:05:12 +03:00
|
|
|
width: cfg.width
|
|
|
|
height: cfg.height
|
|
|
|
config: cfg
|
2020-07-06 20:45:00 +03:00
|
|
|
render_text: cfg.font_path != ''
|
|
|
|
ft: 0
|
2020-06-04 17:05:12 +03:00
|
|
|
}
|
2020-08-19 08:10:42 +03:00
|
|
|
g.set_bg_color(cfg.bg_color)
|
2020-08-02 00:40:25 +03:00
|
|
|
// C.printf('new_context() %p\n', cfg.user_data)
|
2020-06-04 17:05:12 +03:00
|
|
|
window := C.sapp_desc{
|
|
|
|
user_data: g
|
|
|
|
init_userdata_cb: gg_init_sokol_window
|
|
|
|
frame_userdata_cb: gg_frame_fn
|
|
|
|
event_userdata_cb: gg_event_fn
|
|
|
|
fail_userdata_cb: gg_fail_fn
|
|
|
|
cleanup_userdata_cb: gg_cleanup_fn
|
|
|
|
window_title: cfg.window_title.str
|
|
|
|
html5_canvas_name: cfg.window_title.str
|
|
|
|
width: cfg.width
|
|
|
|
height: cfg.height
|
2020-06-04 21:09:46 +03:00
|
|
|
high_dpi: true
|
2020-06-05 00:51:54 +03:00
|
|
|
fullscreen: cfg.fullscreen
|
2020-06-04 17:05:12 +03:00
|
|
|
}
|
2020-08-02 00:40:25 +03:00
|
|
|
if cfg.use_ortho {
|
|
|
|
} else {
|
|
|
|
}
|
2020-06-04 17:05:12 +03:00
|
|
|
g.window = window
|
|
|
|
return g
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
|
2020-06-04 21:26:18 +03:00
|
|
|
pub fn (gg &Context) run() {
|
2020-06-04 17:05:12 +03:00
|
|
|
sapp.run(&gg.window)
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
|
2020-08-19 08:10:42 +03:00
|
|
|
pub fn (mut ctx Context) set_bg_color(c gx.Color) {
|
|
|
|
ctx.clear_pass = gfx.create_clear_pass(f32(c.r) / 255.0, f32(c.g) / 255.0,
|
|
|
|
f32(c.b) / 255.0, f32(c.a) / 255.0)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Fix alpha
|
2020-06-04 21:26:18 +03:00
|
|
|
pub fn (ctx &Context) draw_rect(x, y, w, h f32, c gx.Color) {
|
2020-09-20 12:05:30 +03:00
|
|
|
if c.a != 255 {
|
|
|
|
sgl.load_pipeline(ctx.timage_pip)
|
|
|
|
}
|
2020-08-19 08:10:42 +03:00
|
|
|
sgl.c4b(c.r, c.g, c.b, c.a)
|
2020-06-04 17:05:12 +03:00
|
|
|
sgl.begin_quads()
|
|
|
|
sgl.v2f(x * ctx.scale, y * ctx.scale)
|
|
|
|
sgl.v2f((x + w) * ctx.scale, y * ctx.scale)
|
|
|
|
sgl.v2f((x + w) * ctx.scale, (y + h) * ctx.scale)
|
|
|
|
sgl.v2f(x * ctx.scale, (y + h) * ctx.scale)
|
2020-09-20 04:47:22 +03:00
|
|
|
sgl.end()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn (ctx &Context) draw_triangle(x, y, x2, y2, x3, y3 f32, c gx.Color) {
|
2020-09-20 12:05:30 +03:00
|
|
|
if c.a != 255 {
|
|
|
|
sgl.load_pipeline(ctx.timage_pip)
|
|
|
|
}
|
2020-09-20 04:47:22 +03:00
|
|
|
sgl.c4b(c.r, c.g, c.b, c.a)
|
|
|
|
sgl.begin_quads()
|
|
|
|
sgl.v2f(x * ctx.scale, y * ctx.scale)
|
|
|
|
sgl.v2f(x2 * ctx.scale, y2 * ctx.scale)
|
|
|
|
sgl.v2f(x3 * ctx.scale, y3 * ctx.scale)
|
2020-06-04 17:05:12 +03:00
|
|
|
sgl.end()
|
|
|
|
}
|
|
|
|
|
2020-06-05 00:51:54 +03:00
|
|
|
pub fn (ctx &Context) draw_empty_rect(x, y, w, h f32, c gx.Color) {
|
2020-09-20 12:05:30 +03:00
|
|
|
if c.a != 255 {
|
|
|
|
sgl.load_pipeline(ctx.timage_pip)
|
|
|
|
}
|
2020-08-19 08:10:42 +03:00
|
|
|
sgl.c4b(c.r, c.g, c.b, c.a)
|
2020-06-04 17:05:12 +03:00
|
|
|
sgl.begin_line_strip()
|
2020-06-05 00:51:54 +03:00
|
|
|
if ctx.scale == 1 {
|
|
|
|
sgl.v2f(x, y)
|
|
|
|
sgl.v2f(x + w, y)
|
|
|
|
sgl.v2f(x + w, y + h)
|
|
|
|
sgl.v2f(x, y + h)
|
|
|
|
sgl.v2f(x, y)
|
2020-08-02 00:40:25 +03:00
|
|
|
} else {
|
2020-06-05 00:51:54 +03:00
|
|
|
sgl.v2f(x * ctx.scale, y * ctx.scale)
|
|
|
|
sgl.v2f((x + w) * ctx.scale, y * ctx.scale)
|
|
|
|
sgl.v2f((x + w) * ctx.scale, (y + h) * ctx.scale)
|
|
|
|
sgl.v2f(x * ctx.scale, (y + h) * ctx.scale)
|
2020-08-02 00:40:25 +03:00
|
|
|
sgl.v2f(x * ctx.scale, y * ctx.scale)
|
2020-06-05 00:51:54 +03:00
|
|
|
}
|
2020-06-04 17:05:12 +03:00
|
|
|
sgl.end()
|
2020-01-16 00:17:40 +03:00
|
|
|
}
|
|
|
|
|
2020-07-13 02:02:47 +03:00
|
|
|
pub fn (ctx &Context) draw_circle(x, y, r f32, c gx.Color) {
|
|
|
|
}
|
|
|
|
|
2020-06-04 21:26:18 +03:00
|
|
|
pub fn (gg &Context) begin() {
|
2020-07-06 20:45:00 +03:00
|
|
|
if gg.render_text && gg.font_inited {
|
|
|
|
gg.ft.flush()
|
|
|
|
}
|
2020-06-04 17:05:12 +03:00
|
|
|
sgl.defaults()
|
|
|
|
sgl.matrix_mode_projection()
|
|
|
|
sgl.ortho(0.0, f32(sapp.width()), f32(sapp.height()), 0.0, -1.0, 1.0)
|
2020-01-27 22:42:32 +03:00
|
|
|
}
|
|
|
|
|
2020-06-04 21:26:18 +03:00
|
|
|
pub fn (gg &Context) end() {
|
2020-06-04 17:05:12 +03:00
|
|
|
gfx.begin_default_pass(gg.clear_pass, sapp.width(), sapp.height())
|
|
|
|
sgl.draw()
|
|
|
|
gfx.end_pass()
|
|
|
|
gfx.commit()
|
|
|
|
if gg.config.wait_events {
|
2020-08-02 00:40:25 +03:00
|
|
|
// println('gg: waiting')
|
2020-06-04 17:05:12 +03:00
|
|
|
wait_events()
|
|
|
|
}
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
2019-07-20 18:38:00 +03:00
|
|
|
|
2020-08-23 05:57:12 +03:00
|
|
|
fn abs(a f32) f32 {
|
|
|
|
if a >= 0 {
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
return -a
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-05 16:43:44 +03:00
|
|
|
pub fn (ctx &Context) draw_line(x, y, x2, y2 f32, c gx.Color) {
|
2020-09-20 12:05:30 +03:00
|
|
|
if c.a != 255 {
|
|
|
|
sgl.load_pipeline(ctx.timage_pip)
|
|
|
|
}
|
2020-08-23 05:57:12 +03:00
|
|
|
if ctx.scale > 1 {
|
|
|
|
// Make the line more clear on hi dpi screens: draw a rectangle
|
|
|
|
mut width := abs(x2 - x)
|
|
|
|
mut height := abs(y2 - y)
|
|
|
|
if width == 0 {
|
|
|
|
width = 1
|
|
|
|
}
|
|
|
|
else if height == 0 {
|
|
|
|
height = 1
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.draw_rect(x, y, width, height, c)
|
|
|
|
return
|
|
|
|
}
|
2020-08-19 08:10:42 +03:00
|
|
|
sgl.c4b(c.r, c.g, c.b, c.a)
|
2020-06-05 16:43:44 +03:00
|
|
|
sgl.begin_line_strip()
|
|
|
|
sgl.v2f(x * ctx.scale, y * ctx.scale)
|
|
|
|
sgl.v2f(x2 * ctx.scale, y2 * ctx.scale)
|
|
|
|
sgl.end()
|
2020-08-02 00:40:25 +03:00
|
|
|
}
|
2020-06-05 16:43:44 +03:00
|
|
|
|
2020-06-07 16:20:32 +03:00
|
|
|
pub fn (ctx &Context) draw_rounded_rect(x, y, width, height, radius f32, color gx.Color) {
|
2020-07-06 20:45:00 +03:00
|
|
|
}
|
|
|
|
|
2020-06-07 16:20:32 +03:00
|
|
|
pub fn (ctx &Context) draw_empty_rounded_rect(x, y, width, height, radius f32, border_color gx.Color) {
|
|
|
|
}
|
|
|
|
|
2020-06-04 17:05:12 +03:00
|
|
|
fn C.WaitMessage()
|
2020-01-16 00:17:40 +03:00
|
|
|
|
2020-06-04 17:05:12 +03:00
|
|
|
pub fn wait_events() {
|
|
|
|
unsafe {
|
2020-08-02 00:40:25 +03:00
|
|
|
$if macos {
|
|
|
|
#NSEvent *event = [NSApp nextEventMatchingMask:NSEventMaskAny
|
|
|
|
#untilDate:[NSDate distantFuture]
|
|
|
|
#inMode:NSDefaultRunLoopMode
|
|
|
|
#dequeue:YES];
|
|
|
|
#[NSApp sendEvent:event];
|
2020-06-04 17:05:12 +03:00
|
|
|
}
|
|
|
|
$if windows {
|
|
|
|
C.WaitMessage()
|
|
|
|
}
|
|
|
|
}
|
2020-02-03 06:01:39 +03:00
|
|
|
}
|