;;; guile-2d ;;; Copyright (C) 2013 David Thompson ;;; ;;; Guile-2d is free software: you can redistribute it and/or modify it ;;; under the terms of the GNU Lesser General Public License as ;;; published by the Free Software Foundation, either version 3 of the ;;; License, or (at your option) any later version. ;;; ;;; Guile-2d is distributed in the hope that it will be useful, but ;;; WITHOUT ANY WARRANTY; without even the implied warranty of ;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU ;;; Lesser General Public License for more details. ;;; ;;; You should have received a copy of the GNU Lesser General Public ;;; License along with this program. If not, see ;;; . ;;; Commentary: ;; ;; Textures and texture regions are high level wrappers over OpenGL ;; textures. ;; ;;; Code: (define-module (2d texture) #:use-module (srfi srfi-9) #:use-module (srfi srfi-42) #:use-module ((sdl sdl) #:prefix SDL:) #:use-module (figl gl) #:use-module (2d wrappers gl) #:use-module (2d helpers) #:export (make-texture make-texture-region load-texture texture? texture-region? texture-id texture-width texture-height texture-s1 texture-t1 texture-s2 texture-t2 surface->texture draw-texture split-texture)) ;;; ;;; Textures ;;; ;; The object is a simple wrapper around an OpenGL texture ;; id. (define-record-type (%make-texture id parent width height s1 t1 s2 t2) texture? (id texture-id) (parent texture-parent) (width texture-width) (height texture-height) (s1 texture-s1) (t1 texture-t1) (s2 texture-s2) (t2 texture-t2)) (define (texture-region? texture) (texture? (texture-parent texture))) (define (make-texture id parent width height s1 t1 s2 t2) (let ((texture (%make-texture id parent width height s1 t1 s2 t2))) (texture-guardian texture) texture)) (define (make-texture-region texture x y width height) "Creates a new texture region given a texture and a pixel region." (let* ((w (texture-width texture)) (h (texture-height texture))) (make-texture (texture-id texture) texture width height (/ x w) (/ y h) (/ (+ x width) w) (/ (+ y height) h)))) ;; Use a guardian and an after GC hook that ensures that OpenGL ;; textures are deleted when texture objects are GC'd. (define texture-guardian (make-guardian)) (define (reap-textures) (let loop ((texture (texture-guardian))) (when texture ;; Do not reap texture regions (unless (texture-region? texture) ;; When attempting to reap structures upon guile exit, the ;; dynamic pointer to gl-delete-textures becomes invalid. So, we ;; ignore the error and move on. (catch 'misc-error (lambda () (gl-delete-texture (texture-id texture))) (lambda (key . args) #f))) (loop (texture-guardian))))) (add-hook! after-gc-hook reap-textures) (define (surface-pixel-format surface) "Returns the OpenGL pixel format for a surface. RGB and RGBA formats are supported." (case (SDL:surface:depth surface) ((24) (pixel-format* rgb)) ((32) (pixel-format* rgba)) (else (throw 'unsupported-pixel-format (SDL:surface:depth surface))))) (define (surface->texture surface) "Translates an SDL surface into an OpenGL texture. Currently only works with RGBA format surfaces." (let ((texture-id (gl-generate-texture)) (pixel-format (surface-pixel-format surface))) (with-gl-bind-texture (texture-target texture-2d) texture-id (gl-texture-parameter (texture-target texture-2d) (texture-parameter-name texture-min-filter) (texture-min-filter linear)) (gl-texture-parameter (texture-target texture-2d) (texture-parameter-name texture-mag-filter) (texture-mag-filter linear)) (gl-texture-image-2d (texture-target texture-2d) 0 pixel-format (SDL:surface:w surface) (SDL:surface:h surface) 0 pixel-format (color-pointer-type unsigned-byte) (SDL:surface-pixels surface))) (make-texture texture-id #f (SDL:surface:w surface) (SDL:surface:h surface) 0 0 1 1))) (define (load-texture filename) "Loads a texture from a file." (surface->texture (SDL:load-image filename))) (define* (draw-texture texture x y #:optional (color #xffffffff)) "Renders a textured quad in GL immediate mode." (let* ((x2 (+ x (texture-width texture))) (y2 (+ y (texture-height texture))) (color (rgba->gl-color color)) (r (vector-ref color 0)) (g (vector-ref color 1)) (b (vector-ref color 2)) (a (vector-ref color 3)) (s1 (texture-s1 texture)) (t1 (texture-t1 texture)) (s2 (texture-s2 texture)) (t2 (texture-t2 texture))) (with-gl-bind-texture (texture-target texture-2d) (texture-id texture) (gl-begin (begin-mode quads) (gl-color r g b a) (gl-texture-coordinates s1 t1) (gl-vertex x y) (gl-texture-coordinates s1 t2) (gl-vertex x y2) (gl-texture-coordinates s2 t2) (gl-vertex x2 y2) (gl-texture-coordinates s2 t1) (gl-vertex x2 y))))) (define* (split-texture texture width height #:optional #:key (margin 0) (spacing 0)) "Splits a texture into a vector of texture regions of width x height size." (define (build-tile tx ty) (let* ((x (+ (* tx (+ width spacing)) margin)) (y (+ (* ty (+ height spacing)) margin))) (make-texture-region texture x y width height))) (let* ((tw (texture-width texture)) (th (texture-height texture)) (rows (/ (- tw margin) (+ width spacing))) (columns (/ (- tw margin) (+ height spacing)))) (vector-ec (: y rows) (: x columns) (build-tile x y))))