v/vlib/gg/image.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()
}