]> granicus.if.org Git - imagemagick/blob - MagickCore/color.c
(no commit message)
[imagemagick] / MagickCore / color.c
1 /*
2 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3 %                                                                             %
4 %                                                                             %
5 %                       CCCC   OOO   L       OOO   RRRR                       %
6 %                      C      O   O  L      O   O  R   R                      %
7 %                      C      O   O  L      O   O  RRRR                       %
8 %                      C      O   O  L      O   O  R R                        %
9 %                       CCCC   OOO   LLLLL   OOO   R  R                       %
10 %                                                                             %
11 %                                                                             %
12 %                          MagickCore Color Methods                           %
13 %                                                                             %
14 %                              Software Design                                %
15 %                                   Cristy                                    %
16 %                                 July 1992                                   %
17 %                                                                             %
18 %                                                                             %
19 %  Copyright 1999-2014 ImageMagick Studio LLC, a non-profit organization      %
20 %  dedicated to making software imaging solutions freely available.           %
21 %                                                                             %
22 %  You may not use this file except in compliance with the License.  You may  %
23 %  obtain a copy of the License at                                            %
24 %                                                                             %
25 %    http://www.imagemagick.org/script/license.php                            %
26 %                                                                             %
27 %  Unless required by applicable law or agreed to in writing, software        %
28 %  distributed under the License is distributed on an "AS IS" BASIS,          %
29 %  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.   %
30 %  See the License for the specific language governing permissions and        %
31 %  limitations under the License.                                             %
32 %                                                                             %
33 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
34 %
35 %  We use linked-lists because splay-trees do not currently support duplicate
36 %  key / value pairs (.e.g X11 green compliance and SVG green compliance).
37 %
38 */
39 \f
40 /*
41   Include declarations.
42 */
43 #include "MagickCore/studio.h"
44 #include "MagickCore/blob.h"
45 #include "MagickCore/cache-view.h"
46 #include "MagickCore/cache.h"
47 #include "MagickCore/color.h"
48 #include "MagickCore/color-private.h"
49 #include "MagickCore/colorspace-private.h"
50 #include "MagickCore/client.h"
51 #include "MagickCore/configure.h"
52 #include "MagickCore/exception.h"
53 #include "MagickCore/exception-private.h"
54 #include "MagickCore/gem.h"
55 #include "MagickCore/gem-private.h"
56 #include "MagickCore/geometry.h"
57 #include "MagickCore/image-private.h"
58 #include "MagickCore/memory_.h"
59 #include "MagickCore/monitor.h"
60 #include "MagickCore/monitor-private.h"
61 #include "MagickCore/option.h"
62 #include "MagickCore/pixel-accessor.h"
63 #include "MagickCore/quantize.h"
64 #include "MagickCore/quantum.h"
65 #include "MagickCore/quantum-private.h"
66 #include "MagickCore/semaphore.h"
67 #include "MagickCore/string_.h"
68 #include "MagickCore/token.h"
69 #include "MagickCore/utility.h"
70 #include "MagickCore/utility-private.h"
71 #include "MagickCore/xml-tree.h"
72 \f
73 /*
74   Define declarations.
75 */
76 #define ColorFilename  "colors.xml"
77 \f
78 /*
79   Typedef declarations.
80 */
81 typedef struct _ColorMapInfo
82 {
83   const char
84     *name;
85
86   const unsigned char
87     red,
88     green,
89     blue;
90
91   const float
92     alpha;
93
94   const ssize_t
95     compliance;
96 } ColorMapInfo;
97 \f
98 /*
99   Static declarations.
100 */
101 static const ColorMapInfo
102   ColorMap[] =
103   {
104     { "none", 0, 0, 0, 0, SVGCompliance | XPMCompliance },
105     { "black", 0, 0, 0, 1, SVGCompliance | X11Compliance | XPMCompliance },
106     { "red", 255, 0, 0, 1, SVGCompliance | X11Compliance | XPMCompliance },
107     { "magenta", 255, 0, 255, 1, SVGCompliance | X11Compliance | XPMCompliance },
108     { "green", 0, 128, 0, 1, SVGCompliance },
109     { "cyan", 0, 255, 255, 1, SVGCompliance | X11Compliance | XPMCompliance },
110     { "blue", 0, 0, 255, 1, SVGCompliance | X11Compliance | XPMCompliance },
111     { "yellow", 255, 255, 0, 1, SVGCompliance | X11Compliance | XPMCompliance },
112     { "white", 255, 255, 255, 1, SVGCompliance | X11Compliance | XPMCompliance },
113     { "AliceBlue", 240, 248, 255, 1, SVGCompliance | X11Compliance | XPMCompliance },
114     { "AntiqueWhite", 250, 235, 215, 1, SVGCompliance | X11Compliance | XPMCompliance },
115     { "AntiqueWhite1", 255, 239, 219, 1, X11Compliance },
116     { "AntiqueWhite2", 238, 223, 204, 1, X11Compliance },
117     { "AntiqueWhite3", 205, 192, 176, 1, X11Compliance },
118     { "AntiqueWhite4", 139, 131, 120, 1, X11Compliance },
119     { "aqua", 0, 255, 255, 1, SVGCompliance },
120     { "aquamarine", 127, 255, 212, 1, SVGCompliance | X11Compliance | XPMCompliance },
121     { "aquamarine1", 127, 255, 212, 1, X11Compliance },
122     { "aquamarine2", 118, 238, 198, 1, X11Compliance },
123     { "aquamarine3", 102, 205, 170, 1, X11Compliance },
124     { "aquamarine4", 69, 139, 116, 1, X11Compliance },
125     { "azure", 240, 255, 255, 1, SVGCompliance | X11Compliance | XPMCompliance },
126     { "azure1", 240, 255, 255, 1, X11Compliance },
127     { "azure2", 224, 238, 238, 1, X11Compliance },
128     { "azure3", 193, 205, 205, 1, X11Compliance },
129     { "azure4", 131, 139, 139, 1, X11Compliance },
130     { "beige", 245, 245, 220, 1, SVGCompliance | X11Compliance | XPMCompliance },
131     { "bisque", 255, 228, 196, 1, SVGCompliance | X11Compliance | XPMCompliance },
132     { "bisque1", 255, 228, 196, 1, X11Compliance },
133     { "bisque2", 238, 213, 183, 1, X11Compliance },
134     { "bisque3", 205, 183, 158, 1, X11Compliance },
135     { "bisque4", 139, 125, 107, 1, X11Compliance },
136     { "BlanchedAlmond", 255, 235, 205, 1, SVGCompliance | X11Compliance | XPMCompliance },
137     { "blue1", 0, 0, 255, 1, X11Compliance },
138     { "blue2", 0, 0, 238, 1, X11Compliance },
139     { "blue3", 0, 0, 205, 1, X11Compliance },
140     { "blue4", 0, 0, 139, 1, X11Compliance },
141     { "BlueViolet", 138, 43, 226, 1, SVGCompliance | X11Compliance | XPMCompliance },
142     { "brown", 165, 42, 42, 1, SVGCompliance | X11Compliance | XPMCompliance },
143     { "brown1", 255, 64, 64, 1, X11Compliance },
144     { "brown2", 238, 59, 59, 1, X11Compliance },
145     { "brown3", 205, 51, 51, 1, X11Compliance },
146     { "brown4", 139, 35, 35, 1, X11Compliance },
147     { "burlywood", 222, 184, 135, 1, SVGCompliance | X11Compliance | XPMCompliance },
148     { "burlywood1", 255, 211, 155, 1, X11Compliance },
149     { "burlywood2", 238, 197, 145, 1, X11Compliance },
150     { "burlywood3", 205, 170, 125, 1, X11Compliance },
151     { "burlywood4", 139, 115, 85, 1, X11Compliance },
152     { "CadetBlue", 95, 158, 160, 1, SVGCompliance | X11Compliance | XPMCompliance },
153     { "CadetBlue1", 152, 245, 255, 1, X11Compliance },
154     { "CadetBlue2", 142, 229, 238, 1, X11Compliance },
155     { "CadetBlue3", 122, 197, 205, 1, X11Compliance },
156     { "CadetBlue4", 83, 134, 139, 1, X11Compliance },
157     { "chartreuse", 127, 255, 0, 1, SVGCompliance | X11Compliance | XPMCompliance },
158     { "chartreuse1", 127, 255, 0, 1, X11Compliance },
159     { "chartreuse2", 118, 238, 0, 1, X11Compliance },
160     { "chartreuse3", 102, 205, 0, 1, X11Compliance },
161     { "chartreuse4", 69, 139, 0, 1, X11Compliance },
162     { "chocolate", 210, 105, 30, 1, SVGCompliance | X11Compliance | XPMCompliance },
163     { "chocolate1", 255, 127, 36, 1, X11Compliance },
164     { "chocolate2", 238, 118, 33, 1, X11Compliance },
165     { "chocolate3", 205, 102, 29, 1, X11Compliance },
166     { "chocolate4", 139, 69, 19, 1, X11Compliance },
167     { "coral", 255, 127, 80, 1, SVGCompliance | X11Compliance | XPMCompliance },
168     { "coral1", 255, 114, 86, 1, X11Compliance },
169     { "coral2", 238, 106, 80, 1, X11Compliance },
170     { "coral3", 205, 91, 69, 1, X11Compliance },
171     { "coral4", 139, 62, 47, 1, X11Compliance },
172     { "CornflowerBlue", 100, 149, 237, 1, SVGCompliance | X11Compliance | XPMCompliance },
173     { "cornsilk", 255, 248, 220, 1, SVGCompliance | X11Compliance | XPMCompliance },
174     { "cornsilk1", 255, 248, 220, 1, X11Compliance },
175     { "cornsilk2", 238, 232, 205, 1, X11Compliance },
176     { "cornsilk3", 205, 200, 177, 1, X11Compliance },
177     { "cornsilk4", 139, 136, 120, 1, X11Compliance },
178     { "crimson", 220, 20, 60, 1, SVGCompliance },
179     { "cyan1", 0, 255, 255, 1, X11Compliance },
180     { "cyan2", 0, 238, 238, 1, X11Compliance },
181     { "cyan3", 0, 205, 205, 1, X11Compliance },
182     { "cyan4", 0, 139, 139, 1, X11Compliance },
183     { "DarkBlue", 0, 0, 139, 1, SVGCompliance | X11Compliance },
184     { "DarkCyan", 0, 139, 139, 1, SVGCompliance | X11Compliance },
185     { "DarkGoldenrod", 184, 134, 11, 1, SVGCompliance | X11Compliance | XPMCompliance },
186     { "DarkGoldenrod1", 255, 185, 15, 1, X11Compliance },
187     { "DarkGoldenrod2", 238, 173, 14, 1, X11Compliance },
188     { "DarkGoldenrod3", 205, 149, 12, 1, X11Compliance },
189     { "DarkGoldenrod4", 139, 101, 8, 1, X11Compliance },
190     { "DarkGray", 169, 169, 169, 1, SVGCompliance | X11Compliance },
191     { "DarkGreen", 0, 100, 0, 1, SVGCompliance | X11Compliance | XPMCompliance },
192     { "DarkGrey", 169, 169, 169, 1, SVGCompliance | X11Compliance },
193     { "DarkKhaki", 189, 183, 107, 1, SVGCompliance | X11Compliance | XPMCompliance },
194     { "DarkMagenta", 139, 0, 139, 1, SVGCompliance | X11Compliance },
195     { "DarkOliveGreen", 85, 107, 47, 1, SVGCompliance | X11Compliance | XPMCompliance },
196     { "DarkOliveGreen1", 202, 255, 112, 1, X11Compliance },
197     { "DarkOliveGreen2", 188, 238, 104, 1, X11Compliance },
198     { "DarkOliveGreen3", 162, 205, 90, 1, X11Compliance },
199     { "DarkOliveGreen4", 110, 139, 61, 1, X11Compliance },
200     { "DarkOrange", 255, 140, 0, 1, SVGCompliance | X11Compliance | XPMCompliance },
201     { "DarkOrange1", 255, 127, 0, 1, X11Compliance },
202     { "DarkOrange2", 238, 118, 0, 1, X11Compliance },
203     { "DarkOrange3", 205, 102, 0, 1, X11Compliance },
204     { "DarkOrange4", 139, 69, 0, 1, X11Compliance },
205     { "DarkOrchid", 153, 50, 204, 1, SVGCompliance | X11Compliance | XPMCompliance },
206     { "DarkOrchid1", 191, 62, 255, 1, X11Compliance },
207     { "DarkOrchid2", 178, 58, 238, 1, X11Compliance },
208     { "DarkOrchid3", 154, 50, 205, 1, X11Compliance },
209     { "DarkOrchid4", 104, 34, 139, 1, X11Compliance },
210     { "DarkRed", 139, 0, 0, 1, SVGCompliance | X11Compliance },
211     { "DarkSalmon", 233, 150, 122, 1, SVGCompliance | X11Compliance | XPMCompliance },
212     { "DarkSeaGreen", 143, 188, 143, 1, SVGCompliance | X11Compliance | XPMCompliance },
213     { "DarkSeaGreen1", 193, 255, 193, 1, X11Compliance },
214     { "DarkSeaGreen2", 180, 238, 180, 1, X11Compliance },
215     { "DarkSeaGreen3", 155, 205, 155, 1, X11Compliance },
216     { "DarkSeaGreen4", 105, 139, 105, 1, X11Compliance },
217     { "DarkSlateBlue", 72, 61, 139, 1, SVGCompliance | X11Compliance | XPMCompliance },
218     { "DarkSlateGray", 47, 79, 79, 1, SVGCompliance | X11Compliance | XPMCompliance },
219     { "DarkSlateGray1", 151, 255, 255, 1, X11Compliance },
220     { "DarkSlateGray2", 141, 238, 238, 1, X11Compliance },
221     { "DarkSlateGray3", 121, 205, 205, 1, X11Compliance },
222     { "DarkSlateGray4", 82, 139, 139, 1, X11Compliance },
223     { "DarkSlateGrey", 47, 79, 79, 1, SVGCompliance | X11Compliance },
224     { "DarkTurquoise", 0, 206, 209, 1, SVGCompliance | X11Compliance | XPMCompliance },
225     { "DarkViolet", 148, 0, 211, 1, SVGCompliance | X11Compliance | XPMCompliance },
226     { "DeepPink", 255, 20, 147, 1, SVGCompliance | X11Compliance | XPMCompliance },
227     { "DeepPink1", 255, 20, 147, 1, X11Compliance },
228     { "DeepPink2", 238, 18, 137, 1, X11Compliance },
229     { "DeepPink3", 205, 16, 118, 1, X11Compliance },
230     { "DeepPink4", 139, 10, 80, 1, X11Compliance },
231     { "DeepSkyBlue", 0, 191, 255, 1, SVGCompliance | X11Compliance | XPMCompliance },
232     { "DeepSkyBlue1", 0, 191, 255, 1, X11Compliance },
233     { "DeepSkyBlue2", 0, 178, 238, 1, X11Compliance },
234     { "DeepSkyBlue3", 0, 154, 205, 1, X11Compliance },
235     { "DeepSkyBlue4", 0, 104, 139, 1, X11Compliance },
236     { "DimGray", 105, 105, 105, 1, SVGCompliance | X11Compliance | XPMCompliance },
237     { "DimGrey", 105, 105, 105, 1, SVGCompliance | X11Compliance },
238     { "DodgerBlue", 30, 144, 255, 1, SVGCompliance | X11Compliance | XPMCompliance },
239     { "DodgerBlue1", 30, 144, 255, 1, X11Compliance },
240     { "DodgerBlue2", 28, 134, 238, 1, X11Compliance },
241     { "DodgerBlue3", 24, 116, 205, 1, X11Compliance },
242     { "DodgerBlue4", 16, 78, 139, 1, X11Compliance },
243     { "firebrick", 178, 34, 34, 1, SVGCompliance | X11Compliance | XPMCompliance },
244     { "firebrick1", 255, 48, 48, 1, X11Compliance },
245     { "firebrick2", 238, 44, 44, 1, X11Compliance },
246     { "firebrick3", 205, 38, 38, 1, X11Compliance },
247     { "firebrick4", 139, 26, 26, 1, X11Compliance },
248     { "FloralWhite", 255, 250, 240, 1, SVGCompliance | X11Compliance | XPMCompliance },
249     { "ForestGreen", 34, 139, 34, 1, SVGCompliance | X11Compliance | XPMCompliance },
250     { "fractal", 128, 128, 128, 1, SVGCompliance },
251     { "freeze", 0, 0, 0, 0, SVGCompliance },
252     { "fuchsia", 255, 0, 255, 1, SVGCompliance },
253     { "gainsboro", 220, 220, 220, 1, SVGCompliance | X11Compliance | XPMCompliance },
254     { "GhostWhite", 248, 248, 255, 1, SVGCompliance | X11Compliance | XPMCompliance },
255     { "gold", 255, 215, 0, 1, SVGCompliance | X11Compliance | XPMCompliance },
256     { "gold1", 255, 215, 0, 1, X11Compliance },
257     { "gold2", 238, 201, 0, 1, X11Compliance },
258     { "gold3", 205, 173, 0, 1, X11Compliance },
259     { "gold4", 139, 117, 0, 1, X11Compliance },
260     { "goldenrod", 218, 165, 32, 1, SVGCompliance | X11Compliance | XPMCompliance },
261     { "goldenrod1", 255, 193, 37, 1, X11Compliance },
262     { "goldenrod2", 238, 180, 34, 1, X11Compliance },
263     { "goldenrod3", 205, 155, 29, 1, X11Compliance },
264     { "goldenrod4", 139, 105, 20, 1, X11Compliance },
265     { "gray", 126, 126, 126, 1, SVGCompliance },
266     { "gray", 190, 190, 190, 1, X11Compliance | XPMCompliance },
267     { "gray0", 0, 0, 0, 1, X11Compliance | XPMCompliance },
268     { "gray1", 3, 3, 3, 1, X11Compliance | XPMCompliance },
269     { "gray10", 26, 26, 26, 1, X11Compliance | XPMCompliance },
270     { "gray100", 255, 255, 255, 1, X11Compliance | XPMCompliance },
271     { "gray100", 255, 255, 255, 1, X11Compliance | XPMCompliance },
272     { "gray11", 28, 28, 28, 1, X11Compliance | XPMCompliance },
273     { "gray12", 31, 31, 31, 1, X11Compliance | XPMCompliance },
274     { "gray13", 33, 33, 33, 1, X11Compliance | XPMCompliance },
275     { "gray14", 36, 36, 36, 1, X11Compliance | XPMCompliance },
276     { "gray15", 38, 38, 38, 1, X11Compliance | XPMCompliance },
277     { "gray16", 41, 41, 41, 1, X11Compliance | XPMCompliance },
278     { "gray17", 43, 43, 43, 1, X11Compliance | XPMCompliance },
279     { "gray18", 46, 46, 46, 1, X11Compliance | XPMCompliance },
280     { "gray19", 48, 48, 48, 1, X11Compliance | XPMCompliance },
281     { "gray2", 5, 5, 5, 1, X11Compliance | XPMCompliance },
282     { "gray20", 51, 51, 51, 1, X11Compliance | XPMCompliance },
283     { "gray21", 54, 54, 54, 1, X11Compliance | XPMCompliance },
284     { "gray22", 56, 56, 56, 1, X11Compliance | XPMCompliance },
285     { "gray23", 59, 59, 59, 1, X11Compliance | XPMCompliance },
286     { "gray24", 61, 61, 61, 1, X11Compliance | XPMCompliance },
287     { "gray25", 64, 64, 64, 1, X11Compliance | XPMCompliance },
288     { "gray26", 66, 66, 66, 1, X11Compliance | XPMCompliance },
289     { "gray27", 69, 69, 69, 1, X11Compliance | XPMCompliance },
290     { "gray28", 71, 71, 71, 1, X11Compliance | XPMCompliance },
291     { "gray29", 74, 74, 74, 1, X11Compliance | XPMCompliance },
292     { "gray3", 8, 8, 8, 1, X11Compliance | XPMCompliance },
293     { "gray30", 77, 77, 77, 1, X11Compliance | XPMCompliance },
294     { "gray31", 79, 79, 79, 1, X11Compliance | XPMCompliance },
295     { "gray32", 82, 82, 82, 1, X11Compliance | XPMCompliance },
296     { "gray33", 84, 84, 84, 1, X11Compliance | XPMCompliance },
297     { "gray34", 87, 87, 87, 1, X11Compliance | XPMCompliance },
298     { "gray35", 89, 89, 89, 1, X11Compliance | XPMCompliance },
299     { "gray36", 92, 92, 92, 1, X11Compliance | XPMCompliance },
300     { "gray37", 94, 94, 94, 1, X11Compliance | XPMCompliance },
301     { "gray38", 97, 97, 97, 1, X11Compliance | XPMCompliance },
302     { "gray39", 99, 99, 99, 1, X11Compliance | XPMCompliance },
303     { "gray4", 10, 10, 10, 1, X11Compliance | XPMCompliance },
304     { "gray40", 102, 102, 102, 1, X11Compliance | XPMCompliance },
305     { "gray41", 105, 105, 105, 1, X11Compliance | XPMCompliance },
306     { "gray42", 107, 107, 107, 1, X11Compliance | XPMCompliance },
307     { "gray43", 110, 110, 110, 1, X11Compliance | XPMCompliance },
308     { "gray44", 112, 112, 112, 1, X11Compliance | XPMCompliance },
309     { "gray45", 115, 115, 115, 1, X11Compliance | XPMCompliance },
310     { "gray46", 117, 117, 117, 1, X11Compliance | XPMCompliance },
311     { "gray47", 120, 120, 120, 1, X11Compliance | XPMCompliance },
312     { "gray48", 122, 122, 122, 1, X11Compliance | XPMCompliance },
313     { "gray49", 125, 125, 125, 1, X11Compliance | XPMCompliance },
314     { "gray5", 13, 13, 13, 1, X11Compliance | XPMCompliance },
315     { "gray50", 127, 127, 127, 1, X11Compliance | XPMCompliance },
316     { "gray51", 130, 130, 130, 1, X11Compliance | XPMCompliance },
317     { "gray52", 133, 133, 133, 1, X11Compliance | XPMCompliance },
318     { "gray53", 135, 135, 135, 1, X11Compliance | XPMCompliance },
319     { "gray54", 138, 138, 138, 1, X11Compliance | XPMCompliance },
320     { "gray55", 140, 140, 140, 1, X11Compliance | XPMCompliance },
321     { "gray56", 143, 143, 143, 1, X11Compliance | XPMCompliance },
322     { "gray57", 145, 145, 145, 1, X11Compliance | XPMCompliance },
323     { "gray58", 148, 148, 148, 1, X11Compliance | XPMCompliance },
324     { "gray59", 150, 150, 150, 1, X11Compliance | XPMCompliance },
325     { "gray6", 15, 15, 15, 1, X11Compliance | XPMCompliance },
326     { "gray60", 153, 153, 153, 1, X11Compliance | XPMCompliance },
327     { "gray61", 156, 156, 156, 1, X11Compliance | XPMCompliance },
328     { "gray62", 158, 158, 158, 1, X11Compliance | XPMCompliance },
329     { "gray63", 161, 161, 161, 1, X11Compliance | XPMCompliance },
330     { "gray64", 163, 163, 163, 1, X11Compliance | XPMCompliance },
331     { "gray65", 166, 166, 166, 1, X11Compliance | XPMCompliance },
332     { "gray66", 168, 168, 168, 1, X11Compliance | XPMCompliance },
333     { "gray67", 171, 171, 171, 1, X11Compliance | XPMCompliance },
334     { "gray68", 173, 173, 173, 1, X11Compliance | XPMCompliance },
335     { "gray69", 176, 176, 176, 1, X11Compliance | XPMCompliance },
336     { "gray7", 18, 18, 18, 1, X11Compliance | XPMCompliance },
337     { "gray70", 179, 179, 179, 1, X11Compliance | XPMCompliance },
338     { "gray71", 181, 181, 181, 1, X11Compliance | XPMCompliance },
339     { "gray72", 184, 184, 184, 1, X11Compliance | XPMCompliance },
340     { "gray73", 186, 186, 186, 1, X11Compliance | XPMCompliance },
341     { "gray74", 189, 189, 189, 1, X11Compliance | XPMCompliance },
342     { "gray75", 191, 191, 191, 1, X11Compliance | XPMCompliance },
343     { "gray76", 194, 194, 194, 1, X11Compliance | XPMCompliance },
344     { "gray77", 196, 196, 196, 1, X11Compliance | XPMCompliance },
345     { "gray78", 199, 199, 199, 1, X11Compliance | XPMCompliance },
346     { "gray79", 201, 201, 201, 1, X11Compliance | XPMCompliance },
347     { "gray8", 20, 20, 20, 1, X11Compliance | XPMCompliance },
348     { "gray80", 204, 204, 204, 1, X11Compliance | XPMCompliance },
349     { "gray81", 207, 207, 207, 1, X11Compliance | XPMCompliance },
350     { "gray82", 209, 209, 209, 1, X11Compliance | XPMCompliance },
351     { "gray83", 212, 212, 212, 1, X11Compliance | XPMCompliance },
352     { "gray84", 214, 214, 214, 1, X11Compliance | XPMCompliance },
353     { "gray85", 217, 217, 217, 1, X11Compliance | XPMCompliance },
354     { "gray86", 219, 219, 219, 1, X11Compliance | XPMCompliance },
355     { "gray87", 222, 222, 222, 1, X11Compliance | XPMCompliance },
356     { "gray88", 224, 224, 224, 1, X11Compliance | XPMCompliance },
357     { "gray89", 227, 227, 227, 1, X11Compliance | XPMCompliance },
358     { "gray9", 23, 23, 23, 1, X11Compliance | XPMCompliance },
359     { "gray90", 229, 229, 229, 1, X11Compliance | XPMCompliance },
360     { "gray91", 232, 232, 232, 1, X11Compliance | XPMCompliance },
361     { "gray92", 235, 235, 235, 1, X11Compliance | XPMCompliance },
362     { "gray93", 237, 237, 237, 1, X11Compliance | XPMCompliance },
363     { "gray94", 240, 240, 240, 1, X11Compliance | XPMCompliance },
364     { "gray95", 242, 242, 242, 1, X11Compliance | XPMCompliance },
365     { "gray96", 245, 245, 245, 1, X11Compliance | XPMCompliance },
366     { "gray97", 247, 247, 247, 1, X11Compliance | XPMCompliance },
367     { "gray98", 250, 250, 250, 1, X11Compliance | XPMCompliance },
368     { "gray99", 252, 252, 252, 1, X11Compliance | XPMCompliance },
369     { "green", 0, 255, 0, 1, X11Compliance | XPMCompliance },
370     { "green1", 0, 255, 0, 1, X11Compliance },
371     { "green2", 0, 238, 0, 1, X11Compliance },
372     { "green3", 0, 205, 0, 1, X11Compliance },
373     { "green4", 0, 139, 0, 1, X11Compliance },
374     { "GreenYellow", 173, 255, 47, 1, X11Compliance | XPMCompliance },
375     { "grey", 190, 190, 190, 1, SVGCompliance | X11Compliance },
376     { "grey0", 0, 0, 0, 1, SVGCompliance | X11Compliance },
377     { "grey1", 3, 3, 3, 1, SVGCompliance | X11Compliance },
378     { "grey10", 26, 26, 26, 1, SVGCompliance | X11Compliance },
379     { "grey100", 255, 255, 255, 1, SVGCompliance | X11Compliance },
380     { "grey11", 28, 28, 28, 1, SVGCompliance | X11Compliance },
381     { "grey12", 31, 31, 31, 1, SVGCompliance | X11Compliance },
382     { "grey13", 33, 33, 33, 1, SVGCompliance | X11Compliance },
383     { "grey14", 36, 36, 36, 1, SVGCompliance | X11Compliance },
384     { "grey15", 38, 38, 38, 1, SVGCompliance | X11Compliance },
385     { "grey16", 41, 41, 41, 1, SVGCompliance | X11Compliance },
386     { "grey17", 43, 43, 43, 1, SVGCompliance | X11Compliance },
387     { "grey18", 46, 46, 46, 1, SVGCompliance | X11Compliance },
388     { "grey19", 48, 48, 48, 1, SVGCompliance | X11Compliance },
389     { "grey2", 5, 5, 5, 1, SVGCompliance | X11Compliance },
390     { "grey20", 51, 51, 51, 1, SVGCompliance | X11Compliance },
391     { "grey21", 54, 54, 54, 1, SVGCompliance | X11Compliance },
392     { "grey22", 56, 56, 56, 1, SVGCompliance | X11Compliance },
393     { "grey23", 59, 59, 59, 1, SVGCompliance | X11Compliance },
394     { "grey24", 61, 61, 61, 1, SVGCompliance | X11Compliance },
395     { "grey25", 64, 64, 64, 1, SVGCompliance | X11Compliance },
396     { "grey26", 66, 66, 66, 1, SVGCompliance | X11Compliance },
397     { "grey27", 69, 69, 69, 1, SVGCompliance | X11Compliance },
398     { "grey28", 71, 71, 71, 1, SVGCompliance | X11Compliance },
399     { "grey29", 74, 74, 74, 1, SVGCompliance | X11Compliance },
400     { "grey3", 8, 8, 8, 1, SVGCompliance | X11Compliance },
401     { "grey30", 77, 77, 77, 1, SVGCompliance | X11Compliance },
402     { "grey31", 79, 79, 79, 1, SVGCompliance | X11Compliance },
403     { "grey32", 82, 82, 82, 1, SVGCompliance | X11Compliance },
404     { "grey33", 84, 84, 84, 1, SVGCompliance | X11Compliance },
405     { "grey34", 87, 87, 87, 1, SVGCompliance | X11Compliance },
406     { "grey35", 89, 89, 89, 1, SVGCompliance | X11Compliance },
407     { "grey36", 92, 92, 92, 1, SVGCompliance | X11Compliance },
408     { "grey37", 94, 94, 94, 1, SVGCompliance | X11Compliance },
409     { "grey38", 97, 97, 97, 1, SVGCompliance | X11Compliance },
410     { "grey39", 99, 99, 99, 1, SVGCompliance | X11Compliance },
411     { "grey4", 10, 10, 10, 1, SVGCompliance | X11Compliance },
412     { "grey40", 102, 102, 102, 1, SVGCompliance | X11Compliance },
413     { "grey41", 105, 105, 105, 1, SVGCompliance | X11Compliance },
414     { "grey42", 107, 107, 107, 1, SVGCompliance | X11Compliance },
415     { "grey43", 110, 110, 110, 1, SVGCompliance | X11Compliance },
416     { "grey44", 112, 112, 112, 1, SVGCompliance | X11Compliance },
417     { "grey45", 115, 115, 115, 1, SVGCompliance | X11Compliance },
418     { "grey46", 117, 117, 117, 1, SVGCompliance | X11Compliance },
419     { "grey47", 120, 120, 120, 1, SVGCompliance | X11Compliance },
420     { "grey48", 122, 122, 122, 1, SVGCompliance | X11Compliance },
421     { "grey49", 125, 125, 125, 1, SVGCompliance | X11Compliance },
422     { "grey5", 13, 13, 13, 1, SVGCompliance | X11Compliance },
423     { "grey50", 127, 127, 127, 1, SVGCompliance | X11Compliance },
424     { "grey51", 130, 130, 130, 1, SVGCompliance | X11Compliance },
425     { "grey52", 133, 133, 133, 1, SVGCompliance | X11Compliance },
426     { "grey53", 135, 135, 135, 1, SVGCompliance | X11Compliance },
427     { "grey54", 138, 138, 138, 1, SVGCompliance | X11Compliance },
428     { "grey55", 140, 140, 140, 1, SVGCompliance | X11Compliance },
429     { "grey56", 143, 143, 143, 1, SVGCompliance | X11Compliance },
430     { "grey57", 145, 145, 145, 1, SVGCompliance | X11Compliance },
431     { "grey58", 148, 148, 148, 1, SVGCompliance | X11Compliance },
432     { "grey59", 150, 150, 150, 1, SVGCompliance | X11Compliance },
433     { "grey6", 15, 15, 15, 1, SVGCompliance | X11Compliance },
434     { "grey60", 153, 153, 153, 1, SVGCompliance | X11Compliance },
435     { "grey61", 156, 156, 156, 1, SVGCompliance | X11Compliance },
436     { "grey62", 158, 158, 158, 1, SVGCompliance | X11Compliance },
437     { "grey63", 161, 161, 161, 1, SVGCompliance | X11Compliance },
438     { "grey64", 163, 163, 163, 1, SVGCompliance | X11Compliance },
439     { "grey65", 166, 166, 166, 1, SVGCompliance | X11Compliance },
440     { "grey66", 168, 168, 168, 1, SVGCompliance | X11Compliance },
441     { "grey67", 171, 171, 171, 1, SVGCompliance | X11Compliance },
442     { "grey68", 173, 173, 173, 1, SVGCompliance | X11Compliance },
443     { "grey69", 176, 176, 176, 1, SVGCompliance | X11Compliance },
444     { "grey7", 18, 18, 18, 1, SVGCompliance | X11Compliance },
445     { "grey70", 179, 179, 179, 1, SVGCompliance | X11Compliance },
446     { "grey71", 181, 181, 181, 1, SVGCompliance | X11Compliance },
447     { "grey72", 184, 184, 184, 1, SVGCompliance | X11Compliance },
448     { "grey73", 186, 186, 186, 1, SVGCompliance | X11Compliance },
449     { "grey74", 189, 189, 189, 1, SVGCompliance | X11Compliance },
450     { "grey75", 191, 191, 191, 1, SVGCompliance | X11Compliance },
451     { "grey76", 194, 194, 194, 1, SVGCompliance | X11Compliance },
452     { "grey77", 196, 196, 196, 1, SVGCompliance | X11Compliance },
453     { "grey78", 199, 199, 199, 1, SVGCompliance | X11Compliance },
454     { "grey79", 201, 201, 201, 1, SVGCompliance | X11Compliance },
455     { "grey8", 20, 20, 20, 1, SVGCompliance | X11Compliance },
456     { "grey80", 204, 204, 204, 1, SVGCompliance | X11Compliance },
457     { "grey81", 207, 207, 207, 1, SVGCompliance | X11Compliance },
458     { "grey82", 209, 209, 209, 1, SVGCompliance | X11Compliance },
459     { "grey83", 212, 212, 212, 1, SVGCompliance | X11Compliance },
460     { "grey84", 214, 214, 214, 1, SVGCompliance | X11Compliance },
461     { "grey85", 217, 217, 217, 1, SVGCompliance | X11Compliance },
462     { "grey86", 219, 219, 219, 1, SVGCompliance | X11Compliance },
463     { "grey87", 222, 222, 222, 1, SVGCompliance | X11Compliance },
464     { "grey88", 224, 224, 224, 1, SVGCompliance | X11Compliance },
465     { "grey89", 227, 227, 227, 1, SVGCompliance | X11Compliance },
466     { "grey9", 23, 23, 23, 1, SVGCompliance | X11Compliance },
467     { "grey90", 229, 229, 229, 1, SVGCompliance | X11Compliance },
468     { "grey91", 232, 232, 232, 1, SVGCompliance | X11Compliance },
469     { "grey92", 235, 235, 235, 1, SVGCompliance | X11Compliance },
470     { "grey93", 237, 237, 237, 1, SVGCompliance | X11Compliance },
471     { "grey94", 240, 240, 240, 1, SVGCompliance | X11Compliance },
472     { "grey95", 242, 242, 242, 1, SVGCompliance | X11Compliance },
473     { "grey96", 245, 245, 245, 1, SVGCompliance | X11Compliance },
474     { "grey97", 247, 247, 247, 1, SVGCompliance | X11Compliance },
475     { "grey98", 250, 250, 250, 1, SVGCompliance | X11Compliance },
476     { "grey99", 252, 252, 252, 1, SVGCompliance | X11Compliance },
477     { "honeydew", 240, 255, 240, 1, SVGCompliance | X11Compliance | XPMCompliance },
478     { "honeydew1", 240, 255, 240, 1, X11Compliance },
479     { "honeydew2", 224, 238, 224, 1, X11Compliance },
480     { "honeydew3", 193, 205, 193, 1, X11Compliance },
481     { "honeydew4", 131, 139, 131, 1, X11Compliance },
482     { "HotPink", 255, 105, 180, 1, SVGCompliance | X11Compliance | XPMCompliance },
483     { "HotPink1", 255, 110, 180, 1, X11Compliance },
484     { "HotPink2", 238, 106, 167, 1, X11Compliance },
485     { "HotPink3", 205, 96, 144, 1, X11Compliance },
486     { "HotPink4", 139, 58, 98, 1, X11Compliance },
487     { "IndianRed", 205, 92, 92, 1, SVGCompliance | X11Compliance | XPMCompliance },
488     { "IndianRed1", 255, 106, 106, 1, X11Compliance },
489     { "IndianRed2", 238, 99, 99, 1, X11Compliance },
490     { "IndianRed3", 205, 85, 85, 1, X11Compliance },
491     { "IndianRed4", 139, 58, 58, 1, X11Compliance },
492     { "indigo", 75, 0, 130, 1, SVGCompliance },
493     { "ivory", 255, 255, 240, 1, SVGCompliance | X11Compliance | XPMCompliance },
494     { "ivory1", 255, 255, 240, 1, X11Compliance },
495     { "ivory2", 238, 238, 224, 1, X11Compliance },
496     { "ivory3", 205, 205, 193, 1, X11Compliance },
497     { "ivory4", 139, 139, 131, 1, X11Compliance },
498     { "khaki", 240, 230, 140, 1, SVGCompliance | X11Compliance | XPMCompliance },
499     { "khaki1", 255, 246, 143, 1, X11Compliance },
500     { "khaki2", 238, 230, 133, 1, X11Compliance },
501     { "khaki3", 205, 198, 115, 1, X11Compliance },
502     { "khaki4", 139, 134, 78, 1, X11Compliance },
503     { "lavender", 230, 230, 250, 1, SVGCompliance | X11Compliance | XPMCompliance },
504     { "LavenderBlush", 255, 240, 245, 1, SVGCompliance | X11Compliance | XPMCompliance },
505     { "LavenderBlush1", 255, 240, 245, 1, X11Compliance },
506     { "LavenderBlush2", 238, 224, 229, 1, X11Compliance },
507     { "LavenderBlush3", 205, 193, 197, 1, X11Compliance },
508     { "LavenderBlush4", 139, 131, 134, 1, X11Compliance },
509     { "LawnGreen", 124, 252, 0, 1, SVGCompliance | X11Compliance | XPMCompliance },
510     { "LemonChiffon", 255, 250, 205, 1, SVGCompliance | X11Compliance | XPMCompliance },
511     { "LemonChiffon1", 255, 250, 205, 1, X11Compliance },
512     { "LemonChiffon2", 238, 233, 191, 1, X11Compliance },
513     { "LemonChiffon3", 205, 201, 165, 1, X11Compliance },
514     { "LemonChiffon4", 139, 137, 112, 1, X11Compliance },
515     { "LightBlue", 173, 216, 230, 1, SVGCompliance | X11Compliance | XPMCompliance },
516     { "LightBlue1", 191, 239, 255, 1, X11Compliance },
517     { "LightBlue2", 178, 223, 238, 1, X11Compliance },
518     { "LightBlue3", 154, 192, 205, 1, X11Compliance },
519     { "LightBlue4", 104, 131, 139, 1, X11Compliance },
520     { "LightCoral", 240, 128, 128, 1, SVGCompliance | X11Compliance | XPMCompliance },
521     { "LightCyan", 224, 255, 255, 1, SVGCompliance | X11Compliance | XPMCompliance },
522     { "LightCyan1", 224, 255, 255, 1, X11Compliance },
523     { "LightCyan2", 209, 238, 238, 1, X11Compliance },
524     { "LightCyan3", 180, 205, 205, 1, X11Compliance },
525     { "LightCyan4", 122, 139, 139, 1, X11Compliance },
526     { "LightGoldenrod", 238, 221, 130, 1, X11Compliance | XPMCompliance },
527     { "LightGoldenrod1", 255, 236, 139, 1, X11Compliance },
528     { "LightGoldenrod2", 238, 220, 130, 1, X11Compliance },
529     { "LightGoldenrod3", 205, 190, 112, 1, X11Compliance },
530     { "LightGoldenrod4", 139, 129, 76, 1, X11Compliance },
531     { "LightGoldenrodYellow", 250, 250, 210, 1, SVGCompliance | X11Compliance | XPMCompliance },
532     { "LightGray", 211, 211, 211, 1, SVGCompliance | X11Compliance | XPMCompliance },
533     { "LightGreen", 144, 238, 144, 1, SVGCompliance | X11Compliance },
534     { "LightGrey", 211, 211, 211, 1, SVGCompliance | X11Compliance },
535     { "LightPink", 255, 182, 193, 1, SVGCompliance | X11Compliance | XPMCompliance },
536     { "LightPink1", 255, 174, 185, 1, X11Compliance },
537     { "LightPink2", 238, 162, 173, 1, X11Compliance },
538     { "LightPink3", 205, 140, 149, 1, X11Compliance },
539     { "LightPink4", 139, 95, 101, 1, X11Compliance },
540     { "LightSalmon", 255, 160, 122, 1, SVGCompliance | X11Compliance | XPMCompliance },
541     { "LightSalmon1", 255, 160, 122, 1, X11Compliance },
542     { "LightSalmon2", 238, 149, 114, 1, X11Compliance },
543     { "LightSalmon3", 205, 129, 98, 1, X11Compliance },
544     { "LightSalmon4", 139, 87, 66, 1, X11Compliance },
545     { "LightSeaGreen", 32, 178, 170, 1, SVGCompliance | X11Compliance | XPMCompliance },
546     { "LightSkyBlue", 135, 206, 250, 1, SVGCompliance | X11Compliance | XPMCompliance },
547     { "LightSkyBlue1", 176, 226, 255, 1, X11Compliance },
548     { "LightSkyBlue2", 164, 211, 238, 1, X11Compliance },
549     { "LightSkyBlue3", 141, 182, 205, 1, X11Compliance },
550     { "LightSkyBlue4", 96, 123, 139, 1, X11Compliance },
551     { "LightSlateBlue", 132, 112, 255, 1, X11Compliance | XPMCompliance },
552     { "LightSlateGray", 119, 136, 153, 1, SVGCompliance | X11Compliance | XPMCompliance },
553     { "LightSlateGrey", 119, 136, 153, 1, SVGCompliance | X11Compliance },
554     { "LightSteelBlue", 176, 196, 222, 1, SVGCompliance | X11Compliance | XPMCompliance },
555     { "LightSteelBlue1", 202, 225, 255, 1, X11Compliance },
556     { "LightSteelBlue2", 188, 210, 238, 1, X11Compliance },
557     { "LightSteelBlue3", 162, 181, 205, 1, X11Compliance },
558     { "LightSteelBlue4", 110, 123, 139, 1, X11Compliance },
559     { "LightYellow", 255, 255, 224, 1, SVGCompliance | X11Compliance | XPMCompliance },
560     { "LightYellow1", 255, 255, 224, 1, X11Compliance },
561     { "LightYellow2", 238, 238, 209, 1, X11Compliance },
562     { "LightYellow3", 205, 205, 180, 1, X11Compliance },
563     { "LightYellow4", 139, 139, 122, 1, X11Compliance },
564     { "lime", 0, 255, 0, 1, SVGCompliance },
565     { "LimeGreen", 50, 205, 50, 1, SVGCompliance | X11Compliance | XPMCompliance },
566     { "linen", 250, 240, 230, 1, SVGCompliance | X11Compliance | XPMCompliance },
567     { "magenta1", 255, 0, 255, 1, X11Compliance },
568     { "magenta2", 238, 0, 238, 1, X11Compliance },
569     { "magenta3", 205, 0, 205, 1, X11Compliance },
570     { "magenta4", 139, 0, 139, 1, X11Compliance },
571     { "maroon", 128, 0, 0, 1, SVGCompliance },
572     { "maroon", 176, 48, 96, 1, X11Compliance | XPMCompliance },
573     { "maroon1", 255, 52, 179, 1, X11Compliance },
574     { "maroon2", 238, 48, 167, 1, X11Compliance },
575     { "maroon3", 205, 41, 144, 1, X11Compliance },
576     { "maroon4", 139, 28, 98, 1, X11Compliance },
577     { "MediumAquamarine", 102, 205, 170, 1, SVGCompliance | X11Compliance | XPMCompliance },
578     { "MediumBlue", 0, 0, 205, 1, SVGCompliance | X11Compliance | XPMCompliance },
579     { "MediumForestGreen", 50, 129, 75, 1, X11Compliance | XPMCompliance },
580     { "MediumGoldenRod", 209, 193, 102, 1, X11Compliance | XPMCompliance },
581     { "MediumOrchid", 186, 85, 211, 1, SVGCompliance | X11Compliance | XPMCompliance },
582     { "MediumOrchid1", 224, 102, 255, 1, X11Compliance },
583     { "MediumOrchid2", 209, 95, 238, 1, X11Compliance },
584     { "MediumOrchid3", 180, 82, 205, 1, X11Compliance },
585     { "MediumOrchid4", 122, 55, 139, 1, X11Compliance },
586     { "MediumPurple", 147, 112, 219, 1, SVGCompliance | X11Compliance | XPMCompliance },
587     { "MediumPurple1", 171, 130, 255, 1, X11Compliance },
588     { "MediumPurple2", 159, 121, 238, 1, X11Compliance },
589     { "MediumPurple3", 137, 104, 205, 1, X11Compliance },
590     { "MediumPurple4", 93, 71, 139, 1, X11Compliance },
591     { "MediumSeaGreen", 60, 179, 113, 1, SVGCompliance | X11Compliance | XPMCompliance },
592     { "MediumSlateBlue", 123, 104, 238, 1, SVGCompliance | X11Compliance | XPMCompliance },
593     { "MediumSpringGreen", 0, 250, 154, 1, SVGCompliance | X11Compliance | XPMCompliance },
594     { "MediumTurquoise", 72, 209, 204, 1, SVGCompliance | X11Compliance | XPMCompliance },
595     { "MediumVioletRed", 199, 21, 133, 1, SVGCompliance | X11Compliance | XPMCompliance },
596     { "MidnightBlue", 25, 25, 112, 1, SVGCompliance | X11Compliance | XPMCompliance },
597     { "MintCream", 245, 255, 250, 1, SVGCompliance | X11Compliance | XPMCompliance },
598     { "MistyRose", 255, 228, 225, 1, SVGCompliance | X11Compliance | XPMCompliance },
599     { "MistyRose1", 255, 228, 225, 1, X11Compliance },
600     { "MistyRose2", 238, 213, 210, 1, X11Compliance },
601     { "MistyRose3", 205, 183, 181, 1, X11Compliance },
602     { "MistyRose4", 139, 125, 123, 1, X11Compliance },
603     { "moccasin", 255, 228, 181, 1, SVGCompliance | X11Compliance | XPMCompliance },
604     { "NavajoWhite", 255, 222, 173, 1, SVGCompliance | X11Compliance | XPMCompliance },
605     { "NavajoWhite1", 255, 222, 173, 1, X11Compliance },
606     { "NavajoWhite2", 238, 207, 161, 1, X11Compliance },
607     { "NavajoWhite3", 205, 179, 139, 1, X11Compliance },
608     { "NavajoWhite4", 139, 121, 94, 1, X11Compliance },
609     { "navy", 0, 0, 128, 1, SVGCompliance | X11Compliance | XPMCompliance },
610     { "NavyBlue", 0, 0, 128, 1, X11Compliance | XPMCompliance },
611     { "matte", 0, 0, 0, 0, SVGCompliance },
612     { "OldLace", 253, 245, 230, 1, SVGCompliance | X11Compliance | XPMCompliance },
613     { "olive", 128, 128, 0, 1, SVGCompliance },
614     { "OliveDrab", 107, 142, 35, 1, SVGCompliance | X11Compliance | XPMCompliance },
615     { "OliveDrab1", 192, 255, 62, 1, X11Compliance },
616     { "OliveDrab2", 179, 238, 58, 1, X11Compliance },
617     { "OliveDrab3", 154, 205, 50, 1, X11Compliance },
618     { "OliveDrab4", 105, 139, 34, 1, X11Compliance },
619     { "opaque", 0, 0, 0, 1, SVGCompliance },
620     { "orange", 255, 165, 0, 1, SVGCompliance | X11Compliance | XPMCompliance },
621     { "orange1", 255, 165, 0, 1, X11Compliance },
622     { "orange2", 238, 154, 0, 1, X11Compliance },
623     { "orange3", 205, 133, 0, 1, X11Compliance },
624     { "orange4", 139, 90, 0, 1, X11Compliance },
625     { "OrangeRed", 255, 69, 0, 1, SVGCompliance | X11Compliance | XPMCompliance },
626     { "OrangeRed1", 255, 69, 0, 1, X11Compliance },
627     { "OrangeRed2", 238, 64, 0, 1, X11Compliance },
628     { "OrangeRed3", 205, 55, 0, 1, X11Compliance },
629     { "OrangeRed4", 139, 37, 0, 1, X11Compliance },
630     { "orchid", 218, 112, 214, 1, SVGCompliance | X11Compliance | XPMCompliance },
631     { "orchid1", 255, 131, 250, 1, X11Compliance },
632     { "orchid2", 238, 122, 233, 1, X11Compliance },
633     { "orchid3", 205, 105, 201, 1, X11Compliance },
634     { "orchid4", 139, 71, 137, 1, X11Compliance },
635     { "PaleGoldenrod", 238, 232, 170, 1, SVGCompliance | X11Compliance | XPMCompliance },
636     { "PaleGreen", 152, 251, 152, 1, SVGCompliance | X11Compliance | XPMCompliance },
637     { "PaleGreen1", 154, 255, 154, 1, X11Compliance },
638     { "PaleGreen2", 144, 238, 144, 1, X11Compliance },
639     { "PaleGreen3", 124, 205, 124, 1, X11Compliance },
640     { "PaleGreen4", 84, 139, 84, 1, X11Compliance },
641     { "PaleTurquoise", 175, 238, 238, 1, SVGCompliance | X11Compliance | XPMCompliance },
642     { "PaleTurquoise1", 187, 255, 255, 1, X11Compliance },
643     { "PaleTurquoise2", 174, 238, 238, 1, X11Compliance },
644     { "PaleTurquoise3", 150, 205, 205, 1, X11Compliance },
645     { "PaleTurquoise4", 102, 139, 139, 1, X11Compliance },
646     { "PaleVioletRed", 219, 112, 147, 1, SVGCompliance | X11Compliance | XPMCompliance },
647     { "PaleVioletRed1", 255, 130, 171, 1, X11Compliance },
648     { "PaleVioletRed2", 238, 121, 159, 1, X11Compliance },
649     { "PaleVioletRed3", 205, 104, 137, 1, X11Compliance },
650     { "PaleVioletRed4", 139, 71, 93, 1, X11Compliance },
651     { "PapayaWhip", 255, 239, 213, 1, SVGCompliance | X11Compliance | XPMCompliance },
652     { "PeachPuff", 255, 218, 185, 1, SVGCompliance | X11Compliance | XPMCompliance },
653     { "PeachPuff1", 255, 218, 185, 1, X11Compliance },
654     { "PeachPuff2", 238, 203, 173, 1, X11Compliance },
655     { "PeachPuff3", 205, 175, 149, 1, X11Compliance },
656     { "PeachPuff4", 139, 119, 101, 1, X11Compliance },
657     { "peru", 205, 133, 63, 1, SVGCompliance | X11Compliance | XPMCompliance },
658     { "pink", 255, 192, 203, 1, SVGCompliance | X11Compliance | XPMCompliance },
659     { "pink1", 255, 181, 197, 1, X11Compliance },
660     { "pink2", 238, 169, 184, 1, X11Compliance },
661     { "pink3", 205, 145, 158, 1, X11Compliance },
662     { "pink4", 139, 99, 108, 1, X11Compliance },
663     { "plum", 221, 160, 221, 1, SVGCompliance | X11Compliance | XPMCompliance },
664     { "plum1", 255, 187, 255, 1, X11Compliance },
665     { "plum2", 238, 174, 238, 1, X11Compliance },
666     { "plum3", 205, 150, 205, 1, X11Compliance },
667     { "plum4", 139, 102, 139, 1, X11Compliance },
668     { "PowderBlue", 176, 224, 230, 1, SVGCompliance | X11Compliance | XPMCompliance },
669     { "purple", 128, 0, 128, 1, SVGCompliance },
670     { "purple", 160, 32, 240, 1, X11Compliance | XPMCompliance },
671     { "purple1", 155, 48, 255, 1, X11Compliance },
672     { "purple2", 145, 44, 238, 1, X11Compliance },
673     { "purple3", 125, 38, 205, 1, X11Compliance },
674     { "purple4", 85, 26, 139, 1, X11Compliance },
675     { "red1", 255, 0, 0, 1, X11Compliance },
676     { "red2", 238, 0, 0, 1, X11Compliance },
677     { "red3", 205, 0, 0, 1, X11Compliance },
678     { "red4", 139, 0, 0, 1, X11Compliance },
679     { "RosyBrown", 188, 143, 143, 1, SVGCompliance | X11Compliance | XPMCompliance },
680     { "RosyBrown1", 255, 193, 193, 1, X11Compliance },
681     { "RosyBrown2", 238, 180, 180, 1, X11Compliance },
682     { "RosyBrown3", 205, 155, 155, 1, X11Compliance },
683     { "RosyBrown4", 139, 105, 105, 1, X11Compliance },
684     { "RoyalBlue", 65, 105, 225, 1, SVGCompliance | X11Compliance | XPMCompliance },
685     { "RoyalBlue1", 72, 118, 255, 1, X11Compliance },
686     { "RoyalBlue2", 67, 110, 238, 1, X11Compliance },
687     { "RoyalBlue3", 58, 95, 205, 1, X11Compliance },
688     { "RoyalBlue4", 39, 64, 139, 1, X11Compliance },
689     { "SaddleBrown", 139, 69, 19, 1, SVGCompliance | X11Compliance | XPMCompliance },
690     { "salmon", 250, 128, 114, 1, SVGCompliance | X11Compliance | XPMCompliance },
691     { "salmon1", 255, 140, 105, 1, X11Compliance },
692     { "salmon2", 238, 130, 98, 1, X11Compliance },
693     { "salmon3", 205, 112, 84, 1, X11Compliance },
694     { "salmon4", 139, 76, 57, 1, X11Compliance },
695     { "SandyBrown", 244, 164, 96, 1, SVGCompliance | X11Compliance | XPMCompliance },
696     { "SeaGreen", 46, 139, 87, 1, SVGCompliance | X11Compliance | XPMCompliance },
697     { "SeaGreen1", 84, 255, 159, 1, X11Compliance },
698     { "SeaGreen2", 78, 238, 148, 1, X11Compliance },
699     { "SeaGreen3", 67, 205, 128, 1, X11Compliance },
700     { "SeaGreen4", 46, 139, 87, 1, X11Compliance },
701     { "seashell", 255, 245, 238, 1, SVGCompliance | X11Compliance | XPMCompliance },
702     { "seashell1", 255, 245, 238, 1, X11Compliance },
703     { "seashell2", 238, 229, 222, 1, X11Compliance },
704     { "seashell3", 205, 197, 191, 1, X11Compliance },
705     { "seashell4", 139, 134, 130, 1, X11Compliance },
706     { "sienna", 160, 82, 45, 1, SVGCompliance | X11Compliance | XPMCompliance },
707     { "sienna1", 255, 130, 71, 1, X11Compliance },
708     { "sienna2", 238, 121, 66, 1, X11Compliance },
709     { "sienna3", 205, 104, 57, 1, X11Compliance },
710     { "sienna4", 139, 71, 38, 1, X11Compliance },
711     { "silver", 192, 192, 192, 1, SVGCompliance },
712     { "SkyBlue", 135, 206, 235, 1, SVGCompliance | X11Compliance | XPMCompliance },
713     { "SkyBlue1", 135, 206, 255, 1, X11Compliance },
714     { "SkyBlue2", 126, 192, 238, 1, X11Compliance },
715     { "SkyBlue3", 108, 166, 205, 1, X11Compliance },
716     { "SkyBlue4", 74, 112, 139, 1, X11Compliance },
717     { "SlateBlue", 106, 90, 205, 1, SVGCompliance | X11Compliance | XPMCompliance },
718     { "SlateBlue1", 131, 111, 255, 1, X11Compliance },
719     { "SlateBlue2", 122, 103, 238, 1, X11Compliance },
720     { "SlateBlue3", 105, 89, 205, 1, X11Compliance },
721     { "SlateBlue4", 71, 60, 139, 1, X11Compliance },
722     { "SlateGray", 112, 128, 144, 1, SVGCompliance | X11Compliance | XPMCompliance },
723     { "SlateGray1", 198, 226, 255, 1, X11Compliance },
724     { "SlateGray2", 185, 211, 238, 1, X11Compliance },
725     { "SlateGray3", 159, 182, 205, 1, X11Compliance },
726     { "SlateGray4", 108, 123, 139, 1, X11Compliance },
727     { "SlateGrey", 112, 128, 144, 1, SVGCompliance | X11Compliance },
728     { "snow", 255, 250, 250, 1, SVGCompliance | X11Compliance | XPMCompliance },
729     { "snow1", 255, 250, 250, 1, X11Compliance },
730     { "snow2", 238, 233, 233, 1, X11Compliance },
731     { "snow3", 205, 201, 201, 1, X11Compliance },
732     { "snow4", 139, 137, 137, 1, X11Compliance },
733     { "SpringGreen", 0, 255, 127, 1, SVGCompliance | X11Compliance | XPMCompliance },
734     { "SpringGreen1", 0, 255, 127, 1, X11Compliance },
735     { "SpringGreen2", 0, 238, 118, 1, X11Compliance },
736     { "SpringGreen3", 0, 205, 102, 1, X11Compliance },
737     { "SpringGreen4", 0, 139, 69, 1, X11Compliance },
738     { "SteelBlue", 70, 130, 180, 1, SVGCompliance | X11Compliance | XPMCompliance },
739     { "SteelBlue1", 99, 184, 255, 1, X11Compliance },
740     { "SteelBlue2", 92, 172, 238, 1, X11Compliance },
741     { "SteelBlue3", 79, 148, 205, 1, X11Compliance },
742     { "SteelBlue4", 54, 100, 139, 1, X11Compliance },
743     { "tan", 210, 180, 140, 1, SVGCompliance | X11Compliance | XPMCompliance },
744     { "tan1", 255, 165, 79, 1, X11Compliance },
745     { "tan2", 238, 154, 73, 1, X11Compliance },
746     { "tan3", 205, 133, 63, 1, X11Compliance },
747     { "tan4", 139, 90, 43, 1, X11Compliance },
748     { "teal", 0, 128, 128, 1, SVGCompliance },
749     { "thistle", 216, 191, 216, 1, SVGCompliance | X11Compliance | XPMCompliance },
750     { "thistle1", 255, 225, 255, 1, X11Compliance },
751     { "thistle2", 238, 210, 238, 1, X11Compliance },
752     { "thistle3", 205, 181, 205, 1, X11Compliance },
753     { "thistle4", 139, 123, 139, 1, X11Compliance },
754     { "tomato", 255, 99, 71, 1, SVGCompliance | X11Compliance | XPMCompliance },
755     { "tomato1", 255, 99, 71, 1, X11Compliance },
756     { "tomato2", 238, 92, 66, 1, X11Compliance },
757     { "tomato3", 205, 79, 57, 1, X11Compliance },
758     { "tomato4", 139, 54, 38, 1, X11Compliance },
759     { "transparent", 0, 0, 0, 0, SVGCompliance },
760     { "turquoise", 64, 224, 208, 1, SVGCompliance | X11Compliance | XPMCompliance },
761     { "turquoise1", 0, 245, 255, 1, X11Compliance },
762     { "turquoise2", 0, 229, 238, 1, X11Compliance },
763     { "turquoise3", 0, 197, 205, 1, X11Compliance },
764     { "turquoise4", 0, 134, 139, 1, X11Compliance },
765     { "violet", 238, 130, 238, 1, SVGCompliance | X11Compliance | XPMCompliance },
766     { "VioletRed", 208, 32, 144, 1, X11Compliance | XPMCompliance },
767     { "VioletRed1", 255, 62, 150, 1, X11Compliance },
768     { "VioletRed2", 238, 58, 140, 1, X11Compliance },
769     { "VioletRed3", 205, 50, 120, 1, X11Compliance },
770     { "VioletRed4", 139, 34, 82, 1, X11Compliance },
771     { "wheat", 245, 222, 179, 1, SVGCompliance | X11Compliance | XPMCompliance },
772     { "wheat1", 255, 231, 186, 1, X11Compliance },
773     { "wheat2", 238, 216, 174, 1, X11Compliance },
774     { "wheat3", 205, 186, 150, 1, X11Compliance },
775     { "wheat4", 139, 126, 102, 1, X11Compliance },
776     { "WhiteSmoke", 245, 245, 245, 1, SVGCompliance | X11Compliance | XPMCompliance },
777     { "yellow1", 255, 255, 0, 1, X11Compliance },
778     { "yellow2", 238, 238, 0, 1, X11Compliance },
779     { "yellow3", 205, 205, 0, 1, X11Compliance },
780     { "yellow4", 139, 139, 0, 1, X11Compliance },
781     { "YellowGreen", 154, 205, 50, 1, SVGCompliance | X11Compliance | XPMCompliance }
782   };
783 \f
784 /*
785   Static declarations.
786 */
787 static LinkedListInfo
788   *color_list = (LinkedListInfo *) NULL;
789
790 static SemaphoreInfo
791   *color_semaphore = (SemaphoreInfo *) NULL;
792
793 static volatile MagickBooleanType
794   instantiate_color = MagickFalse;
795 \f
796 /*
797   Forward declarations.
798 */
799 static MagickBooleanType
800   InitializeColorList(ExceptionInfo *),
801   LoadColorLists(const char *,ExceptionInfo *);
802 \f
803 /*
804 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
805 %                                                                             %
806 %                                                                             %
807 %                                                                             %
808 +   C o l o r C o m p o n e n t G e n e s i s                                 %
809 %                                                                             %
810 %                                                                             %
811 %                                                                             %
812 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
813 %
814 %  ColorComponentGenesis() instantiates the color component.
815 %
816 %  The format of the ColorComponentGenesis method is:
817 %
818 %      MagickBooleanType ColorComponentGenesis(void)
819 %
820 */
821 MagickPrivate MagickBooleanType ColorComponentGenesis(void)
822 {
823   color_semaphore=AcquireSemaphoreInfo();
824   return(MagickTrue);
825 }
826 \f
827 /*
828 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
829 %                                                                             %
830 %                                                                             %
831 %                                                                             %
832 +   C o l o r C o m p o n e n t T e r m i n u s                               %
833 %                                                                             %
834 %                                                                             %
835 %                                                                             %
836 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
837 %
838 %  ColorComponentTerminus() destroys the color component.
839 %
840 %  The format of the ColorComponentTerminus method is:
841 %
842 %      ColorComponentTerminus(void)
843 %
844 */
845
846 static void *DestroyColorElement(void *color_info)
847 {
848   register ColorInfo
849     *p;
850
851   p=(ColorInfo *) color_info;
852   if (IfMagickFalse(p->exempt))
853     {
854       if (p->path != (char *) NULL)
855         p->path=DestroyString(p->path);
856       if (p->name != (char *) NULL)
857         p->name=DestroyString(p->name);
858     }
859   p=(ColorInfo *) RelinquishMagickMemory(p);
860   return((void *) NULL);
861 }
862
863 MagickPrivate void ColorComponentTerminus(void)
864 {
865   if (color_semaphore == (SemaphoreInfo *) NULL)
866     ActivateSemaphoreInfo(&color_semaphore);
867   LockSemaphoreInfo(color_semaphore);
868   if (color_list != (LinkedListInfo *) NULL)
869     color_list=DestroyLinkedList(color_list,DestroyColorElement);
870   instantiate_color=MagickFalse;
871   UnlockSemaphoreInfo(color_semaphore);
872   RelinquishSemaphoreInfo(&color_semaphore);
873 }
874 \f
875 /*
876 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
877 %                                                                             %
878 %                                                                             %
879 %                                                                             %
880 +   G e t C o l o r C o m p l i a n c e                                       %
881 %                                                                             %
882 %                                                                             %
883 %                                                                             %
884 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
885 %
886 %  GetColorInfo() searches the color list for the specified name and standards
887 %  compliance and if found returns attributes for that color.
888 %
889 %  The format of the GetColorInfo method is:
890 %
891 %      const PixelInfo *GetColorInfo(const char *name,
892 %        const ComplianceType compliance,ExceptionInfo *exception)
893 %
894 %  A description of each parameter follows:
895 %
896 %    o name: the color name.
897 %
898 %    o compliance: Adhere to this color standard: SVG, X11, or XPM.
899 %
900 %    o exception: return any errors or warnings in this structure.
901 %
902 */
903 MagickExport const ColorInfo *GetColorCompliance(const char *name,
904   const ComplianceType compliance,ExceptionInfo *exception)
905 {
906   char
907     colorname[MaxTextExtent];
908
909   register const ColorInfo
910     *p;
911
912   register char
913     *q;
914
915   assert(exception != (ExceptionInfo *) NULL);
916   if ((color_list == (LinkedListInfo *) NULL) ||
917        IfMagickFalse(instantiate_color))
918     if (IfMagickFalse(InitializeColorList(exception)))
919       return((const ColorInfo *) NULL);
920   if ((color_list == (LinkedListInfo *) NULL) ||
921       IfMagickTrue(IsLinkedListEmpty(color_list)))
922     return((const ColorInfo *) NULL);
923   if ((name == (const char *) NULL) || (LocaleCompare(name,"*") == 0))
924     return((const ColorInfo *) GetValueFromLinkedList(color_list,0));
925   /*
926     Strip names of whitespace.
927   */
928   (void) CopyMagickString(colorname,name,MaxTextExtent);
929   for (q=colorname; *q != '\0'; q++)
930   {
931     if (isspace((int) ((unsigned char) *q)) == 0)
932       continue;
933     (void) CopyMagickString(q,q+1,MaxTextExtent);
934     q--;
935   }
936   /*
937     Search for color tag.
938   */
939   LockSemaphoreInfo(color_semaphore);
940   ResetLinkedListIterator(color_list);
941   p=(const ColorInfo *) GetNextValueInLinkedList(color_list);
942   while (p != (const ColorInfo *) NULL)
943   {
944     if (((p->compliance & compliance) != 0) &&
945         (LocaleCompare(colorname,p->name) == 0))
946       break;
947     p=(const ColorInfo *) GetNextValueInLinkedList(color_list);
948   }
949   if (p == (ColorInfo *) NULL)
950     (void) ThrowMagickException(exception,GetMagickModule(),OptionWarning,
951       "UnrecognizedColor","`%s'",name);
952   else
953     (void) InsertValueInLinkedList(color_list,0,
954       RemoveElementByValueFromLinkedList(color_list,p));
955   UnlockSemaphoreInfo(color_semaphore);
956   return(p);
957 }
958 \f
959 /*
960 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
961 %                                                                             %
962 %                                                                             %
963 %                                                                             %
964 +   G e t C o l o r I n f o                                                   %
965 %                                                                             %
966 %                                                                             %
967 %                                                                             %
968 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
969 %
970 %  GetColorInfo() searches the color list for the specified name and if found
971 %  returns attributes for that color.
972 %
973 %  The format of the GetColorInfo method is:
974 %
975 %      const PixelInfo *GetColorInfo(const char *name,
976 %        ExceptionInfo *exception)
977 %
978 %  A description of each parameter follows:
979 %
980 %    o color_info: search the color list for the specified name and if found
981 %      return attributes for that color.
982 %
983 %    o name: the color name.
984 %
985 %    o exception: return any errors or warnings in this structure.
986 %
987 */
988 MagickExport const ColorInfo *GetColorInfo(const char *name,
989   ExceptionInfo *exception)
990 {
991   return(GetColorCompliance(name,AllCompliance,exception));
992 }
993 \f
994 /*
995 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
996 %                                                                             %
997 %                                                                             %
998 %                                                                             %
999 +   C o n c a t e n a t e C o l o r C o m p o n e n t                         %
1000 %                                                                             %
1001 %                                                                             %
1002 %                                                                             %
1003 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1004 %
1005 %  ConcatenateColorComponent() returns the pixel as a canonical string.
1006 %
1007 %  The format of the ConcatenateColorComponent() method is:
1008 %
1009 %      void ConcatenateColorComponent(const PixelInfo *pixel,
1010 %        const PixelChannel component,const ComplianceType compliance,
1011 %        char *tuple)
1012 %
1013 %  A description of each parameter follows.
1014 %
1015 %    o pixel:  The pixel.
1016 %
1017 %    o component:  The pixel component.
1018 %
1019 %    o compliance: Adhere to this color standard: SVG, X11, or XPM.
1020 %
1021 %    o tuple:  The color tuple.
1022 %
1023 */
1024 MagickExport void ConcatenateColorComponent(const PixelInfo *pixel,
1025   const PixelChannel component,const ComplianceType compliance,char *tuple)
1026 {
1027   char
1028     text[MaxTextExtent];
1029
1030   double
1031     color;
1032
1033   color=0.0;
1034   switch (component)
1035   {
1036     case RedPixelChannel:
1037     {
1038       color=pixel->red;
1039       break;
1040     }
1041     case GreenPixelChannel:
1042     {
1043       color=pixel->green;
1044       break;
1045     }
1046     case BluePixelChannel:
1047     {
1048       color=pixel->blue;
1049       break;
1050     }
1051     case AlphaPixelChannel:
1052     {
1053       color=pixel->alpha;
1054       break;
1055     }
1056     case BlackPixelChannel:
1057     {
1058       color=pixel->black;
1059       break;
1060     }
1061     default:
1062       break;
1063   }
1064   if (compliance == NoCompliance)
1065     {
1066       (void) FormatLocaleString(text,MaxTextExtent,"%.*g",GetMagickPrecision(),color);
1067       (void) ConcatenateMagickString(tuple,text,MaxTextExtent);
1068       return;
1069     }
1070   if (compliance != SVGCompliance)
1071     {
1072       if (pixel->depth > 16)
1073         {
1074           (void) FormatLocaleString(text,MaxTextExtent,"%10lu",(unsigned long)
1075             ScaleQuantumToLong(ClampToQuantum(color)));
1076           (void) ConcatenateMagickString(tuple,text,MaxTextExtent);
1077           return;
1078         }
1079       if (pixel->depth > 8)
1080         {
1081           (void) FormatLocaleString(text,MaxTextExtent,"%5d",
1082             ScaleQuantumToShort(ClampToQuantum(color)));
1083           (void) ConcatenateMagickString(tuple,text,MaxTextExtent);
1084           return;
1085         }
1086       (void) FormatLocaleString(text,MaxTextExtent,"%3d",
1087         ScaleQuantumToChar(ClampToQuantum(color)));
1088       (void) ConcatenateMagickString(tuple,text,MaxTextExtent);
1089       return;
1090     }
1091   if (component == AlphaPixelChannel)
1092     {
1093       (void) FormatLocaleString(text,MaxTextExtent,"%.*g",GetMagickPrecision(),
1094         (QuantumScale*color));
1095       (void) ConcatenateMagickString(tuple,text,MaxTextExtent);
1096       return;
1097     }
1098   if ((pixel->colorspace == HCLColorspace) ||
1099       (pixel->colorspace == HCLpColorspace) ||
1100       (pixel->colorspace == HSBColorspace) ||
1101       (pixel->colorspace == HSIColorspace) ||
1102       (pixel->colorspace == HSLColorspace) ||
1103       (pixel->colorspace == HSVColorspace) ||
1104       (pixel->colorspace == HWBColorspace))
1105     {
1106       (void) FormatLocaleString(text,MaxTextExtent,"%.*g%%",GetMagickPrecision(),
1107         (100.0*QuantumScale*color));
1108       (void) ConcatenateMagickString(tuple,text,MaxTextExtent);
1109       return;
1110     }
1111   if (pixel->depth > 8)
1112     {
1113       (void) FormatLocaleString(text,MaxTextExtent,"%.*g%%",GetMagickPrecision(),
1114         (100.0*QuantumScale*color));
1115       (void) ConcatenateMagickString(tuple,text,MaxTextExtent);
1116       return;
1117     }
1118   (void) FormatLocaleString(text,MaxTextExtent,"%d",ScaleQuantumToChar(
1119     ClampToQuantum(color)));
1120   (void) ConcatenateMagickString(tuple,text,MaxTextExtent);
1121 }
1122 \f
1123 /*
1124 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1125 %                                                                             %
1126 %                                                                             %
1127 %                                                                             %
1128 %   G e t C o l o r I n f o L i s t                                           %
1129 %                                                                             %
1130 %                                                                             %
1131 %                                                                             %
1132 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1133 %
1134 %  GetColorInfoList() returns any colors that match the specified pattern.
1135 %
1136 %  The format of the GetColorInfoList function is:
1137 %
1138 %      const ColorInfo **GetColorInfoList(const char *pattern,
1139 %        size_t *number_colors,ExceptionInfo *exception)
1140 %
1141 %  A description of each parameter follows:
1142 %
1143 %    o pattern: Specifies a pointer to a text string containing a pattern.
1144 %
1145 %    o number_colors:  This integer returns the number of colors in the list.
1146 %
1147 %    o exception: return any errors or warnings in this structure.
1148 %
1149 */
1150
1151 #if defined(__cplusplus) || defined(c_plusplus)
1152 extern "C" {
1153 #endif
1154
1155 static int ColorInfoCompare(const void *x,const void *y)
1156 {
1157   const ColorInfo
1158     **p,
1159     **q;
1160
1161   p=(const ColorInfo **) x,
1162   q=(const ColorInfo **) y;
1163   if (LocaleCompare((*p)->path,(*q)->path) == 0)
1164     return(LocaleCompare((*p)->name,(*q)->name));
1165   return(LocaleCompare((*p)->path,(*q)->path));
1166 }
1167
1168 #if defined(__cplusplus) || defined(c_plusplus)
1169 }
1170 #endif
1171
1172 MagickExport const ColorInfo **GetColorInfoList(const char *pattern,
1173   size_t *number_colors,ExceptionInfo *exception)
1174 {
1175   const ColorInfo
1176     **colors;
1177
1178   register const ColorInfo
1179     *p;
1180
1181   register ssize_t
1182     i;
1183
1184   /*
1185     Allocate color list.
1186   */
1187   assert(pattern != (char *) NULL);
1188   (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",pattern);
1189   assert(number_colors != (size_t *) NULL);
1190   *number_colors=0;
1191   p=GetColorInfo("*",exception);
1192   if (p == (const ColorInfo *) NULL)
1193     return((const ColorInfo **) NULL);
1194   colors=(const ColorInfo **) AcquireQuantumMemory((size_t)
1195     GetNumberOfElementsInLinkedList(color_list)+1UL,sizeof(*colors));
1196   if (colors == (const ColorInfo **) NULL)
1197     return((const ColorInfo **) NULL);
1198   /*
1199     Generate color list.
1200   */
1201   LockSemaphoreInfo(color_semaphore);
1202   ResetLinkedListIterator(color_list);
1203   p=(const ColorInfo *) GetNextValueInLinkedList(color_list);
1204   for (i=0; p != (const ColorInfo *) NULL; )
1205   {
1206     if (IsMagickFalse(p->stealth) &&
1207         IsMagickTrue(GlobExpression(p->name,pattern,MagickFalse)))
1208       colors[i++]=p;
1209     p=(const ColorInfo *) GetNextValueInLinkedList(color_list);
1210   }
1211   UnlockSemaphoreInfo(color_semaphore);
1212   qsort((void *) colors,(size_t) i,sizeof(*colors),ColorInfoCompare);
1213   colors[i]=(ColorInfo *) NULL;
1214   *number_colors=(size_t) i;
1215   return(colors);
1216 }
1217 \f
1218 /*
1219 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1220 %                                                                             %
1221 %                                                                             %
1222 %                                                                             %
1223 %   G e t C o l o r L i s t                                                   %
1224 %                                                                             %
1225 %                                                                             %
1226 %                                                                             %
1227 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1228 %
1229 %  GetColorList() returns any colors that match the specified pattern.
1230 %
1231 %  The format of the GetColorList function is:
1232 %
1233 %      char **GetColorList(const char *pattern,size_t *number_colors,
1234 %        ExceptionInfo *exception)
1235 %
1236 %  A description of each parameter follows:
1237 %
1238 %    o pattern: Specifies a pointer to a text string containing a pattern.
1239 %
1240 %    o number_colors:  This integer returns the number of colors in the list.
1241 %
1242 %    o exception: return any errors or warnings in this structure.
1243 %
1244 */
1245
1246 #if defined(__cplusplus) || defined(c_plusplus)
1247 extern "C" {
1248 #endif
1249
1250 static int ColorCompare(const void *x,const void *y)
1251 {
1252   register const char
1253     **p,
1254     **q;
1255
1256   p=(const char **) x;
1257   q=(const char **) y;
1258   return(LocaleCompare(*p,*q));
1259 }
1260
1261 #if defined(__cplusplus) || defined(c_plusplus)
1262 }
1263 #endif
1264
1265 MagickExport char **GetColorList(const char *pattern,
1266   size_t *number_colors,ExceptionInfo *exception)
1267 {
1268   char
1269     **colors;
1270
1271   register const ColorInfo
1272     *p;
1273
1274   register ssize_t
1275     i;
1276
1277   /*
1278     Allocate color list.
1279   */
1280   assert(pattern != (char *) NULL);
1281   (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",pattern);
1282   assert(number_colors != (size_t *) NULL);
1283   *number_colors=0;
1284   p=GetColorInfo("*",exception);
1285   if (p == (const ColorInfo *) NULL)
1286     return((char **) NULL);
1287   colors=(char **) AcquireQuantumMemory((size_t)
1288     GetNumberOfElementsInLinkedList(color_list)+1UL,sizeof(*colors));
1289   if (colors == (char **) NULL)
1290     return((char **) NULL);
1291   /*
1292     Generate color list.
1293   */
1294   LockSemaphoreInfo(color_semaphore);
1295   ResetLinkedListIterator(color_list);
1296   p=(const ColorInfo *) GetNextValueInLinkedList(color_list);
1297   for (i=0; p != (const ColorInfo *) NULL; )
1298   {
1299     if (IsMagickFalse(p->stealth) &&
1300         IsMagickTrue(GlobExpression(p->name,pattern,MagickFalse)))
1301       colors[i++]=ConstantString(p->name);
1302     p=(const ColorInfo *) GetNextValueInLinkedList(color_list);
1303   }
1304   UnlockSemaphoreInfo(color_semaphore);
1305   qsort((void *) colors,(size_t) i,sizeof(*colors),ColorCompare);
1306   colors[i]=(char *) NULL;
1307   *number_colors=(size_t) i;
1308   return(colors);
1309 }
1310 \f
1311 /*
1312 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1313 %                                                                             %
1314 %                                                                             %
1315 %                                                                             %
1316 +   G e t C o l o r T u p l e                                                 %
1317 %                                                                             %
1318 %                                                                             %
1319 %                                                                             %
1320 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1321 %
1322 %  GetColorTuple() returns a color as a color tuple string (e.g. rgba(255,0,0))
1323 %  or hex string (e.g. #FF0000).
1324 %
1325 %  The format of the GetColorTuple method is:
1326 %
1327 %      GetColorTuple(const PixelInfo *pixel,const MagickBooleanType hex,
1328 %        char *tuple)
1329 %
1330 %  A description of each parameter follows.
1331 %
1332 %    o pixel: the pixel.
1333 %
1334 %    o hex: A value other than zero returns the tuple in a hexidecimal format.
1335 %
1336 %    o tuple: Return the color tuple as this string.
1337 %
1338 */
1339
1340 static void ConcatentateHexColorComponent(const PixelInfo *pixel,
1341   const PixelChannel channel,char *tuple)
1342 {
1343   char
1344     component[MaxTextExtent];
1345
1346   double
1347     color;
1348
1349   color=0.0;
1350   switch (channel)
1351   {
1352     case RedPixelChannel:
1353     {
1354       color=pixel->red;
1355       break;
1356     }
1357     case GreenPixelChannel:
1358     {
1359       color=pixel->green;
1360       break;
1361     }
1362     case BluePixelChannel:
1363     {
1364       color=pixel->blue;
1365       break;
1366     }
1367     case AlphaPixelChannel:
1368     {
1369       color=pixel->alpha;
1370       break;
1371     }
1372     case BlackPixelChannel:
1373     {
1374       color=pixel->black;
1375       break;
1376     }
1377     default:
1378       break;
1379   }
1380   if (pixel->depth > 32)
1381     {
1382       (void) FormatLocaleString(component,MaxTextExtent,"%08lX%08lX",
1383         (unsigned long) ScaleQuantumToLong(ClampToQuantum(color)),
1384         (unsigned long) ScaleQuantumToLong(ClampToQuantum(color)));
1385       (void) ConcatenateMagickString(tuple,component,MaxTextExtent);
1386       return;
1387     }
1388   if (pixel->depth > 16)
1389     {
1390       (void) FormatLocaleString(component,MaxTextExtent,"%08X",
1391         (unsigned int) ScaleQuantumToLong(ClampToQuantum(color)));
1392       (void) ConcatenateMagickString(tuple,component,MaxTextExtent);
1393       return;
1394     }
1395   if (pixel->depth > 8)
1396     {
1397       (void) FormatLocaleString(component,MaxTextExtent,"%04X",
1398         ScaleQuantumToShort(ClampToQuantum(color)));
1399       (void) ConcatenateMagickString(tuple,component,MaxTextExtent);
1400       return;
1401     }
1402   (void) FormatLocaleString(component,MaxTextExtent,"%02X",
1403     ScaleQuantumToChar(ClampToQuantum(color)));
1404   (void) ConcatenateMagickString(tuple,component,MaxTextExtent);
1405   return;
1406 }
1407
1408 MagickExport void GetColorTuple(const PixelInfo *pixel,
1409   const MagickBooleanType hex,char *tuple)
1410 {
1411   PixelInfo
1412     color;
1413
1414   assert(pixel != (const PixelInfo *) NULL);
1415   assert(tuple != (char *) NULL);
1416   (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",tuple);
1417   *tuple='\0';
1418   if (IfMagickTrue(hex))
1419     {
1420       /*
1421         Convert pixel to hex color.
1422       */
1423       (void) ConcatenateMagickString(tuple,"#",MaxTextExtent);
1424       ConcatentateHexColorComponent(pixel,RedPixelChannel,tuple);
1425       ConcatentateHexColorComponent(pixel,GreenPixelChannel,tuple);
1426       ConcatentateHexColorComponent(pixel,BluePixelChannel,tuple);
1427       if (pixel->colorspace == CMYKColorspace)
1428         ConcatentateHexColorComponent(pixel,BlackPixelChannel,tuple);
1429       if ((pixel->alpha_trait == BlendPixelTrait) &&
1430           (pixel->alpha != OpaqueAlpha))
1431         ConcatentateHexColorComponent(pixel,AlphaPixelChannel,tuple);
1432       return;
1433     }
1434   /*
1435     Convert pixel to rgb() or cmyk() color.
1436   */
1437   color=(*pixel);
1438   if (color.depth > 8)
1439     {
1440 #define SVGCompliant(component) ((double) \
1441    ScaleCharToQuantum(ScaleQuantumToChar(ClampToQuantum(component))))
1442
1443       MagickStatusType
1444         status;
1445
1446       /*
1447         SVG requires color depths > 8 expressed as percentages.
1448       */
1449       status=IsMagickTrue(fabs(color.red-SVGCompliant(color.red))
1450            < MagickEpsilon);
1451       status&=IsMagickTrue(fabs(color.green-SVGCompliant(color.green))
1452            < MagickEpsilon);
1453       status&=IsMagickTrue(fabs(color.blue-SVGCompliant(color.blue))
1454            < MagickEpsilon);
1455       if (color.colorspace-CMYKColorspace)
1456         status&=IsMagickTrue(fabs(color.black-SVGCompliant(color.black))
1457              < MagickEpsilon);
1458       if (color.alpha_trait == BlendPixelTrait)
1459         status&=IsMagickTrue(fabs(color.alpha-SVGCompliant(color.alpha))
1460              < MagickEpsilon);
1461       if (IfMagickTrue(status))
1462         color.depth=8;
1463     }
1464   (void) ConcatenateMagickString(tuple,CommandOptionToMnemonic(
1465     MagickColorspaceOptions,(ssize_t) color.colorspace),MaxTextExtent);
1466   if (color.alpha_trait == BlendPixelTrait)
1467     (void) ConcatenateMagickString(tuple,"a",MaxTextExtent);
1468   (void) ConcatenateMagickString(tuple,"(",MaxTextExtent);
1469   if (color.colorspace == GRAYColorspace)
1470     ConcatenateColorComponent(&color,GrayPixelChannel,SVGCompliance,tuple);
1471   else
1472     {
1473       ConcatenateColorComponent(&color,RedPixelChannel,SVGCompliance,tuple);
1474       (void) ConcatenateMagickString(tuple,",",MaxTextExtent);
1475       ConcatenateColorComponent(&color,GreenPixelChannel,SVGCompliance,tuple);
1476       (void) ConcatenateMagickString(tuple,",",MaxTextExtent);
1477       ConcatenateColorComponent(&color,BluePixelChannel,SVGCompliance,tuple);
1478     }
1479   if (color.colorspace == CMYKColorspace)
1480     {
1481       (void) ConcatenateMagickString(tuple,",",MaxTextExtent);
1482       ConcatenateColorComponent(&color,BlackPixelChannel,SVGCompliance,tuple);
1483     }
1484   if (color.alpha_trait == BlendPixelTrait)
1485     {
1486       (void) ConcatenateMagickString(tuple,",",MaxTextExtent);
1487       ConcatenateColorComponent(&color,AlphaPixelChannel,SVGCompliance,tuple);
1488     }
1489   (void) ConcatenateMagickString(tuple,")",MaxTextExtent);
1490   LocaleLower(tuple);
1491   return;
1492 }
1493 \f
1494 /*
1495 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1496 %                                                                             %
1497 %                                                                             %
1498 %                                                                             %
1499 +   I n i t i a l i z e C o l o r L i s t                                     %
1500 %                                                                             %
1501 %                                                                             %
1502 %                                                                             %
1503 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1504 %
1505 %  InitializeColorList() initializes the color list.
1506 %
1507 %  The format of the InitializeColorList method is:
1508 %
1509 %      MagickBooleanType InitializeColorList(ExceptionInfo *exception)
1510 %
1511 %  A description of each parameter follows.
1512 %
1513 %    o exception: return any errors or warnings in this structure.
1514 %
1515 */
1516 static MagickBooleanType InitializeColorList(ExceptionInfo *exception)
1517 {
1518   if ((color_list == (LinkedListInfo *) NULL) ||
1519       IfMagickFalse(instantiate_color))
1520     {
1521       if (color_semaphore == (SemaphoreInfo *) NULL)
1522         ActivateSemaphoreInfo(&color_semaphore);
1523       LockSemaphoreInfo(color_semaphore);
1524       if ((color_list == (LinkedListInfo *) NULL) ||
1525           IfMagickFalse(instantiate_color))
1526         {
1527           (void) LoadColorLists(ColorFilename,exception);
1528           instantiate_color=MagickTrue;
1529         }
1530       UnlockSemaphoreInfo(color_semaphore);
1531     }
1532   return(color_list != (LinkedListInfo *) NULL ? MagickTrue : MagickFalse);
1533 }
1534 \f
1535 /*
1536 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1537 %                                                                             %
1538 %                                                                             %
1539 %                                                                             %
1540 +   I s A l p h a S i m i l a r                                               %
1541 %                                                                             %
1542 %                                                                             %
1543 %                                                                             %
1544 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1545 %
1546 %  IsEquivalentAlpha() returns true if the distance between two alpha values is
1547 %  less than the specified distance in a linear color space.  This method is
1548 %  used by MatteFloodFill() and other algorithms which compare two alpha values.
1549 %
1550 %  The format of the IsEquivalentAlpha method is:
1551 %
1552 %      void IsEquivalentAlpha(const Image *image,const PixelInfo *p,
1553 %        const PixelInfo *q)
1554 %
1555 %  A description of each parameter follows:
1556 %
1557 %    o image: the image.
1558 %
1559 %    o p: Pixel p.
1560 %
1561 %    o q: Pixel q.
1562 %
1563 */
1564
1565 static inline double MagickMax(const double x,const double y)
1566 {
1567   if (x > y)
1568     return(x);
1569   return(y);
1570 }
1571
1572 MagickExport MagickBooleanType IsEquivalentAlpha(const Image *image,
1573   const PixelInfo *p,const PixelInfo *q)
1574 {
1575   double
1576     fuzz,
1577     pixel;
1578
1579   register double
1580     distance;
1581
1582   if (image->alpha_trait != BlendPixelTrait)
1583     return(MagickTrue);
1584   if (p->alpha == q->alpha)
1585     return(MagickTrue);
1586   fuzz=MagickMax(image->fuzz,MagickSQ1_2)*MagickMax(image->fuzz,MagickSQ1_2);
1587   pixel=(double) p->alpha-(double) q->alpha;
1588   distance=pixel*pixel;
1589   if (distance > fuzz)
1590     return(MagickFalse);
1591   return(MagickTrue);
1592 }
1593 \f
1594 /*
1595 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1596 %                                                                             %
1597 %                                                                             %
1598 %                                                                             %
1599 +   I s I m a g e S i m i l a r                                               %
1600 %                                                                             %
1601 %                                                                             %
1602 %                                                                             %
1603 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1604 %
1605 %  IsEquivalentImage() returns true if the target is similar to a region of the
1606 %  image.
1607 %
1608 %  The format of the IsEquivalentImage method is:
1609 %
1610 %      MagickBooleanType IsEquivalentImage(const Image *image,
1611 %        const Image *target_image,ssize_t *x_offset,ssize_t *y_offset,
1612 %        ExceptionInfo *exception)
1613 %
1614 %  A description of each parameter follows:
1615 %
1616 %    o image: the image.
1617 %
1618 %    o target_image: the target image.
1619 %
1620 %    o x_offset: On input the starting x position to search for a match;
1621 %      on output the x position of the first match found.
1622 %
1623 %    o y_offset: On input the starting y position to search for a match;
1624 %      on output the y position of the first match found.
1625 %
1626 %    o exception: return any errors or warnings in this structure.
1627 %
1628 */
1629 MagickExport MagickBooleanType IsEquivalentImage(const Image *image,
1630   const Image *target_image,ssize_t *x_offset,ssize_t *y_offset,
1631   ExceptionInfo *exception)
1632 {
1633 #define SearchImageText  "  Searching image...  "
1634
1635   CacheView
1636     *image_view,
1637     *target_view;
1638
1639   MagickBooleanType
1640     status;
1641
1642   PixelInfo
1643     target,
1644     pixel;
1645
1646   register const Quantum
1647     *p,
1648     *q;
1649
1650   register ssize_t
1651     i,
1652     x;
1653
1654   ssize_t
1655     j,
1656     y;
1657
1658   assert(image != (Image *) NULL);
1659   assert(image->signature == MagickSignature);
1660   if (IfMagickTrue(image->debug))
1661     (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
1662   assert(target_image != (Image *) NULL);
1663   assert(target_image->signature == MagickSignature);
1664   assert(x_offset != (ssize_t *) NULL);
1665   assert(y_offset != (ssize_t *) NULL);
1666   assert(exception != (ExceptionInfo *) NULL);
1667   x=0;
1668   status=MagickTrue;
1669   GetPixelInfo(image,&pixel);
1670   GetPixelInfo(image,&target);
1671   image_view=AcquireVirtualCacheView(image,exception);
1672   target_view=AcquireVirtualCacheView(target_image,exception);
1673   for (y=(*y_offset); y < (ssize_t) image->rows; y++)
1674   {
1675     for (x=y == 0 ? *x_offset : 0; x < (ssize_t) image->columns; x++)
1676     {
1677       for (j=0; j < (ssize_t) target_image->rows; j++)
1678       {
1679         for (i=0; i < (ssize_t) target_image->columns; i++)
1680         {
1681           p=GetCacheViewVirtualPixels(image_view,x+i,y+j,1,1,exception);
1682           GetPixelInfoPixel(image,p,&pixel);
1683           q=GetCacheViewVirtualPixels(target_view,i,j,1,1,exception);
1684           GetPixelInfoPixel(image,q,&target);
1685           if (IfMagickFalse(IsFuzzyEquivalencePixelInfo(&pixel,&target)))
1686             break;
1687         }
1688         if (i < (ssize_t) target_image->columns)
1689           break;
1690       }
1691       if (j == (ssize_t) target_image->rows)
1692         break;
1693     }
1694     if (x < (ssize_t) image->columns)
1695       break;
1696     if (image->progress_monitor != (MagickProgressMonitor) NULL)
1697       {
1698         MagickBooleanType
1699           proceed;
1700
1701         proceed=SetImageProgress(image,SearchImageText,(MagickOffsetType) y,
1702           image->rows);
1703         if( IfMagickFalse(proceed) )
1704           status=MagickFalse;
1705       }
1706   }
1707   target_view=DestroyCacheView(target_view);
1708   image_view=DestroyCacheView(image_view);
1709   *x_offset=x;
1710   *y_offset=y;
1711   if (IfMagickFalse(status))
1712     return(status);
1713   return(IsMagickTrue(y < (ssize_t) image->rows));
1714 }
1715 \f
1716 /*
1717 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1718 %                                                                             %
1719 %                                                                             %
1720 %                                                                             %
1721 %  L i s t C o l o r I n f o                                                  %
1722 %                                                                             %
1723 %                                                                             %
1724 %                                                                             %
1725 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1726 %
1727 %  ListColorInfo() lists color names to the specified file.  Color names
1728 %  are a convenience.  Rather than defining a color by its red, green, and
1729 %  blue intensities just use a color name such as white, blue, or yellow.
1730 %
1731 %  The format of the ListColorInfo method is:
1732 %
1733 %      MagickBooleanType ListColorInfo(FILE *file,ExceptionInfo *exception)
1734 %
1735 %  A description of each parameter follows.
1736 %
1737 %    o file:  List color names to this file handle.
1738 %
1739 %    o exception: return any errors or warnings in this structure.
1740 %
1741 */
1742 MagickExport MagickBooleanType ListColorInfo(FILE *file,
1743   ExceptionInfo *exception)
1744 {
1745   char
1746     tuple[MaxTextExtent];
1747
1748   const char
1749     *path;
1750
1751   const ColorInfo
1752     **color_info;
1753
1754   register ssize_t
1755     i;
1756
1757   size_t
1758     number_colors;
1759
1760   /*
1761     List name and attributes of each color in the list.
1762   */
1763   if (file == (const FILE *) NULL)
1764     file=stdout;
1765   color_info=GetColorInfoList("*",&number_colors,exception);
1766   if (color_info == (const ColorInfo **) NULL)
1767     return(MagickFalse);
1768   path=(const char *) NULL;
1769   for (i=0; i < (ssize_t) number_colors; i++)
1770   {
1771     if (IfMagickTrue(color_info[i]->stealth))
1772       continue;
1773     if ((path == (const char *) NULL) ||
1774         (LocaleCompare(path,color_info[i]->path) != 0))
1775       {
1776         if (color_info[i]->path != (char *) NULL)
1777           (void) FormatLocaleFile(file,"\nPath: %s\n\n",color_info[i]->path);
1778         (void) FormatLocaleFile(file,
1779           "Name                  Color                  "
1780           "                       Compliance\n");
1781         (void) FormatLocaleFile(file,
1782           "-------------------------------------------------"
1783           "------------------------------\n");
1784       }
1785     path=color_info[i]->path;
1786     (void) FormatLocaleFile(file,"%-21.21s ",color_info[i]->name);
1787     GetColorTuple(&color_info[i]->color,MagickFalse,tuple);
1788     (void) FormatLocaleFile(file,"%-45.45s ",tuple);
1789     if ((color_info[i]->compliance & SVGCompliance) != 0)
1790       (void) FormatLocaleFile(file,"SVG ");
1791     if ((color_info[i]->compliance & X11Compliance) != 0)
1792       (void) FormatLocaleFile(file,"X11 ");
1793     if ((color_info[i]->compliance & XPMCompliance) != 0)
1794       (void) FormatLocaleFile(file,"XPM ");
1795     (void) FormatLocaleFile(file,"\n");
1796   }
1797   color_info=(const ColorInfo **) RelinquishMagickMemory((void *) color_info);
1798   (void) fflush(file);
1799   return(MagickTrue);
1800 }
1801 \f
1802 /*
1803 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1804 %                                                                             %
1805 %                                                                             %
1806 %                                                                             %
1807 +   L o a d C o l o r L i s t                                                 %
1808 %                                                                             %
1809 %                                                                             %
1810 %                                                                             %
1811 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1812 %
1813 %  LoadColorList() loads the color configuration file which provides a mapping
1814 %  between color attributes and a color name.
1815 %
1816 %  The format of the LoadColorList method is:
1817 %
1818 %      MagickBooleanType LoadColorList(const char *xml,const char *filename,
1819 %        const size_t depth,ExceptionInfo *exception)
1820 %
1821 %  A description of each parameter follows:
1822 %
1823 %    o xml:  The color list in XML format.
1824 %
1825 %    o filename:  The color list filename.
1826 %
1827 %    o depth: depth of <include /> statements.
1828 %
1829 %    o exception: return any errors or warnings in this structure.
1830 %
1831 */
1832 static MagickBooleanType LoadColorList(const char *xml,const char *filename,
1833   const size_t depth,ExceptionInfo *exception)
1834 {
1835   char
1836     keyword[MaxTextExtent],
1837     *token;
1838
1839   ColorInfo
1840     *color_info;
1841
1842   const char
1843     *q;
1844
1845   MagickBooleanType
1846     status;
1847
1848   /*
1849     Load the color map file.
1850   */
1851   (void) LogMagickEvent(ConfigureEvent,GetMagickModule(),
1852     "Loading color file \"%s\" ...",filename);
1853   if (xml == (char *) NULL)
1854     return(MagickFalse);
1855   if (color_list == (LinkedListInfo *) NULL)
1856     {
1857       color_list=NewLinkedList(0);
1858       if (color_list == (LinkedListInfo *) NULL)
1859         {
1860           ThrowFileException(exception,ResourceLimitError,
1861             "MemoryAllocationFailed",filename);
1862           return(MagickFalse);
1863         }
1864     }
1865   status=MagickTrue;
1866   color_info=(ColorInfo *) NULL;
1867   token=AcquireString(xml);
1868   for (q=(char *) xml; *q != '\0'; )
1869   {
1870     /*
1871       Interpret XML.
1872     */
1873     GetMagickToken(q,&q,token);
1874     if (*token == '\0')
1875       break;
1876     (void) CopyMagickString(keyword,token,MaxTextExtent);
1877     if (LocaleNCompare(keyword,"<!DOCTYPE",9) == 0)
1878       {
1879         /*
1880           Doctype element.
1881         */
1882         while ((LocaleNCompare(q,"]>",2) != 0) && (*q != '\0'))
1883           GetMagickToken(q,&q,token);
1884         continue;
1885       }
1886     if (LocaleNCompare(keyword,"<!--",4) == 0)
1887       {
1888         /*
1889           Comment element.
1890         */
1891         while ((LocaleNCompare(q,"->",2) != 0) && (*q != '\0'))
1892           GetMagickToken(q,&q,token);
1893         continue;
1894       }
1895     if (LocaleCompare(keyword,"<include") == 0)
1896       {
1897         /*
1898           Include element.
1899         */
1900         while (((*token != '/') && (*(token+1) != '>')) && (*q != '\0'))
1901         {
1902           (void) CopyMagickString(keyword,token,MaxTextExtent);
1903           GetMagickToken(q,&q,token);
1904           if (*token != '=')
1905             continue;
1906           GetMagickToken(q,&q,token);
1907           if (LocaleCompare(keyword,"file") == 0)
1908             {
1909               if (depth > 200)
1910                 (void) ThrowMagickException(exception,GetMagickModule(),
1911                   ConfigureError,"IncludeElementNestedTooDeeply","`%s'",token);
1912               else
1913                 {
1914                   char
1915                     path[MaxTextExtent],
1916                     *xml;
1917
1918                   GetPathComponent(filename,HeadPath,path);
1919                   if (*path != '\0')
1920                     (void) ConcatenateMagickString(path,DirectorySeparator,
1921                       MaxTextExtent);
1922                   if (*token == *DirectorySeparator)
1923                     (void) CopyMagickString(path,token,MaxTextExtent);
1924                   else
1925                     (void) ConcatenateMagickString(path,token,MaxTextExtent);
1926                   xml=FileToString(path,~0UL,exception);
1927                   if (xml != (char *) NULL)
1928                     {
1929                       status=LoadColorList(xml,path,depth+1,exception);
1930                       xml=(char *) RelinquishMagickMemory(xml);
1931                     }
1932                 }
1933             }
1934         }
1935         continue;
1936       }
1937     if (LocaleCompare(keyword,"<color") == 0)
1938       {
1939         /*
1940           Color element.
1941         */
1942         color_info=(ColorInfo *) AcquireMagickMemory(sizeof(*color_info));
1943         if (color_info == (ColorInfo *) NULL)
1944           ThrowFatalException(ResourceLimitFatalError,"MemoryAllocationFailed");
1945         (void) ResetMagickMemory(color_info,0,sizeof(*color_info));
1946         color_info->path=ConstantString(filename);
1947         color_info->exempt=MagickFalse;
1948         color_info->signature=MagickSignature;
1949         continue;
1950       }
1951     if (color_info == (ColorInfo *) NULL)
1952       continue;
1953     if (LocaleCompare(keyword,"/>") == 0)
1954       {
1955         status=AppendValueToLinkedList(color_list,color_info);
1956         if (IfMagickFalse(status))
1957           (void) ThrowMagickException(exception,GetMagickModule(),
1958             ResourceLimitError,"MemoryAllocationFailed","`%s'",
1959             color_info->name);
1960         color_info=(ColorInfo *) NULL;
1961         continue;
1962       }
1963     GetMagickToken(q,(const char **) NULL,token);
1964     if (*token != '=')
1965       continue;
1966     GetMagickToken(q,&q,token);
1967     GetMagickToken(q,&q,token);
1968     switch (*keyword)
1969     {
1970       case 'C':
1971       case 'c':
1972       {
1973         if (LocaleCompare((char *) keyword,"color") == 0)
1974           {
1975             (void) QueryColorCompliance(token,AllCompliance,&color_info->color,
1976               exception);
1977             break;
1978           }
1979         if (LocaleCompare((char *) keyword,"compliance") == 0)
1980           {
1981             ssize_t
1982               compliance;
1983
1984             compliance=color_info->compliance;
1985             if (IfMagickTrue(GlobExpression(token,"*SVG*",MagickTrue)))
1986               compliance|=SVGCompliance;
1987             if (IfMagickTrue(GlobExpression(token,"*X11*",MagickTrue)))
1988               compliance|=X11Compliance;
1989             if (IfMagickTrue(GlobExpression(token,"*XPM*",MagickTrue)))
1990               compliance|=XPMCompliance;
1991             color_info->compliance=(ComplianceType) compliance;
1992             break;
1993           }
1994         break;
1995       }
1996       case 'N':
1997       case 'n':
1998       {
1999         if (LocaleCompare((char *) keyword,"name") == 0)
2000           {
2001             color_info->name=ConstantString(token);
2002             break;
2003           }
2004         break;
2005       }
2006       case 'S':
2007       case 's':
2008       {
2009         if (LocaleCompare((char *) keyword,"stealth") == 0)
2010           {
2011             color_info->stealth=IsStringTrue(token);
2012             break;
2013           }
2014         break;
2015       }
2016       default:
2017         break;
2018     }
2019   }
2020   token=(char *) RelinquishMagickMemory(token);
2021   return(status);
2022 }
2023 \f
2024 /*
2025 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2026 %                                                                             %
2027 %                                                                             %
2028 %                                                                             %
2029 %  L o a d C o l o r L i s t s                                                %
2030 %                                                                             %
2031 %                                                                             %
2032 %                                                                             %
2033 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2034 %
2035 %  LoadColorList() loads one or more color configuration file which provides a
2036 %  mapping between color attributes and a color name.
2037 %
2038 %  The format of the LoadColorLists method is:
2039 %
2040 %      MagickBooleanType LoadColorLists(const char *filename,
2041 %        ExceptionInfo *exception)
2042 %
2043 %  A description of each parameter follows:
2044 %
2045 %    o filename: the font file name.
2046 %
2047 %    o exception: return any errors or warnings in this structure.
2048 %
2049 */
2050 static MagickBooleanType LoadColorLists(const char *filename,
2051   ExceptionInfo *exception)
2052 {
2053   const StringInfo
2054     *option;
2055
2056   LinkedListInfo
2057     *options;
2058
2059   MagickStatusType
2060     status;
2061
2062   register ssize_t
2063     i;
2064
2065   /*
2066     Load external color map.
2067   */
2068   if (color_list == (LinkedListInfo *) NULL)
2069     {
2070       color_list=NewLinkedList(0);
2071       if (color_list == (LinkedListInfo *) NULL)
2072         {
2073           ThrowFileException(exception,ResourceLimitError,
2074             "MemoryAllocationFailed",filename);
2075           return(MagickFalse);
2076         }
2077     }
2078   status=MagickTrue;
2079   options=GetConfigureOptions(filename,exception);
2080   option=(const StringInfo *) GetNextValueInLinkedList(options);
2081   while (option != (const StringInfo *) NULL)
2082   {
2083     status&=LoadColorList((const char *) GetStringInfoDatum(option),
2084       GetStringInfoPath(option),0,exception);
2085     option=(const StringInfo *) GetNextValueInLinkedList(options);
2086   }
2087   options=DestroyConfigureOptions(options);
2088   /*
2089     Load built-in color map.
2090   */
2091   for (i=0; i < (ssize_t) (sizeof(ColorMap)/sizeof(*ColorMap)); i++)
2092   {
2093     ColorInfo
2094       *color_info;
2095
2096     register const ColorMapInfo
2097       *p;
2098
2099     p=ColorMap+i;
2100     color_info=(ColorInfo *) AcquireMagickMemory(sizeof(*color_info));
2101     if (color_info == (ColorInfo *) NULL)
2102       {
2103         (void) ThrowMagickException(exception,GetMagickModule(),
2104           ResourceLimitError,"MemoryAllocationFailed","`%s'",p->name);
2105         continue;
2106       }
2107     (void) ResetMagickMemory(color_info,0,sizeof(*color_info));
2108     color_info->path=(char *) "[built-in]";
2109     color_info->name=(char *) p->name;
2110     GetPixelInfo((Image *) NULL,&color_info->color);
2111     color_info->color.red=(double) ScaleCharToQuantum(p->red);
2112     color_info->color.green=(double) ScaleCharToQuantum(p->green);
2113     color_info->color.blue=(double) ScaleCharToQuantum(p->blue);
2114     color_info->color.alpha=(double) (QuantumRange*p->alpha);
2115     color_info->compliance=(ComplianceType) p->compliance;
2116     color_info->exempt=MagickTrue;
2117     color_info->signature=MagickSignature;
2118     status&=AppendValueToLinkedList(color_list,color_info);
2119     if (IfMagickFalse(status))
2120       (void) ThrowMagickException(exception,GetMagickModule(),
2121         ResourceLimitError,"MemoryAllocationFailed","`%s'",color_info->name);
2122   }
2123   return(status != 0 ? MagickTrue : MagickFalse);
2124 }
2125 \f
2126 /*
2127 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2128 %                                                                             %
2129 %                                                                             %
2130 %                                                                             %
2131 +   Q u e r y C o l o r C o m p l i a n c e                                   %
2132 %                                                                             %
2133 %                                                                             %
2134 %                                                                             %
2135 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2136 %
2137 %  QueryColorCompliance() returns the red, green, blue, and alpha intensities
2138 %  for a given color name and standards compliance.
2139 %
2140 %  The format of the QueryColorCompliance method is:
2141 %
2142 %      MagickBooleanType QueryColorCompliance(const char *name,
2143 %        const ComplianceType compliance,PixelInfo *color,
2144 %        ExceptionInfo *exception)
2145 %
2146 %  A description of each parameter follows:
2147 %
2148 %    o name: the color name (e.g. white, blue, yellow).
2149 %
2150 %    o compliance: Adhere to this color standard: SVG, X11, or XPM.
2151 %
2152 %    o color: the red, green, blue, and opacity intensities values of the
2153 %      named color in this structure.
2154 %
2155 %    o exception: return any errors or warnings in this structure.
2156 %
2157 */
2158 MagickExport MagickBooleanType QueryColorCompliance(const char *name,
2159   const ComplianceType compliance,PixelInfo *color,ExceptionInfo *exception)
2160 {
2161   extern const char
2162     BackgroundColor[];
2163
2164   GeometryInfo
2165     geometry_info;
2166
2167   double
2168     scale;
2169
2170   MagickStatusType
2171     flags;
2172
2173   register const ColorInfo
2174     *p;
2175
2176   register ssize_t
2177     i;
2178
2179   ssize_t
2180     type;
2181
2182   /*
2183     Initialize color return value.
2184   */
2185   assert(name != (const char *) NULL);
2186   (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",name);
2187   assert(color != (PixelInfo *) NULL);
2188   GetPixelInfo((Image *) NULL,color);
2189   if ((name == (char *) NULL) || (*name == '\0'))
2190     name=BackgroundColor;
2191   while (isspace((int) ((unsigned char) *name)) != 0)
2192     name++;
2193   if (*name == '#')
2194     {
2195       char
2196         c;
2197
2198       PixelPacket
2199         pixel;
2200
2201       QuantumAny
2202         range;
2203
2204       size_t
2205         depth,
2206         n;
2207
2208       /*
2209         Parse hex color.
2210       */
2211       (void) ResetMagickMemory(&pixel,0,sizeof(pixel));
2212       name++;
2213       for (n=0; isxdigit((int) ((unsigned char) name[n])) != 0; n++) ;
2214       if ((n % 3) == 0)
2215         {
2216           do
2217           {
2218             pixel.red=pixel.green;
2219             pixel.green=pixel.blue;
2220             pixel.blue=0;
2221             for (i=(ssize_t) (n/3-1); i >= 0; i--)
2222             {
2223               c=(*name++);
2224               pixel.blue<<=4;
2225               if ((c >= '0') && (c <= '9'))
2226                 pixel.blue|=(int) (c-'0');
2227               else
2228                 if ((c >= 'A') && (c <= 'F'))
2229                   pixel.blue|=(int) c-((int) 'A'-10);
2230                 else
2231                   if ((c >= 'a') && (c <= 'f'))
2232                     pixel.blue|=(int) c-((int) 'a'-10);
2233                   else
2234                     return(MagickFalse);
2235             }
2236           } while (isxdigit((int) ((unsigned char) *name)) != 0);
2237           depth=4*(n/3);
2238         }
2239       else
2240         {
2241           if ((n % 4) != 0)
2242             {
2243               (void) ThrowMagickException(exception,GetMagickModule(),
2244                 OptionWarning,"UnrecognizedColor","`%s'",name);
2245               return(MagickFalse);
2246             }
2247           do
2248           {
2249             pixel.red=pixel.green;
2250             pixel.green=pixel.blue;
2251             pixel.blue=pixel.alpha;
2252             pixel.alpha=0;
2253             for (i=(ssize_t) (n/4-1); i >= 0; i--)
2254             {
2255               c=(*name++);
2256               pixel.alpha<<=4;
2257               if ((c >= '0') && (c <= '9'))
2258                 pixel.alpha|=(int) (c-'0');
2259               else
2260                 if ((c >= 'A') && (c <= 'F'))
2261                   pixel.alpha|=(int) c-((int) 'A'-10);
2262                 else
2263                   if ((c >= 'a') && (c <= 'f'))
2264                     pixel.alpha|=(int) c-((int) 'a'-10);
2265                   else
2266                     return(MagickFalse);
2267             }
2268           } while (isxdigit((int) ((unsigned char) *name)) != 0);
2269           depth=4*(n/4);
2270         }
2271       color->colorspace=sRGBColorspace;
2272       color->depth=depth;
2273       color->alpha_trait=UndefinedPixelTrait;
2274       range=GetQuantumRange(depth);
2275       color->red=(double) ScaleAnyToQuantum(pixel.red,range);
2276       color->green=(double) ScaleAnyToQuantum(pixel.green,range);
2277       color->blue=(double) ScaleAnyToQuantum(pixel.blue,range);
2278       color->alpha=(double) OpaqueAlpha;
2279       if ((n % 3) != 0)
2280         {
2281           color->alpha_trait=BlendPixelTrait;
2282           color->alpha=(double) ScaleAnyToQuantum(pixel.alpha,range);
2283         }
2284       color->black=0.0;
2285       return(MagickTrue);
2286     }
2287   if (strchr(name,'(') != (char *) NULL)
2288     {
2289       char
2290         colorspace[MaxTextExtent];
2291
2292       MagickBooleanType
2293         icc_color;
2294
2295       /*
2296         Parse color of the form rgb(100,255,0).
2297       */
2298       (void) CopyMagickString(colorspace,name,MaxTextExtent);
2299       for (i=0; colorspace[i] != '\0'; i++)
2300         if (colorspace[i] == '(')
2301           break;
2302       colorspace[i--]='\0';
2303       scale=(double) ScaleCharToQuantum(1);
2304       icc_color=MagickFalse;
2305       if (LocaleNCompare(colorspace,"device-",7) == 0)
2306         {
2307           (void) CopyMagickString(colorspace,colorspace+7,MaxTextExtent);
2308           scale=(double) QuantumRange;
2309           icc_color=MagickTrue;
2310         }
2311       if (LocaleCompare(colorspace,"icc-color") == 0)
2312         {
2313           register ssize_t
2314             j;
2315
2316           (void) CopyMagickString(colorspace,name+i+2,MaxTextExtent);
2317           for (j=0; colorspace[j] != '\0'; j++)
2318             if (colorspace[j] == ',')
2319               break;
2320           colorspace[j--]='\0';
2321           i+=j+3;
2322           scale=(double) QuantumRange;
2323           icc_color=MagickTrue;
2324         }
2325       LocaleLower(colorspace);
2326       color->alpha_trait=UndefinedPixelTrait;
2327       if ((i > 0) && (colorspace[i] == 'a'))
2328         {
2329           colorspace[i]='\0';
2330           color->alpha_trait=BlendPixelTrait;
2331         }
2332       type=ParseCommandOption(MagickColorspaceOptions,MagickFalse,colorspace);
2333       if (type < 0)
2334         {
2335           (void) ThrowMagickException(exception,GetMagickModule(),
2336             OptionWarning,"UnrecognizedColor","`%s'",name);
2337           return(MagickFalse);
2338         }
2339       color->colorspace=(ColorspaceType) type;
2340       if (IfMagickFalse(icc_color) && (color->colorspace == RGBColorspace))
2341         {
2342           color->colorspace=sRGBColorspace;  /* as required by SVG standard */
2343           color->depth=8;
2344         }
2345       SetGeometryInfo(&geometry_info);
2346       flags=ParseGeometry(name+i+1,&geometry_info);
2347       if (flags == 0)
2348         {
2349           char
2350             *colorname;
2351
2352           ColorspaceType
2353             colorspace;
2354
2355           colorspace=color->colorspace;
2356           colorname=AcquireString(name+i+1);
2357           (void) SubstituteString(&colorname,")","");
2358           (void) QueryColorCompliance(colorname,AllCompliance,color,exception);
2359           colorname=DestroyString(colorname);
2360           color->colorspace=colorspace;
2361         }
2362       else
2363         {
2364           if ((flags & PercentValue) != 0)
2365             scale=(double) (QuantumRange/100.0);
2366           if ((flags & RhoValue) != 0)
2367             color->red=(double) ClampToQuantum(scale*geometry_info.rho);
2368           if ((flags & SigmaValue) != 0)
2369             color->green=(double) ClampToQuantum(scale*
2370               geometry_info.sigma);
2371           if ((flags & XiValue) != 0)
2372             color->blue=(double) ClampToQuantum(scale*geometry_info.xi);
2373           color->alpha=(double) OpaqueAlpha;
2374           if ((flags & PsiValue) != 0)
2375             {
2376               if (color->colorspace == CMYKColorspace)
2377                 color->black=(double) ClampToQuantum(scale*
2378                   geometry_info.psi);
2379               else
2380                 if (color->alpha_trait == BlendPixelTrait)
2381                   color->alpha=(double) ClampToQuantum(QuantumRange*
2382                     geometry_info.psi);
2383             }
2384           if (((flags & ChiValue) != 0) &&
2385               (color->alpha_trait == BlendPixelTrait))
2386             color->alpha=(double) ClampToQuantum(QuantumRange*
2387               geometry_info.chi);
2388           if (LocaleCompare(colorspace,"gray") == 0)
2389             {
2390               color->colorspace=GRAYColorspace;
2391               color->green=color->red;
2392               color->blue=color->red;
2393               if (((flags & SigmaValue) != 0) &&
2394                   (color->alpha_trait == BlendPixelTrait))
2395                 color->alpha=(double) ClampToQuantum(QuantumRange*
2396                   geometry_info.sigma);
2397             }
2398           if ((LocaleCompare(colorspace,"HCL") == 0) ||
2399               (LocaleCompare(colorspace,"HSB") == 0) ||
2400               (LocaleCompare(colorspace,"HSL") == 0) ||
2401               (LocaleCompare(colorspace,"HWB") == 0))
2402             {
2403               double
2404                 blue,
2405                 green,
2406                 red;
2407
2408               if (LocaleCompare(colorspace,"HCL") == 0)
2409                 color->colorspace=HCLColorspace;
2410               else
2411                 if (LocaleCompare(colorspace,"HSB") == 0)
2412                   color->colorspace=HSBColorspace;
2413                 else
2414                   if (LocaleCompare(colorspace,"HSL") == 0)
2415                     color->colorspace=HSLColorspace;
2416                   else
2417                     if (LocaleCompare(colorspace,"HWB") == 0)
2418                       color->colorspace=HWBColorspace;
2419               scale=1.0/360.0;
2420               if ((flags & PercentValue) != 0)
2421                 scale=1.0/100.0;
2422               geometry_info.rho*=360.0*scale;
2423               scale=1.0/255.0;
2424               if ((flags & PercentValue) != 0)
2425                 scale=1.0/100.0;
2426               geometry_info.sigma*=scale;
2427               geometry_info.xi*=scale;
2428               if (LocaleCompare(colorspace,"HCL") == 0)
2429                 ConvertHCLToRGB(fmod(fmod(geometry_info.rho,360.0)+360.0,
2430                   360.0)/360.0,geometry_info.sigma,geometry_info.xi,&red,
2431                   &green,&blue);
2432               else
2433                 if (LocaleCompare(colorspace,"HSB") == 0)
2434                   ConvertHSBToRGB(fmod(fmod(geometry_info.rho,360.0)+360.0,
2435                     360.0)/360.0,geometry_info.sigma,geometry_info.xi,&red,
2436                     &green,&blue);
2437                 else
2438                   if (LocaleCompare(colorspace,"HSL") == 0)
2439                     ConvertHSLToRGB(fmod(fmod(geometry_info.rho,360.0)+360.0,
2440                       360.0)/360.0,geometry_info.sigma,geometry_info.xi,&red,
2441                       &green,&blue);
2442                   else
2443                     ConvertHWBToRGB(fmod(fmod(geometry_info.rho,360.0)+360.0,
2444                       360.0)/360.0,geometry_info.sigma,geometry_info.xi,&red,
2445                       &green,&blue);
2446               color->colorspace=sRGBColorspace;
2447               color->red=(MagickRealType) red;
2448               color->green=(MagickRealType) green;
2449               color->blue=(MagickRealType) blue;
2450             }
2451         }
2452       return(MagickTrue);
2453     }
2454   /*
2455     Parse named color.
2456   */
2457   p=GetColorCompliance(name,compliance,exception);
2458   if (p == (const ColorInfo *) NULL)
2459     return(MagickFalse);
2460   color->colorspace=sRGBColorspace;
2461   color->depth=8;
2462   color->alpha_trait=p->color.alpha != OpaqueAlpha ? BlendPixelTrait :
2463     UndefinedPixelTrait;
2464   color->red=(double) p->color.red;
2465   color->green=(double) p->color.green;
2466   color->blue=(double) p->color.blue;
2467   color->alpha=(double) p->color.alpha;
2468   color->black=0.0;
2469   return(MagickTrue);
2470 }
2471 \f
2472 /*
2473 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2474 %                                                                             %
2475 %                                                                             %
2476 %                                                                             %
2477 %  Q u e r y C o l o r n a m e                                                %
2478 %                                                                             %
2479 %                                                                             %
2480 %                                                                             %
2481 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2482 %
2483 %  QueryColorname() returns a named color for the given color intensity.
2484 %  If an exact match is not found, a hex value is returned instead.  For
2485 %  example an intensity of rgb:(0,0,0) returns black whereas rgb:(223,223,223)
2486 %  returns #dfdfdf.
2487 %
2488 %  UPDATE: the 'image' argument is no longer needed as all information should
2489 %  have been preset using GetPixelInfo().
2490 %
2491 %  The format of the QueryColorname method is:
2492 %
2493 %      MagickBooleanType QueryColorname(const Image *image,
2494 %        const PixelInfo *color,const ComplianceType compliance,char *name,
2495 %        ExceptionInfo *exception)
2496 %
2497 %  A description of each parameter follows.
2498 %
2499 %    o image: the image. (not used! - color gets settings from GetPixelInfo()
2500 %
2501 %    o color: the color intensities.
2502 %
2503 %    o Compliance: Adhere to this color standard: SVG, X11, or XPM.
2504 %
2505 %    o name: Return the color name or hex value.
2506 %
2507 %    o exception: return any errors or warnings in this structure.
2508 %
2509 */
2510
2511 MagickExport MagickBooleanType QueryColorname(
2512   const Image *magick_unused(image),const PixelInfo *color,
2513   const ComplianceType compliance,char *name,ExceptionInfo *exception)
2514 {
2515   PixelInfo
2516     pixel;
2517
2518   double
2519     alpha;
2520
2521   register const ColorInfo
2522     *p;
2523
2524   *name='\0';
2525   pixel=(*color);
2526   if (compliance == XPMCompliance)
2527     {
2528       pixel.alpha_trait=UndefinedPixelTrait;
2529       if ( pixel.depth > 16 )
2530         pixel.depth=16;
2531     }
2532   GetColorTuple(&pixel,IsMagickTrue(compliance != SVGCompliance),name);
2533   if (IfMagickFalse(IssRGBColorspace(pixel.colorspace)))
2534     return(MagickFalse);
2535   alpha=color->alpha_trait == BlendPixelTrait ? color->alpha : OpaqueAlpha;
2536   (void) GetColorInfo("*",exception);
2537   ResetLinkedListIterator(color_list);
2538   p=(const ColorInfo *) GetNextValueInLinkedList(color_list);
2539   while (p != (const ColorInfo *) NULL)
2540   {
2541     if (((p->compliance & compliance) != 0) &&
2542         ((fabs(p->color.red-color->red) < MagickEpsilon)) &&
2543          (fabs(p->color.green-color->green) < MagickEpsilon) &&
2544          (fabs(p->color.blue-color->blue) < MagickEpsilon) &&
2545          (fabs(p->color.alpha-alpha) < MagickEpsilon))
2546       {
2547         (void) CopyMagickString(name,p->name,MaxTextExtent);
2548         break;
2549       }
2550     p=(const ColorInfo *) GetNextValueInLinkedList(color_list);
2551   }
2552   return(MagickTrue);
2553 }