-
Notifications
You must be signed in to change notification settings - Fork 9
/
style.clj
306 lines (245 loc) · 11.1 KB
/
style.clj
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
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
(ns excel-clj.style
"The basic unit of spreadsheet data is the cell, which can be embellished
with style data, e.g.
{:data-format :percent,
:font {:bold true :font-height-in-points 10}}
The goal of the style map is to reuse all of the functionality built in to
the underlying Apache POI objects, but with immutable data structures.
The primary advantage is the ease with which we can merge styles as maps
rather than trying to create some new POI object out of two other objects,
reading and combining all of their attributes and nested attributes.
## Mechanics
Style map data are representative of nested calls to the corresponding setter
methods in the Apache POI framework starting with a `CellStyle` object. That
is, the above example is roughly interpreted as:
;; A CellStyle POI object created under the hood during rendering
(let [cell-style ...]
;; The style map attributes are converted to camel cased setters
(doto cell-style
(.setDataFormat :percent)
(.setFont
(doto <create a font>
(.setBold true)
(.setFontHeightInPoints 10)))))
The two nontrivial challenges are
- creating nested objects, e.g. (.setFont cell-style <font>) needs to be
called with a POI Font object; and
- translating keywords like :percent to POI objects.
Both are solved with the `coerce-to-obj` multimethod specifying how to
coerce different attributes to POI objects, which has the shape
(fn [workbook attribute value] => Object)
and dispatches on the `attribute` (a keyword).
We coerce key value pairs to objects from the bottom of the style map upwards,
meaning that by the time coerce-to-obj is being invoked for some attribute,
any nested attributes in the value have already been coerced.
A more nuanced representation of how the style map 'expands':
;; {:data-format :percent, :font {:bold true :font-height-in-points 10}}}
;; expands to
(let [cell-style ..., workbook ...] ;; POI objects created during rendering
(doto cell-style
(.setDataFormat (coerce-to-obj workbook :data-format :percent))
;; The {:bold true :font-height-in-points 10} expands recursively
(.setFont
(coerce-to-obj
workbook :font {:bold true :font-height-in-points 10}))))
"
{:author "Matthew Downey"}
(:require [clojure.string :as string])
(:import (org.apache.poi.ss.usermodel
DataFormat BorderStyle HorizontalAlignment VerticalAlignment
FillPatternType Workbook VerticalAlignment FontUnderline)
(org.apache.poi.xssf.usermodel XSSFColor DefaultIndexedColorMap
XSSFCellStyle XSSFFont XSSFWorkbook)))
;;; Code to allow specification of Excel CellStyle objects as nested maps. You
;;; might touch this code to add an implementation of `coerce-to-obj` for some
;;; cell style attribute.
(defn- do-set!
"Set an attribute on a Java object & return the object. E.g.
(let [attr :font-height-in-points]
(do-set! some-obj attr 14))
;; Equivalent to (doto some-obj (.setFontHeightInPoints 14))"
[obj attr val]
(let [cap (fn [coll] (map string/capitalize coll))
camel (fn [kw]
(str "set" (-> (name kw) (string/split #"\W") cap string/join)))
setter (eval (read-string (format "(memfn %s arg)" (-> attr camel))))]
(doto obj
(setter val))))
(defn- do-set-all! [base-object attributes]
(reduce-kv do-set! base-object attributes))
(defmulti coerce-to-obj
"For some keyword attribute of a CellStyle object, attempt to coerce clojure
data (either a keyword or a map) to the Java object the setter is expecting.
This allows nesting of style specification maps
{:font {:bold true, :color :yellow}}
so that when it's time to generate a CellStyle object, we can say that we
know how to go from an attribute map to a Font object for :font attributes,
from a keyword to a Color object for :color attributes, etc."
(fn [^Workbook workbook attr-keyword value]
attr-keyword))
;; Coercions from simple map lookups
(defmacro ^:private coerce-from-map
([attr-keyword coercion-map]
`(coerce-from-map ~attr-keyword ~coercion-map (fn [a# b# val#] val#)))
([attr-keyword coercion-map otherwise]
`(defmethod coerce-to-obj ~attr-keyword
[wb# akw# val#]
(if (keyword? val#)
(or
(get ~coercion-map val#)
(-> ~(str "No " attr-keyword " registered.")
(ex-info {:given val# :have (keys ~coercion-map)})
(throw)))
(~otherwise wb# akw# val#)))))
(def alignments
{:general HorizontalAlignment/GENERAL
:left HorizontalAlignment/LEFT
:center HorizontalAlignment/CENTER
:right HorizontalAlignment/RIGHT
:fill HorizontalAlignment/FILL
:justify HorizontalAlignment/JUSTIFY
:center-selection HorizontalAlignment/CENTER_SELECTION
:distributed HorizontalAlignment/DISTRIBUTED})
(def valignments
{:top VerticalAlignment/TOP
:center VerticalAlignment/CENTER
:bottom VerticalAlignment/BOTTOM
:justify VerticalAlignment/JUSTIFY
:distributed VerticalAlignment/DISTRIBUTED})
(def underlines
{:single FontUnderline/SINGLE
:single-accounting FontUnderline/SINGLE_ACCOUNTING
:double FontUnderline/DOUBLE
:double-accounting FontUnderline/DOUBLE_ACCOUNTING
:none FontUnderline/NONE})
(def borders
{:none BorderStyle/NONE
:thin BorderStyle/THIN
:medium BorderStyle/MEDIUM
:dashed BorderStyle/DASHED
:dotted BorderStyle/DOTTED
:thick BorderStyle/THICK
:double BorderStyle/DOUBLE
:hair BorderStyle/HAIR
:medium-dashed BorderStyle/MEDIUM_DASHED
:dash-dot BorderStyle/DASH_DOT
:medium-dash-dot BorderStyle/MEDIUM_DASH_DOT
:dash-dot-dot BorderStyle/DASH_DOT_DOT
:medium-dash-dot-dot BorderStyle/MEDIUM_DASH_DOT_DOT
:slanted-dash-dot BorderStyle/SLANTED_DASH_DOT})
(def fill-patterns
{:no-fill FillPatternType/NO_FILL
:solid-foreground FillPatternType/SOLID_FOREGROUND
:fine-dots FillPatternType/FINE_DOTS
:alt-bars FillPatternType/ALT_BARS
:sparse-dots FillPatternType/SPARSE_DOTS
:thick-horz-bands FillPatternType/THICK_HORZ_BANDS
:thick-vert-bands FillPatternType/THICK_VERT_BANDS
:thick-backward-diag FillPatternType/THICK_BACKWARD_DIAG
:thick-forward-diag FillPatternType/THICK_FORWARD_DIAG
:big-spots FillPatternType/BIG_SPOTS
:bricks FillPatternType/BRICKS
:thin-horz-bands FillPatternType/THIN_HORZ_BANDS
:thin-vert-bands FillPatternType/THIN_VERT_BANDS
:thin-backward-diag FillPatternType/THIN_BACKWARD_DIAG
:thin-forward-diag FillPatternType/THIN_FORWARD_DIAG
:squares FillPatternType/SQUARES
:diamonds FillPatternType/DIAMONDS
:less_dots FillPatternType/LESS_DOTS
:least_dots FillPatternType/LEAST_DOTS})
(def data-formats
{:accounting "_($* #,##0.00_);_($* (#,##0.00);_($* \"-\"??_);_(@_)"
:number "#.###############"
:ymd "yyyy-MM-dd"
:datetime "yyyy-MM-dd hh:mm:ss"
:percent "0.00%"})
(defn ^XSSFColor rgb-color
"Create an XSSFColor object from the given r g b values."
[r g b]
(XSSFColor. (byte-array [r g b]) (DefaultIndexedColorMap.)))
(def colors
{:white (rgb-color 255 255 255)
:red (rgb-color 255 0 0)
:orange (rgb-color 255 127 0)
:yellow (rgb-color 250 255 204)
:green (rgb-color 221 255 204)
:blue (rgb-color 204 255 255)
:purple (rgb-color 200 0 255)
:gray (rgb-color 232 232 232)
:black (rgb-color 0 0 0)})
(coerce-from-map :alignment alignments)
(coerce-from-map :vertical-alignment valignments)
(coerce-from-map :underline underlines)
(coerce-from-map :border-top borders)
(coerce-from-map :border-left borders)
(coerce-from-map :border-right borders)
(coerce-from-map :border-bottom borders)
(coerce-from-map :fill-pattern fill-patterns)
(letfn [(if-color-not-found [_ _ color]
(if (and (coll? color) (= (count color) 3))
(apply rgb-color color)
(-> "Can only create colors from rgb three-tuples or keywords."
(ex-info {:given color})
(throw))))]
(coerce-from-map :color colors if-color-not-found)
(coerce-from-map :fill-background-color colors if-color-not-found)
(coerce-from-map :fill-foreground-color colors if-color-not-found)
(coerce-from-map :left-border-color colors if-color-not-found)
(coerce-from-map :right-border-color colors if-color-not-found)
(coerce-from-map :top-border-color colors if-color-not-found)
(coerce-from-map :bottom-border-color colors if-color-not-found))
(defmethod coerce-to-obj :font
[^Workbook wb _ font-attrs]
(do-set-all! ^XSSFFont (.createFont wb) font-attrs))
(defmethod coerce-to-obj :data-format
[^Workbook wb _ format]
(if (instance? DataFormat format)
format
(if-let [format' (cond->> format (keyword? format) (get data-formats))]
(let [ch (.getCreationHelper wb)]
(.getFormat ^DataFormat (.createDataFormat ch) (str format')))
(-> "Can't coerce to data format."
(ex-info {:given format :have (keys data-formats)})
(throw)))))
(defmethod coerce-to-obj :default
[_ _ x] x)
(defn- coerce-nested-to-obj
"Given an attribute map, start at the most nested layer and work upwards,
attempting to coerce each attribute to an object."
[wb attributes]
(letfn [(coerce-nested [av-pairs rebuilt]
(if-let [[a v] (first av-pairs)]
(if-not (map? v)
(let [coerced (coerce-to-obj wb a v)]
(recur (rest av-pairs) (assoc rebuilt a coerced)))
#(coerce-nested
(rest av-pairs)
(->>
(coerce-to-obj wb a (trampoline coerce-nested (seq v) {}))
(assoc rebuilt a))))
rebuilt))]
(trampoline coerce-nested attributes {})))
(defn build-style
"Create a CellStyle from the given attrs using the given workbook
CellStyle attributes are anything that can be set with
.setCamelCasedAttribute on a CellStyle object, including
{:data-format string or keyword
:font { ... font attrs ... }
:wrap-text boolean
:hidden boolean
:alignment org.apache.poi.ss.usermodel.HorizontalAlignment
:border-[bottom|left|right|top] org.apache.poi.ss.usermodel.BorderStyle}
Any of the attributes can be java objects. Alternatively, if a `coerce-to-obj`
implementation is provided for some attribute (e.g. :font), the attribute can
be specified as data."
[^Workbook workbook attrs]
(let [attrs' (coerce-nested-to-obj ^XSSFWorkbook workbook attrs)]
(try
(do-set-all! ^XSSFCellStyle (.createCellStyle workbook) attrs')
(catch Exception e
(-> "Failed to create cell style."
(ex-info {:raw-attributes attrs :built-attributes attrs'} e)
(throw))))))
(def default-style
"The default cell style."
{:font {:font-height-in-points 10 :font-name "Arial"}})