151 lines
3.2 KiB
V
151 lines
3.2 KiB
V
// Copyright (c) 2019-2020 Alexander Medvednikov. All rights reserved.
|
|
// Use of this source code is governed by an MIT license that can be found in the LICENSE file.
|
|
module gg
|
|
|
|
import os
|
|
import sokol
|
|
import sokol.sgl
|
|
import stbi
|
|
|
|
pub struct Image {
|
|
pub mut:
|
|
id int // used as an index in image_cache.slots
|
|
width int
|
|
height int
|
|
nr_channels int
|
|
ok bool
|
|
data voidptr
|
|
ext string
|
|
path string
|
|
}
|
|
//
|
|
struct ImageCacheSlot {
|
|
id int
|
|
simg_ok bool
|
|
simg C.sg_image
|
|
}
|
|
[ref_only]
|
|
struct ImageCache {
|
|
mut:
|
|
current_image_id int = -1
|
|
slots []ImageCacheSlot
|
|
}
|
|
fn new_image_cache() &ImageCache {
|
|
return &ImageCache{}
|
|
}
|
|
fn (ic &ImageCache) get_new_id() int {
|
|
mut mic := &ImageCache(0)
|
|
unsafe {
|
|
mic = ic
|
|
}
|
|
mic.current_image_id++
|
|
mic.slots << ImageCacheSlot {
|
|
id: mic.current_image_id
|
|
simg_ok: false
|
|
}
|
|
return ic.current_image_id
|
|
}
|
|
fn (ic &ImageCache) set_simg(id int, simg C.sg_image) {
|
|
if id >= ic.slots.len {
|
|
panic('invalid image cache id: $id, slots.len: $ic.slots.len')
|
|
}
|
|
mut mic := &ImageCache(0)
|
|
unsafe {
|
|
mic = ic
|
|
}
|
|
mic.slots[id] = ImageCacheSlot{ id: id, simg_ok: true, simg: simg }
|
|
}
|
|
const (
|
|
image_cache = new_image_cache()
|
|
)
|
|
//
|
|
pub fn create_image(file string) Image {
|
|
if !os.exists(file) {
|
|
println('gg.create_image(): file not found: $file')
|
|
return Image{} // none
|
|
}
|
|
stb_img := stbi.load(file)
|
|
mut img := Image{
|
|
id: image_cache.get_new_id()
|
|
width: stb_img.width
|
|
height: stb_img.height
|
|
nr_channels: stb_img.nr_channels
|
|
ok: stb_img.ok
|
|
data: stb_img.data
|
|
ext: stb_img.ext
|
|
path: file
|
|
}
|
|
return img
|
|
}
|
|
|
|
pub fn create_image_from_memory(buf byteptr, bufsize int) Image {
|
|
stb_img := stbi.load_from_memory(buf, bufsize)
|
|
mut img := Image{
|
|
id: image_cache.get_new_id()
|
|
width: stb_img.width
|
|
height: stb_img.height
|
|
nr_channels: stb_img.nr_channels
|
|
ok: stb_img.ok
|
|
data: stb_img.data
|
|
ext: stb_img.ext
|
|
}
|
|
return img
|
|
}
|
|
|
|
pub fn create_image_from_byte_array(b []byte) Image {
|
|
return create_image_from_memory(b.data, b.len)
|
|
}
|
|
|
|
pub fn (img &Image) new_sokol_image() C.sg_image {
|
|
//eprintln('> new_sokol_image from img: $img')
|
|
mut img_desc := C.sg_image_desc{
|
|
width: img.width
|
|
height: img.height
|
|
num_mipmaps: 0
|
|
wrap_u: .clamp_to_edge
|
|
wrap_v: .clamp_to_edge
|
|
label: &byte(0)
|
|
d3d11_texture: 0
|
|
}
|
|
img_desc.content.subimage[0][0] = C.sg_subimage_content{
|
|
ptr: img.data
|
|
size: img.nr_channels * img.width * img.height
|
|
}
|
|
simg := C.sg_make_image(&img_desc)
|
|
return simg
|
|
}
|
|
|
|
pub fn (img &Image) get_sokol_image() C.sg_image {
|
|
slot := image_cache.slots[img.id]
|
|
if slot.simg_ok {
|
|
return slot.simg
|
|
}
|
|
simg := img.new_sokol_image()
|
|
image_cache.set_simg(img.id, simg)
|
|
return simg
|
|
}
|
|
|
|
pub fn (ctx &Context) draw_image(x, y, width, height f32, img &Image) {
|
|
simg := img.get_sokol_image()
|
|
u0 := f32(0.0)
|
|
v0 := f32(0.0)
|
|
u1 := f32(1.0)
|
|
v1 := f32(1.0)
|
|
x0 := f32(x) * ctx.scale
|
|
y0 := f32(y) * ctx.scale
|
|
x1 := f32(x + width) * ctx.scale
|
|
y1 := f32(y + height) * ctx.scale
|
|
//
|
|
sgl.load_pipeline(ctx.timage_pip)
|
|
sgl.enable_texture()
|
|
sgl.texture(simg)
|
|
sgl.begin_quads()
|
|
sgl.c4b(255, 255, 255, 255)
|
|
sgl.v2f_t2f(x0, y0, u0, v0)
|
|
sgl.v2f_t2f(x1, y0, u1, v0)
|
|
sgl.v2f_t2f(x1, y1, u1, v1)
|
|
sgl.v2f_t2f(x0, y1, u0, v1)
|
|
sgl.end()
|
|
sgl.disable_texture()
|
|
}
|