-
-
Notifications
You must be signed in to change notification settings - Fork 30
/
tile-data.lisp
82 lines (73 loc) · 3.85 KB
/
tile-data.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
(in-package #:org.shirakumo.fraf.kandria)
(defclass tile-data (compiled-generator multi-resource-asset file-input-asset)
((tile-types :initform () :accessor tile-types)
(tile->bank :initform (make-hash-table :test 'eql) :accessor tile->bank)))
(defun tile-type-int (type)
(destructuring-bind (x y) type
(+ (ash x 8) y)))
(defun parse-tile-types (types)
(labels ((parse-tile-spec (spec)
(case (first spec)
(repeat
(destructuring-bind ((start end) tile) (rest spec)
(loop for x from start to end
collect (subst x 'x tile))))
(T (list spec))))
(parse-type-spec (spec)
(destructuring-bind (type . parts) spec
(list* type (reduce #'append parts :key #'parse-tile-spec)))))
(loop for (name . tiles) in types
collect (list* name (mapcar #'parse-type-spec tiles)))))
(defun compute-tile->bank-mapping (types)
(let ((table (make-hash-table :test 'eql)))
(loop for (bank . type-list) in types
do (dolist (type (rest type-list))
(dolist (tile (rest type))
(destructuring-bind (x y &optional (w 1) (h 1)) tile
(loop for i = x then (+ i (signum w))
until (= i (+ x w))
do (loop for j = y then (+ j (signum h))
until (= j (+ y h))
do (setf (gethash (tile-type-int (list i j)) table) bank)))))))
table))
(defmethod compile-resources ((data tile-data) (path pathname) &key force)
(destructuring-bind (&key source albedo absorption normal &allow-other-keys) (read-src path)
(let ((source (merge-pathnames source path))
(albedo (merge-pathnames albedo path))
(absorption (merge-pathnames absorption path))
(normal (merge-pathnames normal path)))
(when (or force (recompile-needed-p (list albedo absorption normal)
(list path source)))
(v:info :kandria.resources "Compiling tileset from ~a..." source)
(aseprite source "--layer" "albedo" "--save-as" albedo)
(optipng albedo)
(aseprite source "--layer" "absorption" "--save-as" absorption)
(img-convert absorption "-channel" "R" "-alpha" "off" "-set" "colorspace" "Gray" absorption)
(optipng absorption)
(aseprite source "--layer" "normal" "--save-as" normal)
(img-convert normal "-channel" "RGB" "-alpha" "off" "-depth" "8" (format NIL "png24:~a" (uiop:native-namestring normal)))
(optipng normal)))))
(defmethod generate-resources ((data tile-data) (path pathname) &key compile)
(destructuring-bind (&key source albedo absorption normal tile-types) (read-src path)
(let ((albedo (merge-pathnames albedo path))
(absorption (merge-pathnames absorption path))
(normal (merge-pathnames normal path)))
(setf (tile-types data) (parse-tile-types tile-types))
(setf (tile->bank data) (compute-tile->bank-mapping (tile-types data)))
(generate-resources 'image-loader albedo
:resource (resource data 'albedo))
(generate-resources 'image-loader absorption
:resource (resource data 'absorption))
(generate-resources 'image-loader normal
:resource (resource data 'normal))
(list (resource data 'albedo)
(resource data 'absorption)
(resource data 'normal)))))
(defmethod resource ((data tile-data) (name (eql T)))
(resource data 'albedo))
(defmethod notify:files-to-watch append ((asset tile-data))
(list (merge-pathnames (getf (read-src (input* asset)) :source) (input* asset))))
(defmethod notify:notify :before ((asset tile-data) file)
(when (string= "ase" (pathname-type file))
(sleep 1)
(compile-resources asset T)))