1
0
mirror of https://github.com/vlang/v.git synced 2023-08-10 21:13:21 +03:00
v/vlib/gg/image.v
2020-08-05 08:46:20 +03:00

147 lines
3.0 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:
width int
height int
nr_channels int
ok bool
data voidptr
ext string
path string
cache_slot &ImageCacheSlot = 0
}
//
struct ImageCacheSlot {
id int
mut:
is_simg_ok bool
simg C.sg_image
}
fn (mut cs ImageCacheSlot) set_sokol_image(simg C.sg_image) {
cs.simg = simg
cs.is_simg_ok = true
}
//
[ref_only]
struct ImageCache {
mut:
current_image_id int = -1
slots []ImageCacheSlot
}
fn (ic &ImageCache) get_new_cache_slot() &ImageCacheSlot {
mut mic := &ImageCache(0)
unsafe {
mic = ic
}
mic.current_image_id++
mic.slots << ImageCacheSlot {
id: mic.current_image_id
is_simg_ok: false
}
return &mic.slots[mic.slots.len-1]
}
const (
image_cache = &ImageCache{}
)
//
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{
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
cache_slot: image_cache.get_new_cache_slot()
}
return img
}
pub fn create_image_from_memory(buf byteptr, bufsize int) Image {
stb_img := stbi.load_from_memory(buf, bufsize)
mut img := Image{
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
cache_slot: image_cache.get_new_cache_slot()
}
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 := img.cache_slot
if slot.is_simg_ok {
return slot.simg
}
simg := img.new_sokol_image()
unsafe {
mut cs := slot
cs.set_sokol_image(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()
}