-
Notifications
You must be signed in to change notification settings - Fork 35
/
Copy pathgif.lisp
101 lines (90 loc) · 4.26 KB
/
gif.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
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
;;; Copyright (c) 2011 Cyrus Harmon, All rights reserved.
;;; See COPYRIGHT file for details.
(in-package :opticl)
;;; Note: a GIF can contain multiple images. We're going to ignore
;;; this little detail and just return one image for the moment.
(defun skippy-image-to-8-bit-rgb-image (skippy-image global-color-table)
(with-accessors ((height skippy:height)
(width skippy:width)
(image-data skippy:image-data)
(image-color-table skippy:color-table)
(transparency-index skippy:transparency-index))
skippy-image
(let ((color-table (or image-color-table
global-color-table
(error "~@<Could not find color table for image ~A.~@:>"
skippy-image)))
(new-image (if transparency-index
(make-8-bit-rgba-image height width)
(make-8-bit-rgb-image height width))))
(set-pixels (i j) new-image
(let ((index (skippy:pixel-ref skippy-image j i)))
(if (eql index transparency-index)
(values 0 0 0 0)
(multiple-value-call #'values
(skippy:color-rgb
(skippy:color-table-entry color-table index))
#xff))))
new-image)))
(defun read-gif-stream (stream)
(let* ((data-stream (skippy:read-data-stream stream))
(color-table (skippy:color-table data-stream)))
(values-list
(loop for image across (skippy:images data-stream)
collect (skippy-image-to-8-bit-rgb-image image color-table)))))
(defun read-gif-file (pathname)
(with-open-file (stream pathname :direction :input :element-type '(unsigned-byte 8))
(read-gif-stream stream)))
(defun uniform-color-map (r-levels b-levels g-levels
&key (max-val 255))
(let ((array (make-array (* r-levels b-levels g-levels))))
(loop for r below r-levels
do (loop for g below g-levels
do (loop for b below b-levels
do
(setf (aref array (+ (* r g-levels b-levels)
(* g b-levels)
b))
(list (floor (* r (/ max-val (1- r-levels))))
(floor (* g (/ max-val (1- g-levels))))
(floor (* b (/ max-val (1- b-levels)))))))))
array))
(defun assign-color (r g b r-levels g-levels b-levels &key (max-val 255))
(values (* (round (/ r (/ max-val (1- r-levels))))
(/ max-val (1- r-levels)))
(* (round (/ g (/ max-val (1- g-levels))))
(/ max-val (1- g-levels)))
(* (round (/ b (/ max-val (1- b-levels))))
(/ max-val (1- b-levels)))))
(defparameter *red-levels* 6)
(defparameter *green-levels* 6)
(defparameter *blue-levels* 6)
(defun 8-bit-rgb-image-to-skippy-image (image color-table)
(with-image-bounds (height width)
image
(let ((gif-image (skippy:make-image :height height :width width)))
(do-pixels (i j) image
(multiple-value-bind (r g b)
(pixel image i j)
(multiple-value-bind (assigned-r assigned-g assigned-b)
(assign-color r g b *red-levels* *green-levels* *blue-levels*)
(let ((color-index (skippy:ensure-color
(skippy:rgb-color assigned-r assigned-g assigned-b)
color-table)))
(setf (skippy:pixel-ref gif-image j i) color-index)))))
gif-image)))
(defun write-gif-stream (stream image)
(with-image-bounds (height width)
image
(let* ((data-stream (skippy:make-data-stream
:height height :width width :color-table t))
(gif-image (8-bit-rgb-image-to-skippy-image image
(skippy:color-table data-stream))))
(skippy:add-image gif-image data-stream)
(skippy:write-data-stream data-stream stream))))
(defun write-gif-file (pathname image)
(with-open-file (stream pathname :direction :output
:if-exists :supersede
:element-type '(unsigned-byte 8))
(write-gif-stream stream image)
(truename pathname)))