-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathgallery.content.lisp
239 lines (211 loc) · 7.78 KB
/
gallery.content.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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
(defpackage :gallery.content
(:use :cl-user :cl :files-locator :transliterate
:gallery.internal.pics-collection)
(:export #:period
#:period-begin
#:period-end
#:period-contains-p
#:adjust-album-period
#:make-embracing-period
#:item
#:item-id
#:item-owner-id
#:item-thumbnail
#:item-title
#:item-comment
#:item-time
#:picture
#:make-picture
#:pic-url
#:get-exif-time
#:album
#:make-album
#:make-root-album
#:album-name
#:album-items
#:album-delete-items))
(in-package :gallery.content)
(defclass period ()
((begin
:initarg :begin
:initform (local-time:now)
:type local-time:timestamp
:reader period-begin
:documentation
"The least moment in the time period")
(end
:initarg :end
:initform (local-time:now)
:type local-time:timestamp
:reader period-end
:documentation
"The most moment in the time period"))
(:documentation
"A time period - a segment of time. begin is timestamp< than the end."))
(defgeneric find-newest-moment (t1 other-times)
(:documentation "Get the most recent moment of n+1 times or time periods"))
(defgeneric find-oldest-moment (t1 other-times)
(:documentation "Get the most ancient moment of n+1 times or time periods"))
(defgeneric period-contains-one-p (p time)
(:documentation "Check whether the period p contains the time"))
(defun make-embracing-period (times)
"Make the shortest period, contatining t1 and t2"
(make-instance 'period :begin (find-oldest-moment (car times) (cdr times))
:end (find-newest-moment (car times) (cdr times))))
(defun period-contains-p (p &rest times)
"Check whether the period p contains all the given times"
(if (null times)
t
(every #'(lambda (time) (period-contains-one-p p time)) times)))
(defmethod find-newest-moment ((t1 period) (other-times list))
(find-newest-moment (period-end t1) other-times))
(defmethod find-newest-moment ((t1 local-time:timestamp) (other-times list))
(if (null other-times)
t1
(let ((t2 (find-newest-moment (car other-times) (cdr other-times))))
(if (local-time:timestamp< t1 t2) t2 t1))))
(defmethod find-oldest-moment ((t1 period) (other-times list))
(find-oldest-moment (period-begin t1) other-times))
(defmethod find-oldest-moment ((t1 local-time:timestamp) (other-times list))
(if (null other-times)
t1
(let ((t2 (find-oldest-moment (car other-times) (cdr other-times))))
(if (local-time:timestamp< t1 t2) t1 t2))))
(defmethod period-contains-one-p ((p period) (time period))
(and (period-contains-one-p p (period-begin time))
(period-contains-one-p p (period-end time))))
(defmethod period-contains-one-p ((p period) (time local-time:timestamp))
(and (local-time:timestamp< (period-begin p) time)
(local-time:timestamp< time (period-end p))))
(defclass item ()
((id
:initarg :id
:reader item-id
:documentation
"An unique identificator for the item")
(owner-id
:initarg :owner-id
:reader item-owner-id
:documentation
"An id of the container, owning the given item")
(thumbnail
:initarg :thumbnail
:reader item-thumbnail
:documentation
"The url to the small preview picture")
(title
:initarg :title
:accessor item-title)
(comment
:initarg :comment
:accessor item-comment
:documentation
"The commentary to the content.")
(time
:initarg :time
:reader item-time
:documentation
"A time moment or period, when the item took place"))
(:documentation
"A general item, representing a gallery-managed item"))
(defclass picture (item)
((url
:initarg :url
:reader pic-url
:documentation
"The address of the actual full-size content")
(time
:type local-time:timestamp)))
(defclass album (item)
((name
:initarg :name
:reader album-name
:documentation
"The uniq string, used to designate the album among others.")
(items
:initform nil
:initarg :items
:accessor album-items
:documentation
"The collection of all items, contained in the album")
(time
:type period)))
(defun gen-small-pic-fname (fname)
(format nil "~a.thumb.~a" (subseq fname 0 (- (length fname) 4))
(subseq fname (- (length fname) 3) (length fname))));reattach the extension
(defun make-thumb (store fname)
(let ((small-fname (gen-small-pic-fname fname)))
(sb-ext:run-program "/usr/bin/convert"
(list "-scale" "100x100"
(file-pathname store fname)
(file-pathname store small-fname))
:wait t)
small-fname))
(defun get-exif-time (file &optional (default (local-time:now)))
(let ((exif-str (with-output-to-string (out)
(sb-ext:run-program "/usr/bin/exiftool"
(list "-CreateDate"
file)
:output out))))
(if (or (string= "" exif-str)
(< (length exif-str) 53))
default
(local-time:parse-timestring
exif-str
:date-time-separator #\Space :allow-missing-elements t
:date-separator #\: :start 34 :end 53))))
(defun make-picture (store owner-id file title comment date)
(make-instance 'picture
:id (gen-uniq-id-pic-coll)
:owner-id owner-id
:url (file-url store file)
:thumbnail (file-url store (make-thumb store file))
:title title
:comment comment
:time date))
(defun make-album-name (title)
(transliterate (string-downcase (string-trim " " title))))
(defun make-album (store owner-id file title comment period)
(make-instance 'album
:id (gen-uniq-id-pic-coll)
:owner-id owner-id
:name (make-album-name title)
:title title
:comment comment
:thumbnail (file-url store (make-thumb store file))
:time period))
(defun make-root-album (title comment)
(make-instance 'album
:id (gen-uniq-id-pic-coll)
:owner-id nil
:name (make-album-name title)
:title title
:comment comment
:thumbnail nil
:time (make-instance 'period)))
;; TODO: if a time of any item is ajacent to the album period border,
;; recalculate album perioid
(defun album-delete-items (album ids)
(setf (album-items album)
(remove-if #'(lambda (item)
(find (item-id item) ids :test #'equal))
(album-items album))))
(defmethod find-oldest-moment ((i item) other-times)
(find-oldest-moment (item-time i) other-times))
(defmethod find-newest-moment ((i item) other-times)
(find-newest-moment (item-time i) other-times))
(defmethod find-oldest-moment ((i string) other-times)
(find-oldest-moment (local-time:parse-timestring i) other-times))
(defmethod find-newest-moment ((i string) other-times)
(find-newest-moment (local-time:parse-timestring i) other-times))
(defun adjust-direct-album-period (album period)
(setf (slot-value album 'time)
(make-embracing-period (list (slot-value album 'time) period))))
(defun adjust-album-period (item-updater album period)
(if (null (album-items album))
(progn
(setf (slot-value album 'time) period)
(funcall item-updater album))
(unless (period-contains-p (item-time album) period)
(adjust-direct-album-period album period)
(funcall item-updater album))))