mirror of
https://github.com/odin-lang/Odin.git
synced 2026-02-12 14:23:33 +00:00
136 lines
3.8 KiB
Odin
136 lines
3.8 KiB
Odin
#+build ignore
|
|
package png_example
|
|
|
|
import "core:image"
|
|
import "core:image/png"
|
|
import "core:image/tga"
|
|
import "core:fmt"
|
|
import "core:mem"
|
|
|
|
demo :: proc() {
|
|
options := image.Options{.return_metadata}
|
|
err: image.Error
|
|
img: ^image.Image
|
|
|
|
PNG_FILE :: ODIN_ROOT + "misc/logo-slim.png"
|
|
|
|
img, err = png.load(PNG_FILE, options)
|
|
defer png.destroy(img)
|
|
|
|
if err != nil {
|
|
fmt.eprintfln("Trying to read PNG file %v returned %v.", PNG_FILE, err)
|
|
} else {
|
|
fmt.printfln("Image: %vx%vx%v, %v-bit.", img.width, img.height, img.channels, img.depth)
|
|
|
|
if v, ok := img.metadata.(^image.PNG_Info); ok {
|
|
// Handle ancillary chunks as you wish.
|
|
// We provide helper functions for a few types.
|
|
for c in v.chunks {
|
|
#partial switch c.header.type {
|
|
case .tIME:
|
|
if t, t_ok := png.core_time(c); t_ok {
|
|
fmt.printfln("[tIME]: %v", t)
|
|
}
|
|
case .gAMA:
|
|
if gama, gama_ok := png.gamma(c); gama_ok {
|
|
fmt.printfln("[gAMA]: %v", gama)
|
|
}
|
|
case .pHYs:
|
|
if phys, phys_ok := png.phys(c); phys_ok {
|
|
if phys.unit == .Meter {
|
|
xm := f32(img.width) / f32(phys.ppu_x)
|
|
ym := f32(img.height) / f32(phys.ppu_y)
|
|
dpi_x, dpi_y := png.phys_to_dpi(phys)
|
|
fmt.printfln("[pHYs] Image resolution is %v x %v pixels per meter.", phys.ppu_x, phys.ppu_y)
|
|
fmt.printfln("[pHYs] Image resolution is %v x %v DPI.", dpi_x, dpi_y)
|
|
fmt.printfln("[pHYs] Image dimensions are %v x %v meters.", xm, ym)
|
|
} else {
|
|
fmt.printfln("[pHYs] x: %v, y: %v pixels per unknown unit.", phys.ppu_x, phys.ppu_y)
|
|
}
|
|
}
|
|
case .iTXt, .zTXt, .tEXt:
|
|
res, ok_text := png.text(c)
|
|
if ok_text {
|
|
if c.header.type == .iTXt {
|
|
fmt.printfln("[iTXt] %v (%v:%v): %v", res.keyword, res.language, res.keyword_localized, res.text)
|
|
} else {
|
|
fmt.printfln("[tEXt/zTXt] %v: %v", res.keyword, res.text)
|
|
}
|
|
}
|
|
defer png.text_destroy(res)
|
|
case .bKGD:
|
|
fmt.printfln("[bKGD] %v", img.background)
|
|
case .eXIf:
|
|
if res, ok_exif := png.exif(c); ok_exif {
|
|
/*
|
|
Other than checking the signature and byte order, we don't handle Exif data.
|
|
If you wish to interpret it, pass it to an Exif parser.
|
|
*/
|
|
fmt.printfln("[eXIf] %v", res)
|
|
}
|
|
case .PLTE:
|
|
if plte, plte_ok := png.plte(c); plte_ok {
|
|
fmt.printfln("[PLTE] %v", plte)
|
|
} else {
|
|
fmt.printfln("[PLTE] Error")
|
|
}
|
|
case .hIST:
|
|
if res, ok_hist := png.hist(c); ok_hist {
|
|
fmt.printfln("[hIST] %v", res)
|
|
}
|
|
case .cHRM:
|
|
if res, ok_chrm := png.chrm(c); ok_chrm {
|
|
fmt.printfln("[cHRM] %v", res)
|
|
}
|
|
case .sPLT:
|
|
res, ok_splt := png.splt(c)
|
|
if ok_splt {
|
|
fmt.printfln("[sPLT] %v", res)
|
|
}
|
|
png.splt_destroy(res)
|
|
case .sBIT:
|
|
if res, ok_sbit := png.sbit(c); ok_sbit {
|
|
fmt.printfln("[sBIT] %v", res)
|
|
}
|
|
case .iCCP:
|
|
res, ok_iccp := png.iccp(c)
|
|
if ok_iccp {
|
|
fmt.printfln("[iCCP] %v", res)
|
|
}
|
|
png.iccp_destroy(res)
|
|
case .sRGB:
|
|
if res, ok_srgb := png.srgb(c); ok_srgb {
|
|
fmt.printfln("[sRGB] Rendering intent: %v", res)
|
|
}
|
|
case:
|
|
type := c.header.type
|
|
name := png.chunk_type_to_name(&type)
|
|
fmt.printfln("[%v]: %v", name, c.data)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fmt.printfln("Done parsing metadata.")
|
|
|
|
if err == nil && .do_not_decompress_image not_in options && .info not_in options {
|
|
if err = tga.save("out.tga", img); err == nil {
|
|
fmt.println("Saved decoded image.")
|
|
} else {
|
|
fmt.eprintfln("Error %v saving out.ppm.", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
main :: proc() {
|
|
track: mem.Tracking_Allocator
|
|
mem.tracking_allocator_init(&track, context.allocator)
|
|
defer mem.tracking_allocator_destroy(&track)
|
|
context.allocator = mem.tracking_allocator(&track)
|
|
|
|
demo()
|
|
|
|
for _, leak in track.allocation_map {
|
|
fmt.printf("%v leaked %m", leak.location, leak.size)
|
|
}
|
|
} |