Qt
Internal/Contributor docs for the Qt SDK. <b>Note:</b> These are NOT official API docs; those are found <a href='https://doc.qt.io/'>here</a>.
Loading...
Searching...
No Matches
qxpmhandler.cpp
Go to the documentation of this file.
1// Copyright (C) 2016 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3
4#include "private/qxpmhandler_p.h"
5
6#ifndef QT_NO_IMAGEFORMAT_XPM
7
8#include <qbytearraymatcher.h>
9#include <qdebug.h>
10#include <qimage.h>
11#include <qloggingcategory.h>
12#include <qmap.h>
13#include <qtextstream.h>
14#include <qvariant.h>
15
16#include <private/qcolor_p.h>
17#include <private/qduplicatetracker_p.h> // for easier std::pmr detection
18#include <private/qtools_p.h>
19
20#include <algorithm>
21#include <array>
22
24
25using namespace QtMiscUtils;
26
28
29static quint64 xpmHash(const QString &str)
30{
31 unsigned int hashValue = 0;
32 for (int i = 0; i < str.size(); ++i) {
33 hashValue <<= 8;
34 hashValue += (unsigned int)str.at(i).unicode();
35 }
36 return hashValue;
37}
38static quint64 xpmHash(char *str)
39{
40 unsigned int hashValue = 0;
41 while (*str != '\0') {
42 hashValue <<= 8;
43 hashValue += (unsigned int)*str;
44 ++str;
45 }
46 return hashValue;
47}
48
49#ifdef QRGB
50#undef QRGB
51#endif
52#define QRGB(r,g,b) (r*65536 + g*256 + b)
53
54static const int xpmRgbTblSize = 657;
55
56static const struct XPMRGBData {
58 const char name[21];
59} xpmRgbTbl[] = {
60 { QRGB(240,248,255), "aliceblue" },
61 { QRGB(250,235,215), "antiquewhite" },
62 { QRGB(255,239,219), "antiquewhite1" },
63 { QRGB(238,223,204), "antiquewhite2" },
64 { QRGB(205,192,176), "antiquewhite3" },
65 { QRGB(139,131,120), "antiquewhite4" },
66 { QRGB(127,255,212), "aquamarine" },
67 { QRGB(127,255,212), "aquamarine1" },
68 { QRGB(118,238,198), "aquamarine2" },
69 { QRGB(102,205,170), "aquamarine3" },
70 { QRGB( 69,139,116), "aquamarine4" },
71 { QRGB(240,255,255), "azure" },
72 { QRGB(240,255,255), "azure1" },
73 { QRGB(224,238,238), "azure2" },
74 { QRGB(193,205,205), "azure3" },
75 { QRGB(131,139,139), "azure4" },
76 { QRGB(245,245,220), "beige" },
77 { QRGB(255,228,196), "bisque" },
78 { QRGB(255,228,196), "bisque1" },
79 { QRGB(238,213,183), "bisque2" },
80 { QRGB(205,183,158), "bisque3" },
81 { QRGB(139,125,107), "bisque4" },
82 { QRGB( 0, 0, 0), "black" },
83 { QRGB(255,235,205), "blanchedalmond" },
84 { QRGB( 0, 0,255), "blue" },
85 { QRGB( 0, 0,255), "blue1" },
86 { QRGB( 0, 0,238), "blue2" },
87 { QRGB( 0, 0,205), "blue3" },
88 { QRGB( 0, 0,139), "blue4" },
89 { QRGB(138, 43,226), "blueviolet" },
90 { QRGB(165, 42, 42), "brown" },
91 { QRGB(255, 64, 64), "brown1" },
92 { QRGB(238, 59, 59), "brown2" },
93 { QRGB(205, 51, 51), "brown3" },
94 { QRGB(139, 35, 35), "brown4" },
95 { QRGB(222,184,135), "burlywood" },
96 { QRGB(255,211,155), "burlywood1" },
97 { QRGB(238,197,145), "burlywood2" },
98 { QRGB(205,170,125), "burlywood3" },
99 { QRGB(139,115, 85), "burlywood4" },
100 { QRGB( 95,158,160), "cadetblue" },
101 { QRGB(152,245,255), "cadetblue1" },
102 { QRGB(142,229,238), "cadetblue2" },
103 { QRGB(122,197,205), "cadetblue3" },
104 { QRGB( 83,134,139), "cadetblue4" },
105 { QRGB(127,255, 0), "chartreuse" },
106 { QRGB(127,255, 0), "chartreuse1" },
107 { QRGB(118,238, 0), "chartreuse2" },
108 { QRGB(102,205, 0), "chartreuse3" },
109 { QRGB( 69,139, 0), "chartreuse4" },
110 { QRGB(210,105, 30), "chocolate" },
111 { QRGB(255,127, 36), "chocolate1" },
112 { QRGB(238,118, 33), "chocolate2" },
113 { QRGB(205,102, 29), "chocolate3" },
114 { QRGB(139, 69, 19), "chocolate4" },
115 { QRGB(255,127, 80), "coral" },
116 { QRGB(255,114, 86), "coral1" },
117 { QRGB(238,106, 80), "coral2" },
118 { QRGB(205, 91, 69), "coral3" },
119 { QRGB(139, 62, 47), "coral4" },
120 { QRGB(100,149,237), "cornflowerblue" },
121 { QRGB(255,248,220), "cornsilk" },
122 { QRGB(255,248,220), "cornsilk1" },
123 { QRGB(238,232,205), "cornsilk2" },
124 { QRGB(205,200,177), "cornsilk3" },
125 { QRGB(139,136,120), "cornsilk4" },
126 { QRGB( 0,255,255), "cyan" },
127 { QRGB( 0,255,255), "cyan1" },
128 { QRGB( 0,238,238), "cyan2" },
129 { QRGB( 0,205,205), "cyan3" },
130 { QRGB( 0,139,139), "cyan4" },
131 { QRGB( 0, 0,139), "darkblue" },
132 { QRGB( 0,139,139), "darkcyan" },
133 { QRGB(184,134, 11), "darkgoldenrod" },
134 { QRGB(255,185, 15), "darkgoldenrod1" },
135 { QRGB(238,173, 14), "darkgoldenrod2" },
136 { QRGB(205,149, 12), "darkgoldenrod3" },
137 { QRGB(139,101, 8), "darkgoldenrod4" },
138 { QRGB(169,169,169), "darkgray" },
139 { QRGB( 0,100, 0), "darkgreen" },
140 { QRGB(169,169,169), "darkgrey" },
141 { QRGB(189,183,107), "darkkhaki" },
142 { QRGB(139, 0,139), "darkmagenta" },
143 { QRGB( 85,107, 47), "darkolivegreen" },
144 { QRGB(202,255,112), "darkolivegreen1" },
145 { QRGB(188,238,104), "darkolivegreen2" },
146 { QRGB(162,205, 90), "darkolivegreen3" },
147 { QRGB(110,139, 61), "darkolivegreen4" },
148 { QRGB(255,140, 0), "darkorange" },
149 { QRGB(255,127, 0), "darkorange1" },
150 { QRGB(238,118, 0), "darkorange2" },
151 { QRGB(205,102, 0), "darkorange3" },
152 { QRGB(139, 69, 0), "darkorange4" },
153 { QRGB(153, 50,204), "darkorchid" },
154 { QRGB(191, 62,255), "darkorchid1" },
155 { QRGB(178, 58,238), "darkorchid2" },
156 { QRGB(154, 50,205), "darkorchid3" },
157 { QRGB(104, 34,139), "darkorchid4" },
158 { QRGB(139, 0, 0), "darkred" },
159 { QRGB(233,150,122), "darksalmon" },
160 { QRGB(143,188,143), "darkseagreen" },
161 { QRGB(193,255,193), "darkseagreen1" },
162 { QRGB(180,238,180), "darkseagreen2" },
163 { QRGB(155,205,155), "darkseagreen3" },
164 { QRGB(105,139,105), "darkseagreen4" },
165 { QRGB( 72, 61,139), "darkslateblue" },
166 { QRGB( 47, 79, 79), "darkslategray" },
167 { QRGB(151,255,255), "darkslategray1" },
168 { QRGB(141,238,238), "darkslategray2" },
169 { QRGB(121,205,205), "darkslategray3" },
170 { QRGB( 82,139,139), "darkslategray4" },
171 { QRGB( 47, 79, 79), "darkslategrey" },
172 { QRGB( 0,206,209), "darkturquoise" },
173 { QRGB(148, 0,211), "darkviolet" },
174 { QRGB(255, 20,147), "deeppink" },
175 { QRGB(255, 20,147), "deeppink1" },
176 { QRGB(238, 18,137), "deeppink2" },
177 { QRGB(205, 16,118), "deeppink3" },
178 { QRGB(139, 10, 80), "deeppink4" },
179 { QRGB( 0,191,255), "deepskyblue" },
180 { QRGB( 0,191,255), "deepskyblue1" },
181 { QRGB( 0,178,238), "deepskyblue2" },
182 { QRGB( 0,154,205), "deepskyblue3" },
183 { QRGB( 0,104,139), "deepskyblue4" },
184 { QRGB(105,105,105), "dimgray" },
185 { QRGB(105,105,105), "dimgrey" },
186 { QRGB( 30,144,255), "dodgerblue" },
187 { QRGB( 30,144,255), "dodgerblue1" },
188 { QRGB( 28,134,238), "dodgerblue2" },
189 { QRGB( 24,116,205), "dodgerblue3" },
190 { QRGB( 16, 78,139), "dodgerblue4" },
191 { QRGB(178, 34, 34), "firebrick" },
192 { QRGB(255, 48, 48), "firebrick1" },
193 { QRGB(238, 44, 44), "firebrick2" },
194 { QRGB(205, 38, 38), "firebrick3" },
195 { QRGB(139, 26, 26), "firebrick4" },
196 { QRGB(255,250,240), "floralwhite" },
197 { QRGB( 34,139, 34), "forestgreen" },
198 { QRGB(220,220,220), "gainsboro" },
199 { QRGB(248,248,255), "ghostwhite" },
200 { QRGB(255,215, 0), "gold" },
201 { QRGB(255,215, 0), "gold1" },
202 { QRGB(238,201, 0), "gold2" },
203 { QRGB(205,173, 0), "gold3" },
204 { QRGB(139,117, 0), "gold4" },
205 { QRGB(218,165, 32), "goldenrod" },
206 { QRGB(255,193, 37), "goldenrod1" },
207 { QRGB(238,180, 34), "goldenrod2" },
208 { QRGB(205,155, 29), "goldenrod3" },
209 { QRGB(139,105, 20), "goldenrod4" },
210 { QRGB(190,190,190), "gray" },
211 { QRGB( 0, 0, 0), "gray0" },
212 { QRGB( 3, 3, 3), "gray1" },
213 { QRGB( 26, 26, 26), "gray10" },
214 { QRGB(255,255,255), "gray100" },
215 { QRGB( 28, 28, 28), "gray11" },
216 { QRGB( 31, 31, 31), "gray12" },
217 { QRGB( 33, 33, 33), "gray13" },
218 { QRGB( 36, 36, 36), "gray14" },
219 { QRGB( 38, 38, 38), "gray15" },
220 { QRGB( 41, 41, 41), "gray16" },
221 { QRGB( 43, 43, 43), "gray17" },
222 { QRGB( 46, 46, 46), "gray18" },
223 { QRGB( 48, 48, 48), "gray19" },
224 { QRGB( 5, 5, 5), "gray2" },
225 { QRGB( 51, 51, 51), "gray20" },
226 { QRGB( 54, 54, 54), "gray21" },
227 { QRGB( 56, 56, 56), "gray22" },
228 { QRGB( 59, 59, 59), "gray23" },
229 { QRGB( 61, 61, 61), "gray24" },
230 { QRGB( 64, 64, 64), "gray25" },
231 { QRGB( 66, 66, 66), "gray26" },
232 { QRGB( 69, 69, 69), "gray27" },
233 { QRGB( 71, 71, 71), "gray28" },
234 { QRGB( 74, 74, 74), "gray29" },
235 { QRGB( 8, 8, 8), "gray3" },
236 { QRGB( 77, 77, 77), "gray30" },
237 { QRGB( 79, 79, 79), "gray31" },
238 { QRGB( 82, 82, 82), "gray32" },
239 { QRGB( 84, 84, 84), "gray33" },
240 { QRGB( 87, 87, 87), "gray34" },
241 { QRGB( 89, 89, 89), "gray35" },
242 { QRGB( 92, 92, 92), "gray36" },
243 { QRGB( 94, 94, 94), "gray37" },
244 { QRGB( 97, 97, 97), "gray38" },
245 { QRGB( 99, 99, 99), "gray39" },
246 { QRGB( 10, 10, 10), "gray4" },
247 { QRGB(102,102,102), "gray40" },
248 { QRGB(105,105,105), "gray41" },
249 { QRGB(107,107,107), "gray42" },
250 { QRGB(110,110,110), "gray43" },
251 { QRGB(112,112,112), "gray44" },
252 { QRGB(115,115,115), "gray45" },
253 { QRGB(117,117,117), "gray46" },
254 { QRGB(120,120,120), "gray47" },
255 { QRGB(122,122,122), "gray48" },
256 { QRGB(125,125,125), "gray49" },
257 { QRGB( 13, 13, 13), "gray5" },
258 { QRGB(127,127,127), "gray50" },
259 { QRGB(130,130,130), "gray51" },
260 { QRGB(133,133,133), "gray52" },
261 { QRGB(135,135,135), "gray53" },
262 { QRGB(138,138,138), "gray54" },
263 { QRGB(140,140,140), "gray55" },
264 { QRGB(143,143,143), "gray56" },
265 { QRGB(145,145,145), "gray57" },
266 { QRGB(148,148,148), "gray58" },
267 { QRGB(150,150,150), "gray59" },
268 { QRGB( 15, 15, 15), "gray6" },
269 { QRGB(153,153,153), "gray60" },
270 { QRGB(156,156,156), "gray61" },
271 { QRGB(158,158,158), "gray62" },
272 { QRGB(161,161,161), "gray63" },
273 { QRGB(163,163,163), "gray64" },
274 { QRGB(166,166,166), "gray65" },
275 { QRGB(168,168,168), "gray66" },
276 { QRGB(171,171,171), "gray67" },
277 { QRGB(173,173,173), "gray68" },
278 { QRGB(176,176,176), "gray69" },
279 { QRGB( 18, 18, 18), "gray7" },
280 { QRGB(179,179,179), "gray70" },
281 { QRGB(181,181,181), "gray71" },
282 { QRGB(184,184,184), "gray72" },
283 { QRGB(186,186,186), "gray73" },
284 { QRGB(189,189,189), "gray74" },
285 { QRGB(191,191,191), "gray75" },
286 { QRGB(194,194,194), "gray76" },
287 { QRGB(196,196,196), "gray77" },
288 { QRGB(199,199,199), "gray78" },
289 { QRGB(201,201,201), "gray79" },
290 { QRGB( 20, 20, 20), "gray8" },
291 { QRGB(204,204,204), "gray80" },
292 { QRGB(207,207,207), "gray81" },
293 { QRGB(209,209,209), "gray82" },
294 { QRGB(212,212,212), "gray83" },
295 { QRGB(214,214,214), "gray84" },
296 { QRGB(217,217,217), "gray85" },
297 { QRGB(219,219,219), "gray86" },
298 { QRGB(222,222,222), "gray87" },
299 { QRGB(224,224,224), "gray88" },
300 { QRGB(227,227,227), "gray89" },
301 { QRGB( 23, 23, 23), "gray9" },
302 { QRGB(229,229,229), "gray90" },
303 { QRGB(232,232,232), "gray91" },
304 { QRGB(235,235,235), "gray92" },
305 { QRGB(237,237,237), "gray93" },
306 { QRGB(240,240,240), "gray94" },
307 { QRGB(242,242,242), "gray95" },
308 { QRGB(245,245,245), "gray96" },
309 { QRGB(247,247,247), "gray97" },
310 { QRGB(250,250,250), "gray98" },
311 { QRGB(252,252,252), "gray99" },
312 { QRGB( 0,255, 0), "green" },
313 { QRGB( 0,255, 0), "green1" },
314 { QRGB( 0,238, 0), "green2" },
315 { QRGB( 0,205, 0), "green3" },
316 { QRGB( 0,139, 0), "green4" },
317 { QRGB(173,255, 47), "greenyellow" },
318 { QRGB(190,190,190), "grey" },
319 { QRGB( 0, 0, 0), "grey0" },
320 { QRGB( 3, 3, 3), "grey1" },
321 { QRGB( 26, 26, 26), "grey10" },
322 { QRGB(255,255,255), "grey100" },
323 { QRGB( 28, 28, 28), "grey11" },
324 { QRGB( 31, 31, 31), "grey12" },
325 { QRGB( 33, 33, 33), "grey13" },
326 { QRGB( 36, 36, 36), "grey14" },
327 { QRGB( 38, 38, 38), "grey15" },
328 { QRGB( 41, 41, 41), "grey16" },
329 { QRGB( 43, 43, 43), "grey17" },
330 { QRGB( 46, 46, 46), "grey18" },
331 { QRGB( 48, 48, 48), "grey19" },
332 { QRGB( 5, 5, 5), "grey2" },
333 { QRGB( 51, 51, 51), "grey20" },
334 { QRGB( 54, 54, 54), "grey21" },
335 { QRGB( 56, 56, 56), "grey22" },
336 { QRGB( 59, 59, 59), "grey23" },
337 { QRGB( 61, 61, 61), "grey24" },
338 { QRGB( 64, 64, 64), "grey25" },
339 { QRGB( 66, 66, 66), "grey26" },
340 { QRGB( 69, 69, 69), "grey27" },
341 { QRGB( 71, 71, 71), "grey28" },
342 { QRGB( 74, 74, 74), "grey29" },
343 { QRGB( 8, 8, 8), "grey3" },
344 { QRGB( 77, 77, 77), "grey30" },
345 { QRGB( 79, 79, 79), "grey31" },
346 { QRGB( 82, 82, 82), "grey32" },
347 { QRGB( 84, 84, 84), "grey33" },
348 { QRGB( 87, 87, 87), "grey34" },
349 { QRGB( 89, 89, 89), "grey35" },
350 { QRGB( 92, 92, 92), "grey36" },
351 { QRGB( 94, 94, 94), "grey37" },
352 { QRGB( 97, 97, 97), "grey38" },
353 { QRGB( 99, 99, 99), "grey39" },
354 { QRGB( 10, 10, 10), "grey4" },
355 { QRGB(102,102,102), "grey40" },
356 { QRGB(105,105,105), "grey41" },
357 { QRGB(107,107,107), "grey42" },
358 { QRGB(110,110,110), "grey43" },
359 { QRGB(112,112,112), "grey44" },
360 { QRGB(115,115,115), "grey45" },
361 { QRGB(117,117,117), "grey46" },
362 { QRGB(120,120,120), "grey47" },
363 { QRGB(122,122,122), "grey48" },
364 { QRGB(125,125,125), "grey49" },
365 { QRGB( 13, 13, 13), "grey5" },
366 { QRGB(127,127,127), "grey50" },
367 { QRGB(130,130,130), "grey51" },
368 { QRGB(133,133,133), "grey52" },
369 { QRGB(135,135,135), "grey53" },
370 { QRGB(138,138,138), "grey54" },
371 { QRGB(140,140,140), "grey55" },
372 { QRGB(143,143,143), "grey56" },
373 { QRGB(145,145,145), "grey57" },
374 { QRGB(148,148,148), "grey58" },
375 { QRGB(150,150,150), "grey59" },
376 { QRGB( 15, 15, 15), "grey6" },
377 { QRGB(153,153,153), "grey60" },
378 { QRGB(156,156,156), "grey61" },
379 { QRGB(158,158,158), "grey62" },
380 { QRGB(161,161,161), "grey63" },
381 { QRGB(163,163,163), "grey64" },
382 { QRGB(166,166,166), "grey65" },
383 { QRGB(168,168,168), "grey66" },
384 { QRGB(171,171,171), "grey67" },
385 { QRGB(173,173,173), "grey68" },
386 { QRGB(176,176,176), "grey69" },
387 { QRGB( 18, 18, 18), "grey7" },
388 { QRGB(179,179,179), "grey70" },
389 { QRGB(181,181,181), "grey71" },
390 { QRGB(184,184,184), "grey72" },
391 { QRGB(186,186,186), "grey73" },
392 { QRGB(189,189,189), "grey74" },
393 { QRGB(191,191,191), "grey75" },
394 { QRGB(194,194,194), "grey76" },
395 { QRGB(196,196,196), "grey77" },
396 { QRGB(199,199,199), "grey78" },
397 { QRGB(201,201,201), "grey79" },
398 { QRGB( 20, 20, 20), "grey8" },
399 { QRGB(204,204,204), "grey80" },
400 { QRGB(207,207,207), "grey81" },
401 { QRGB(209,209,209), "grey82" },
402 { QRGB(212,212,212), "grey83" },
403 { QRGB(214,214,214), "grey84" },
404 { QRGB(217,217,217), "grey85" },
405 { QRGB(219,219,219), "grey86" },
406 { QRGB(222,222,222), "grey87" },
407 { QRGB(224,224,224), "grey88" },
408 { QRGB(227,227,227), "grey89" },
409 { QRGB( 23, 23, 23), "grey9" },
410 { QRGB(229,229,229), "grey90" },
411 { QRGB(232,232,232), "grey91" },
412 { QRGB(235,235,235), "grey92" },
413 { QRGB(237,237,237), "grey93" },
414 { QRGB(240,240,240), "grey94" },
415 { QRGB(242,242,242), "grey95" },
416 { QRGB(245,245,245), "grey96" },
417 { QRGB(247,247,247), "grey97" },
418 { QRGB(250,250,250), "grey98" },
419 { QRGB(252,252,252), "grey99" },
420 { QRGB(240,255,240), "honeydew" },
421 { QRGB(240,255,240), "honeydew1" },
422 { QRGB(224,238,224), "honeydew2" },
423 { QRGB(193,205,193), "honeydew3" },
424 { QRGB(131,139,131), "honeydew4" },
425 { QRGB(255,105,180), "hotpink" },
426 { QRGB(255,110,180), "hotpink1" },
427 { QRGB(238,106,167), "hotpink2" },
428 { QRGB(205, 96,144), "hotpink3" },
429 { QRGB(139, 58, 98), "hotpink4" },
430 { QRGB(205, 92, 92), "indianred" },
431 { QRGB(255,106,106), "indianred1" },
432 { QRGB(238, 99, 99), "indianred2" },
433 { QRGB(205, 85, 85), "indianred3" },
434 { QRGB(139, 58, 58), "indianred4" },
435 { QRGB(255,255,240), "ivory" },
436 { QRGB(255,255,240), "ivory1" },
437 { QRGB(238,238,224), "ivory2" },
438 { QRGB(205,205,193), "ivory3" },
439 { QRGB(139,139,131), "ivory4" },
440 { QRGB(240,230,140), "khaki" },
441 { QRGB(255,246,143), "khaki1" },
442 { QRGB(238,230,133), "khaki2" },
443 { QRGB(205,198,115), "khaki3" },
444 { QRGB(139,134, 78), "khaki4" },
445 { QRGB(230,230,250), "lavender" },
446 { QRGB(255,240,245), "lavenderblush" },
447 { QRGB(255,240,245), "lavenderblush1" },
448 { QRGB(238,224,229), "lavenderblush2" },
449 { QRGB(205,193,197), "lavenderblush3" },
450 { QRGB(139,131,134), "lavenderblush4" },
451 { QRGB(124,252, 0), "lawngreen" },
452 { QRGB(255,250,205), "lemonchiffon" },
453 { QRGB(255,250,205), "lemonchiffon1" },
454 { QRGB(238,233,191), "lemonchiffon2" },
455 { QRGB(205,201,165), "lemonchiffon3" },
456 { QRGB(139,137,112), "lemonchiffon4" },
457 { QRGB(173,216,230), "lightblue" },
458 { QRGB(191,239,255), "lightblue1" },
459 { QRGB(178,223,238), "lightblue2" },
460 { QRGB(154,192,205), "lightblue3" },
461 { QRGB(104,131,139), "lightblue4" },
462 { QRGB(240,128,128), "lightcoral" },
463 { QRGB(224,255,255), "lightcyan" },
464 { QRGB(224,255,255), "lightcyan1" },
465 { QRGB(209,238,238), "lightcyan2" },
466 { QRGB(180,205,205), "lightcyan3" },
467 { QRGB(122,139,139), "lightcyan4" },
468 { QRGB(238,221,130), "lightgoldenrod" },
469 { QRGB(255,236,139), "lightgoldenrod1" },
470 { QRGB(238,220,130), "lightgoldenrod2" },
471 { QRGB(205,190,112), "lightgoldenrod3" },
472 { QRGB(139,129, 76), "lightgoldenrod4" },
473 { QRGB(250,250,210), "lightgoldenrodyellow" },
474 { QRGB(211,211,211), "lightgray" },
475 { QRGB(144,238,144), "lightgreen" },
476 { QRGB(211,211,211), "lightgrey" },
477 { QRGB(255,182,193), "lightpink" },
478 { QRGB(255,174,185), "lightpink1" },
479 { QRGB(238,162,173), "lightpink2" },
480 { QRGB(205,140,149), "lightpink3" },
481 { QRGB(139, 95,101), "lightpink4" },
482 { QRGB(255,160,122), "lightsalmon" },
483 { QRGB(255,160,122), "lightsalmon1" },
484 { QRGB(238,149,114), "lightsalmon2" },
485 { QRGB(205,129, 98), "lightsalmon3" },
486 { QRGB(139, 87, 66), "lightsalmon4" },
487 { QRGB( 32,178,170), "lightseagreen" },
488 { QRGB(135,206,250), "lightskyblue" },
489 { QRGB(176,226,255), "lightskyblue1" },
490 { QRGB(164,211,238), "lightskyblue2" },
491 { QRGB(141,182,205), "lightskyblue3" },
492 { QRGB( 96,123,139), "lightskyblue4" },
493 { QRGB(132,112,255), "lightslateblue" },
494 { QRGB(119,136,153), "lightslategray" },
495 { QRGB(119,136,153), "lightslategrey" },
496 { QRGB(176,196,222), "lightsteelblue" },
497 { QRGB(202,225,255), "lightsteelblue1" },
498 { QRGB(188,210,238), "lightsteelblue2" },
499 { QRGB(162,181,205), "lightsteelblue3" },
500 { QRGB(110,123,139), "lightsteelblue4" },
501 { QRGB(255,255,224), "lightyellow" },
502 { QRGB(255,255,224), "lightyellow1" },
503 { QRGB(238,238,209), "lightyellow2" },
504 { QRGB(205,205,180), "lightyellow3" },
505 { QRGB(139,139,122), "lightyellow4" },
506 { QRGB( 50,205, 50), "limegreen" },
507 { QRGB(250,240,230), "linen" },
508 { QRGB(255, 0,255), "magenta" },
509 { QRGB(255, 0,255), "magenta1" },
510 { QRGB(238, 0,238), "magenta2" },
511 { QRGB(205, 0,205), "magenta3" },
512 { QRGB(139, 0,139), "magenta4" },
513 { QRGB(176, 48, 96), "maroon" },
514 { QRGB(255, 52,179), "maroon1" },
515 { QRGB(238, 48,167), "maroon2" },
516 { QRGB(205, 41,144), "maroon3" },
517 { QRGB(139, 28, 98), "maroon4" },
518 { QRGB(102,205,170), "mediumaquamarine" },
519 { QRGB( 0, 0,205), "mediumblue" },
520 { QRGB(186, 85,211), "mediumorchid" },
521 { QRGB(224,102,255), "mediumorchid1" },
522 { QRGB(209, 95,238), "mediumorchid2" },
523 { QRGB(180, 82,205), "mediumorchid3" },
524 { QRGB(122, 55,139), "mediumorchid4" },
525 { QRGB(147,112,219), "mediumpurple" },
526 { QRGB(171,130,255), "mediumpurple1" },
527 { QRGB(159,121,238), "mediumpurple2" },
528 { QRGB(137,104,205), "mediumpurple3" },
529 { QRGB( 93, 71,139), "mediumpurple4" },
530 { QRGB( 60,179,113), "mediumseagreen" },
531 { QRGB(123,104,238), "mediumslateblue" },
532 { QRGB( 0,250,154), "mediumspringgreen" },
533 { QRGB( 72,209,204), "mediumturquoise" },
534 { QRGB(199, 21,133), "mediumvioletred" },
535 { QRGB( 25, 25,112), "midnightblue" },
536 { QRGB(245,255,250), "mintcream" },
537 { QRGB(255,228,225), "mistyrose" },
538 { QRGB(255,228,225), "mistyrose1" },
539 { QRGB(238,213,210), "mistyrose2" },
540 { QRGB(205,183,181), "mistyrose3" },
541 { QRGB(139,125,123), "mistyrose4" },
542 { QRGB(255,228,181), "moccasin" },
543 { QRGB(255,222,173), "navajowhite" },
544 { QRGB(255,222,173), "navajowhite1" },
545 { QRGB(238,207,161), "navajowhite2" },
546 { QRGB(205,179,139), "navajowhite3" },
547 { QRGB(139,121, 94), "navajowhite4" },
548 { QRGB( 0, 0,128), "navy" },
549 { QRGB( 0, 0,128), "navyblue" },
550 { QRGB(253,245,230), "oldlace" },
551 { QRGB(107,142, 35), "olivedrab" },
552 { QRGB(192,255, 62), "olivedrab1" },
553 { QRGB(179,238, 58), "olivedrab2" },
554 { QRGB(154,205, 50), "olivedrab3" },
555 { QRGB(105,139, 34), "olivedrab4" },
556 { QRGB(255,165, 0), "orange" },
557 { QRGB(255,165, 0), "orange1" },
558 { QRGB(238,154, 0), "orange2" },
559 { QRGB(205,133, 0), "orange3" },
560 { QRGB(139, 90, 0), "orange4" },
561 { QRGB(255, 69, 0), "orangered" },
562 { QRGB(255, 69, 0), "orangered1" },
563 { QRGB(238, 64, 0), "orangered2" },
564 { QRGB(205, 55, 0), "orangered3" },
565 { QRGB(139, 37, 0), "orangered4" },
566 { QRGB(218,112,214), "orchid" },
567 { QRGB(255,131,250), "orchid1" },
568 { QRGB(238,122,233), "orchid2" },
569 { QRGB(205,105,201), "orchid3" },
570 { QRGB(139, 71,137), "orchid4" },
571 { QRGB(238,232,170), "palegoldenrod" },
572 { QRGB(152,251,152), "palegreen" },
573 { QRGB(154,255,154), "palegreen1" },
574 { QRGB(144,238,144), "palegreen2" },
575 { QRGB(124,205,124), "palegreen3" },
576 { QRGB( 84,139, 84), "palegreen4" },
577 { QRGB(175,238,238), "paleturquoise" },
578 { QRGB(187,255,255), "paleturquoise1" },
579 { QRGB(174,238,238), "paleturquoise2" },
580 { QRGB(150,205,205), "paleturquoise3" },
581 { QRGB(102,139,139), "paleturquoise4" },
582 { QRGB(219,112,147), "palevioletred" },
583 { QRGB(255,130,171), "palevioletred1" },
584 { QRGB(238,121,159), "palevioletred2" },
585 { QRGB(205,104,137), "palevioletred3" },
586 { QRGB(139, 71, 93), "palevioletred4" },
587 { QRGB(255,239,213), "papayawhip" },
588 { QRGB(255,218,185), "peachpuff" },
589 { QRGB(255,218,185), "peachpuff1" },
590 { QRGB(238,203,173), "peachpuff2" },
591 { QRGB(205,175,149), "peachpuff3" },
592 { QRGB(139,119,101), "peachpuff4" },
593 { QRGB(205,133, 63), "peru" },
594 { QRGB(255,192,203), "pink" },
595 { QRGB(255,181,197), "pink1" },
596 { QRGB(238,169,184), "pink2" },
597 { QRGB(205,145,158), "pink3" },
598 { QRGB(139, 99,108), "pink4" },
599 { QRGB(221,160,221), "plum" },
600 { QRGB(255,187,255), "plum1" },
601 { QRGB(238,174,238), "plum2" },
602 { QRGB(205,150,205), "plum3" },
603 { QRGB(139,102,139), "plum4" },
604 { QRGB(176,224,230), "powderblue" },
605 { QRGB(160, 32,240), "purple" },
606 { QRGB(155, 48,255), "purple1" },
607 { QRGB(145, 44,238), "purple2" },
608 { QRGB(125, 38,205), "purple3" },
609 { QRGB( 85, 26,139), "purple4" },
610 { QRGB(255, 0, 0), "red" },
611 { QRGB(255, 0, 0), "red1" },
612 { QRGB(238, 0, 0), "red2" },
613 { QRGB(205, 0, 0), "red3" },
614 { QRGB(139, 0, 0), "red4" },
615 { QRGB(188,143,143), "rosybrown" },
616 { QRGB(255,193,193), "rosybrown1" },
617 { QRGB(238,180,180), "rosybrown2" },
618 { QRGB(205,155,155), "rosybrown3" },
619 { QRGB(139,105,105), "rosybrown4" },
620 { QRGB( 65,105,225), "royalblue" },
621 { QRGB( 72,118,255), "royalblue1" },
622 { QRGB( 67,110,238), "royalblue2" },
623 { QRGB( 58, 95,205), "royalblue3" },
624 { QRGB( 39, 64,139), "royalblue4" },
625 { QRGB(139, 69, 19), "saddlebrown" },
626 { QRGB(250,128,114), "salmon" },
627 { QRGB(255,140,105), "salmon1" },
628 { QRGB(238,130, 98), "salmon2" },
629 { QRGB(205,112, 84), "salmon3" },
630 { QRGB(139, 76, 57), "salmon4" },
631 { QRGB(244,164, 96), "sandybrown" },
632 { QRGB( 46,139, 87), "seagreen" },
633 { QRGB( 84,255,159), "seagreen1" },
634 { QRGB( 78,238,148), "seagreen2" },
635 { QRGB( 67,205,128), "seagreen3" },
636 { QRGB( 46,139, 87), "seagreen4" },
637 { QRGB(255,245,238), "seashell" },
638 { QRGB(255,245,238), "seashell1" },
639 { QRGB(238,229,222), "seashell2" },
640 { QRGB(205,197,191), "seashell3" },
641 { QRGB(139,134,130), "seashell4" },
642 { QRGB(160, 82, 45), "sienna" },
643 { QRGB(255,130, 71), "sienna1" },
644 { QRGB(238,121, 66), "sienna2" },
645 { QRGB(205,104, 57), "sienna3" },
646 { QRGB(139, 71, 38), "sienna4" },
647 { QRGB(135,206,235), "skyblue" },
648 { QRGB(135,206,255), "skyblue1" },
649 { QRGB(126,192,238), "skyblue2" },
650 { QRGB(108,166,205), "skyblue3" },
651 { QRGB( 74,112,139), "skyblue4" },
652 { QRGB(106, 90,205), "slateblue" },
653 { QRGB(131,111,255), "slateblue1" },
654 { QRGB(122,103,238), "slateblue2" },
655 { QRGB(105, 89,205), "slateblue3" },
656 { QRGB( 71, 60,139), "slateblue4" },
657 { QRGB(112,128,144), "slategray" },
658 { QRGB(198,226,255), "slategray1" },
659 { QRGB(185,211,238), "slategray2" },
660 { QRGB(159,182,205), "slategray3" },
661 { QRGB(108,123,139), "slategray4" },
662 { QRGB(112,128,144), "slategrey" },
663 { QRGB(255,250,250), "snow" },
664 { QRGB(255,250,250), "snow1" },
665 { QRGB(238,233,233), "snow2" },
666 { QRGB(205,201,201), "snow3" },
667 { QRGB(139,137,137), "snow4" },
668 { QRGB( 0,255,127), "springgreen" },
669 { QRGB( 0,255,127), "springgreen1" },
670 { QRGB( 0,238,118), "springgreen2" },
671 { QRGB( 0,205,102), "springgreen3" },
672 { QRGB( 0,139, 69), "springgreen4" },
673 { QRGB( 70,130,180), "steelblue" },
674 { QRGB( 99,184,255), "steelblue1" },
675 { QRGB( 92,172,238), "steelblue2" },
676 { QRGB( 79,148,205), "steelblue3" },
677 { QRGB( 54,100,139), "steelblue4" },
678 { QRGB(210,180,140), "tan" },
679 { QRGB(255,165, 79), "tan1" },
680 { QRGB(238,154, 73), "tan2" },
681 { QRGB(205,133, 63), "tan3" },
682 { QRGB(139, 90, 43), "tan4" },
683 { QRGB(216,191,216), "thistle" },
684 { QRGB(255,225,255), "thistle1" },
685 { QRGB(238,210,238), "thistle2" },
686 { QRGB(205,181,205), "thistle3" },
687 { QRGB(139,123,139), "thistle4" },
688 { QRGB(255, 99, 71), "tomato" },
689 { QRGB(255, 99, 71), "tomato1" },
690 { QRGB(238, 92, 66), "tomato2" },
691 { QRGB(205, 79, 57), "tomato3" },
692 { QRGB(139, 54, 38), "tomato4" },
693 { QRGB( 64,224,208), "turquoise" },
694 { QRGB( 0,245,255), "turquoise1" },
695 { QRGB( 0,229,238), "turquoise2" },
696 { QRGB( 0,197,205), "turquoise3" },
697 { QRGB( 0,134,139), "turquoise4" },
698 { QRGB(238,130,238), "violet" },
699 { QRGB(208, 32,144), "violetred" },
700 { QRGB(255, 62,150), "violetred1" },
701 { QRGB(238, 58,140), "violetred2" },
702 { QRGB(205, 50,120), "violetred3" },
703 { QRGB(139, 34, 82), "violetred4" },
704 { QRGB(245,222,179), "wheat" },
705 { QRGB(255,231,186), "wheat1" },
706 { QRGB(238,216,174), "wheat2" },
707 { QRGB(205,186,150), "wheat3" },
708 { QRGB(139,126,102), "wheat4" },
709 { QRGB(255,255,255), "white" },
710 { QRGB(245,245,245), "whitesmoke" },
711 { QRGB(255,255, 0), "yellow" },
712 { QRGB(255,255, 0), "yellow1" },
713 { QRGB(238,238, 0), "yellow2" },
714 { QRGB(205,205, 0), "yellow3" },
715 { QRGB(139,139, 0), "yellow4" },
716 { QRGB(154,205, 50), "yellowgreen" } };
717
718
719inline bool operator<(const char *name, const XPMRGBData &data)
720{ return qstrcmp(name, data.name) < 0; }
721inline bool operator<(const XPMRGBData &data, const char *name)
722{ return qstrcmp(data.name, name) < 0; }
723
724static inline std::optional<QRgb> qt_get_named_xpm_rgb(const char *name_no_space)
725{
726 const XPMRGBData *r = std::lower_bound(xpmRgbTbl, xpmRgbTbl + xpmRgbTblSize, name_no_space);
727 if ((r != xpmRgbTbl + xpmRgbTblSize) && !(name_no_space < *r))
728 return r->value;
729 return {};
730}
731
732/*****************************************************************************
733 Misc. utility functions
734 *****************************************************************************/
735static QString fbname(const QString &fileName) // get file basename (sort of)
736{
738 if (!s.isEmpty()) {
739 int i = qMax(s.lastIndexOf(u'/'), s.lastIndexOf(u'\\'));
740 if (i < 0)
741 i = 0;
742 auto checkChar = [](QChar ch) -> bool {
743 uchar uc = ch.unicode();
744 return isAsciiLetterOrNumber(uc) || uc == '_';
745 };
746 int start = -1;
747 for (; i < s.size(); ++i) {
748 if (checkChar(s.at(i))) {
749 start = i;
750 } else if (start > 0)
751 break;
752 }
753 if (start < 0)
754 s.clear();
755 else
756 s = s.mid(start, i - start);
757 }
758 if (s.isEmpty())
759 s = QString::fromLatin1("dummy");
760 return s;
761}
762
763// Skip until ", read until the next ", return the rest in *buf
764// Returns false on error, true on success
765
766static bool read_xpm_string(QByteArray &buf, QIODevice *d, const char * const *source, int &index,
768{
769 if (source) {
770 buf = source[index++];
771 return true;
772 }
773
774 buf = "";
775 bool gotQuote = false;
776 int offset = 0;
777 forever {
778 if (offset == state.size() || state.isEmpty()) {
779 char buf[2048];
780 qint64 bytesRead = d->read(buf, sizeof(buf));
781 if (bytesRead <= 0)
782 return false;
783 state = QByteArray(buf, int(bytesRead));
784 offset = 0;
785 }
786
787 if (!gotQuote) {
788 if (state.at(offset++) == '"')
789 gotQuote = true;
790 } else {
791 char c = state.at(offset++);
792 if (c == '"')
793 break;
794 buf += c;
795 }
796 }
797 state.remove(0, offset);
798 return true;
799}
800
801// Tests if the given prefix can be the start of an XPM color specification
802
803static bool is_xpm_color_spec_prefix(const QByteArray& prefix)
804{
805 return prefix == "c" ||
806 prefix == "g" ||
807 prefix == "g4" ||
808 prefix == "m" ||
809 prefix == "s";
810}
811
812// Reads XPM header.
813
814static bool read_xpm_header(
815 QIODevice *device, const char * const * source, int& index, QByteArray &state,
816 int *cpp, int *ncols, int *w, int *h)
817{
818 QByteArray buf(200, 0);
819
821 return false;
822
823#ifdef Q_CC_MSVC
824 if (sscanf_s(buf, "%d %d %d %d", w, h, ncols, cpp) < 4)
825#else
826 if (sscanf(buf, "%d %d %d %d", w, h, ncols, cpp) < 4)
827#endif
828 return false; // < 4 numbers parsed
829
830 if (*w <= 0 || *w > 32767 || *h <= 0 || *h > 32767 || *ncols <= 0 || *ncols > (64 * 64 * 64 * 64) || *cpp <= 0 || *cpp > 15)
831 return false; // failed sanity check
832
833 return true;
834}
835
836// Reads XPM body (color information & pixels).
837
838static bool read_xpm_body(
839 QIODevice *device, const char * const * source, int& index, QByteArray& state,
840 int cpp, int ncols, int w, int h, QImage& image)
841{
842 QByteArray buf(200, 0);
843 int i;
844
845 if (cpp < 0 || cpp > 15)
846 return false;
847
848 // For > 256 colors, we delay creation of the image until
849 // after we have read the color specifications, so that we can
850 // create it in correct format (Format_RGB32 vs Format_ARGB32,
851 // depending on absence or presence of "c none", respectively)
852 if (ncols <= 256) {
854 return false;
855 image.setColorCount(ncols);
856 }
857
858 QMap<quint64, int> colorMap;
859 int currentColor;
860 bool hasTransparency = false;
861
862 for(currentColor=0; currentColor < ncols; ++currentColor) {
864 qCWarning(lcImageIo, "XPM color specification missing");
865 return false;
866 }
868 index = buf.left(cpp);
870 QList<QByteArray> tokens = buf.split(' ');
871 i = tokens.indexOf("c");
872 if (i < 0)
873 i = tokens.indexOf("g");
874 if (i < 0)
875 i = tokens.indexOf("g4");
876 if (i < 0)
877 i = tokens.indexOf("m");
878 if (i < 0) {
879 qCWarning(lcImageIo, "XPM color specification is missing: %s", buf.constData());
880 return false; // no c/g/g4/m specification at all
881 }
883 while ((++i < tokens.size()) && !is_xpm_color_spec_prefix(tokens.at(i))) {
884 color.append(tokens.at(i));
885 }
886 if (color.isEmpty()) {
887 qCWarning(lcImageIo, "XPM color value is missing from specification: %s", buf.constData());
888 return false; // no color value
889 }
890 buf = color;
891 if (buf == "none") {
892 hasTransparency = true;
893 int transparentColor = currentColor;
894 if (ncols <= 256) {
895 image.setColor(transparentColor, 0);
896 colorMap.insert(xpmHash(QLatin1StringView(index.constData())), transparentColor);
897 } else {
898 colorMap.insert(xpmHash(QLatin1StringView(index.constData())), 0);
899 }
900 } else {
901 QRgb c_rgb = 0;
902 if (((buf.size()-1) % 3) && (buf[0] == '#')) {
903 buf.truncate(((buf.size()-1) / 4 * 3) + 1); // remove alpha channel left by imagemagick
904 }
905 if (buf[0] == '#') {
906 c_rgb = qt_get_hex_rgb(buf).value_or(0);
907 } else {
908 c_rgb = qt_get_named_xpm_rgb(buf).value_or(0);
909 }
910 if (ncols <= 256) {
911 image.setColor(currentColor, 0xff000000 | c_rgb);
912 colorMap.insert(xpmHash(QLatin1StringView(index.constData())), currentColor);
913 } else {
914 colorMap.insert(xpmHash(QLatin1StringView(index.constData())), 0xff000000 | c_rgb);
915 }
916 }
917 }
918
919 if (ncols > 256) {
920 // Now we can create 32-bit image of appropriate format
921 QImage::Format format = hasTransparency ?
924 return false;
925 }
926
927 // Read pixels
928 for(int y=0; y<h; y++) {
930 qCWarning(lcImageIo, "XPM pixels missing on image line %d", y);
931 return false;
932 }
933 if (image.depth() == 8) {
934 uchar *p = image.scanLine(y);
935 uchar *d = (uchar *)buf.data();
936 uchar *end = d + buf.size();
937 int x;
938 if (cpp == 1) {
939 char b[2];
940 b[1] = '\0';
941 for (x=0; x<w && d<end; x++) {
942 b[0] = *d++;
943 *p++ = (uchar)colorMap[xpmHash(b)];
944 }
945 } else {
946 char b[16];
947 b[cpp] = '\0';
948 for (x = 0; x < w && d + cpp <= end; x++) {
949 memcpy(b, (char *)d, cpp);
950 *p++ = (uchar)colorMap[xpmHash(b)];
951 d += cpp;
952 }
953 }
954 // avoid uninitialized memory for malformed xpms
955 if (x < w) {
956 qCWarning(lcImageIo, "XPM pixels missing on image line %d (possibly a C++ trigraph).", y);
957 memset(p, 0, w - x);
958 }
959 } else {
960 QRgb *p = (QRgb*)image.scanLine(y);
961 uchar *d = (uchar *)buf.data();
962 uchar *end = d + buf.size();
963 int x;
964 char b[16];
965 b[cpp] = '\0';
966 for (x = 0; x < w && d + cpp <= end; x++) {
967 memcpy(b, (char *)d, cpp);
968 *p++ = (QRgb)colorMap[xpmHash(b)];
969 d += cpp;
970 }
971 // avoid uninitialized memory for malformed xpms
972 if (x < w) {
973 qCWarning(lcImageIo, "XPM pixels missing on image line %d (possibly a C++ trigraph).", y);
974 memset(p, 0, (w - x)*4);
975 }
976 }
977 }
978
979 if (device) {
980 // Rewind unused characters, and skip to the end of the XPM struct.
981 for (int i = state.size() - 1; i >= 0; --i)
982 device->ungetChar(state[i]);
983 char c;
984 while (device->getChar(&c) && c != ';') {}
985 while (device->getChar(&c) && c != '\n') {}
986 }
987 return true;
988}
989
990//
991// INTERNAL
992//
993// Reads an .xpm from either the QImageIO or from the QString *.
994// One of the two HAS to be 0, the other one is used.
995//
996
998{
999 if (!source)
1000 return true;
1001
1002 QByteArray buf(200, 0);
1004
1005 int cpp, ncols, w, h, index = 0;
1006
1007 if (device) {
1008 // "/* XPM */"
1009 int readBytes;
1010 if ((readBytes = device->readLine(buf.data(), buf.size())) < 0)
1011 return false;
1012
1013 static constexpr auto matcher = qMakeStaticByteArrayMatcher("/* XPM");
1014
1015 if (matcher.indexIn(buf) != 0) {
1016 while (readBytes > 0) {
1017 device->ungetChar(buf.at(readBytes - 1));
1018 --readBytes;
1019 }
1020 return false;
1021 }// bad magic
1022 }
1023
1024 if (!read_xpm_header(device, source, index, state, &cpp, &ncols, &w, &h))
1025 return false;
1026
1027 return read_xpm_body(device, source, index, state, cpp, ncols, w, h, image);
1028}
1029
1030namespace {
1031template <size_t N>
1032struct CharBuffer : std::array<char, N>
1033{
1034 CharBuffer() {} // avoid value-initializing the whole array
1035};
1036}
1037
1038static const char* xpm_color_name(int cpp, int index, CharBuffer<5> && returnable = {})
1039{
1040 static const char code[] = ".#abcdefghijklmnopqrstuvwxyzABCD"
1041 "EFGHIJKLMNOPQRSTUVWXYZ0123456789";
1042 // cpp is limited to 4 and index is limited to 64^cpp
1043 if (cpp > 1) {
1044 if (cpp > 2) {
1045 if (cpp > 3) {
1046 returnable[4] = '\0';
1047 returnable[3] = code[index % 64];
1048 index /= 64;
1049 } else
1050 returnable[3] = '\0';
1051 returnable[2] = code[index % 64];
1052 index /= 64;
1053 } else
1054 returnable[2] = '\0';
1055 // the following 4 lines are a joke!
1056 if (index == 0)
1057 index = 64*44+21;
1058 else if (index == 64*44+21)
1059 index = 0;
1060 returnable[1] = code[index % 64];
1061 index /= 64;
1062 } else
1063 returnable[1] = '\0';
1064 returnable[0] = code[index];
1065
1066 return returnable.data();
1067}
1068
1069
1070// write XPM image data
1071static bool write_xpm_image(const QImage &sourceImage, QIODevice *device, const QString &fileName)
1072{
1073 if (!device->isWritable())
1074 return false;
1075
1076 QImage image;
1077 if (sourceImage.format() != QImage::Format_RGB32 && sourceImage.format() != QImage::Format_ARGB32 && sourceImage.format() != QImage::Format_ARGB32_Premultiplied)
1078 image = sourceImage.convertToFormat(QImage::Format_RGB32);
1079 else
1080 image = sourceImage;
1081
1082#ifdef __cpp_lib_memory_resource
1083 char buffer[1024];
1084 std::pmr::monotonic_buffer_resource res{&buffer, sizeof buffer};
1085 std::pmr::map<QRgb, int> colorMap(&res);
1086#else
1087 std::map<QRgb, int> colorMap;
1088#endif
1089
1090 const int w = image.width();
1091 const int h = image.height();
1092 int ncolors = 0;
1093
1094 // build color table
1095 for (int y = 0; y < h; ++y) {
1096 const QRgb *yp = reinterpret_cast<const QRgb *>(image.constScanLine(y));
1097 for (int x = 0; x < w; ++x) {
1098 const auto [it, inserted] = colorMap.try_emplace(yp[x], ncolors);
1099 if (inserted)
1100 ++ncolors;
1101 }
1102 }
1103
1104 // number of 64-bit characters per pixel needed to encode all colors
1105 int cpp = 1;
1106 for (int k = 64; ncolors > k; k *= 64) {
1107 ++cpp;
1108 // limit to 4 characters per pixel
1109 // 64^4 colors is enough for a 4096x4096 image
1110 if (cpp > 4) {
1111 qCWarning(lcImageIo, "Qt does not support writing XPM images with more than "
1112 "64^4 colors (requested: %d colors).", ncolors);
1113 return false;
1114 }
1115 }
1116
1117 // write header
1119 s << "/* XPM */" << Qt::endl
1120 << "static char *" << fbname(fileName) << "[]={" << Qt::endl
1121 << '\"' << w << ' ' << h << ' ' << ncolors << ' ' << cpp << '\"';
1122
1123 // write palette
1124 for (const auto &[color, index] : colorMap) {
1125 const QString line = image.format() != QImage::Format_RGB32 && !qAlpha(color)
1126 ? QString::asprintf("\"%s c None\"", xpm_color_name(cpp, index))
1127 : QString::asprintf("\"%s c #%02x%02x%02x\"", xpm_color_name(cpp, index),
1129 s << ',' << Qt::endl << line;
1130 }
1131
1132 // write pixels, limit to 4 characters per pixel
1133 for (int y = 0; y < h; ++y) {
1134 s << ',' << Qt::endl << '\"';
1135 const QRgb *yp = reinterpret_cast<const QRgb *>(image.constScanLine(y));
1136 for (int x = 0; x < w; ++x)
1137 s << xpm_color_name(cpp, colorMap[yp[x]]);
1138 s << '\"';
1139 }
1140 s << "};" << Qt::endl;
1141 return (s.status() == QTextStream::Ok);
1142}
1143
1145 : state(Ready), index(0)
1146{
1147}
1148
1149bool QXpmHandler::readHeader()
1150{
1151 state = Error;
1152 if (!read_xpm_header(device(), nullptr, index, buffer, &cpp, &ncols, &width, &height))
1153 return false;
1154 state = ReadHeader;
1155 return true;
1156}
1157
1158bool QXpmHandler::readImage(QImage *image)
1159{
1160 if (state == Error)
1161 return false;
1162
1163 if (state == Ready && !readHeader()) {
1164 state = Error;
1165 return false;
1166 }
1167
1168 if (!read_xpm_body(device(), nullptr, index, buffer, cpp, ncols, width, height, *image)) {
1169 state = Error;
1170 return false;
1171 }
1172
1173 state = Ready;
1174 return true;
1175}
1176
1178{
1179 if (state == Ready && !canRead(device()))
1180 return false;
1181
1182 if (state != Error) {
1183 setFormat("xpm");
1184 return true;
1185 }
1186
1187 return false;
1188}
1189
1191{
1192 if (!device) {
1193 qCWarning(lcImageIo, "QXpmHandler::canRead() called with no device");
1194 return false;
1195 }
1196
1197 char head[6];
1198 if (device->peek(head, sizeof(head)) != sizeof(head))
1199 return false;
1200
1201 return qstrncmp(head, "/* XPM", 6) == 0;
1202}
1203
1205{
1206 if (!canRead())
1207 return false;
1208 return readImage(image);
1209}
1210
1212{
1213 return write_xpm_image(image, device(), fileName);
1214}
1215
1217{
1218 return option == Name
1219 || option == Size
1220 || option == ImageFormat;
1221}
1222
1224{
1225 if (option == Name) {
1226 return fileName;
1227 } else if (option == Size) {
1228 if (state == Error)
1229 return QVariant();
1230 if (state == Ready && !const_cast<QXpmHandler*>(this)->readHeader())
1231 return QVariant();
1232 return QSize(width, height);
1233 } else if (option == ImageFormat) {
1234 if (state == Error)
1235 return QVariant();
1236 if (state == Ready && !const_cast<QXpmHandler*>(this)->readHeader())
1237 return QVariant();
1238 // If we have more than 256 colors in the table, we need to
1239 // figure out, if it contains transparency. That means reading
1240 // the whole color table, which is too much work work pre-checking
1241 // the image format
1242 if (ncols <= 256)
1244 else
1246 }
1247
1248 return QVariant();
1249}
1250
1252{
1253 if (option == Name)
1254 fileName = value.toString();
1255}
1256
1258
1259#endif // QT_NO_IMAGEFORMAT_XPM
IOBluetoothDevice * device
\inmodule QtCore
Definition qbytearray.h:57
QByteArray trimmed() const &
Definition qbytearray.h:262
QByteArray simplified() const &
Definition qbytearray.h:266
QByteArray left(qsizetype n) const &
Definition qbytearray.h:169
QList< QByteArray > split(char sep) const
Splits the byte array into subarrays wherever sep occurs, and returns the list of those arrays.
QByteArray toLower() const &
Definition qbytearray.h:254
QByteArray & append(char c)
This is an overloaded member function, provided for convenience. It differs from the above function o...
QByteArray mid(qsizetype index, qsizetype len=-1) const &
\inmodule QtCore
\inmodule QtCore \reentrant
Definition qiodevice.h:34
qint64 peek(char *data, qint64 maxlen)
ImageOption
This enum describes the different options supported by QImageIOHandler.
static bool allocateImage(QSize size, QImage::Format format, QImage *image)
QIODevice * device() const
Returns the device currently assigned to the QImageIOHandler.
void setFormat(const QByteArray &format)
Sets the format of the QImageIOHandler to format.
\inmodule QtGui
Definition qimage.h:37
Format
The following image formats are available in Qt.
Definition qimage.h:41
@ Format_RGB32
Definition qimage.h:46
@ Format_Invalid
Definition qimage.h:42
@ Format_Indexed8
Definition qimage.h:45
@ Format_ARGB32_Premultiplied
Definition qimage.h:48
@ Format_ARGB32
Definition qimage.h:47
\inmodule QtCore
Definition qsize.h:25
\macro QT_RESTRICTED_CAST_FROM_ASCII
Definition qstring.h:129
static QString fromLatin1(QByteArrayView ba)
This is an overloaded member function, provided for convenience. It differs from the above function o...
Definition qstring.cpp:5871
qsizetype size() const noexcept
Returns the number of characters in this string.
Definition qstring.h:186
const QChar at(qsizetype i) const
Returns the character at the given index position in the string.
Definition qstring.h:1226
QChar * data()
Returns a pointer to the data stored in the QString.
Definition qstring.h:1240
static QString static QString asprintf(const char *format,...) Q_ATTRIBUTE_FORMAT_PRINTF(1
Definition qstring.cpp:7263
\inmodule QtCore
\inmodule QtCore
Definition qvariant.h:65
bool read(QImage *image) override
Read an image from the device, and stores it in image.
QVariant option(ImageOption option) const override
Returns the value assigned to option as a QVariant.
bool supportsOption(ImageOption option) const override
Returns true if the QImageIOHandler supports the option option; otherwise returns false.
bool canRead() const override
Returns true if an image can be read from the device (i.e., the image format is supported,...
void setOption(ImageOption option, const QVariant &value) override
Sets the option option with the value value.
bool write(const QImage &image) override
Writes the image image to the assigned device.
QString str
[2]
QSet< QString >::iterator it
else opt state
[0]
std::list< QString >::iterator Name
Definition lalr.h:28
Combined button and popup list for selecting options.
constexpr bool isAsciiLetterOrNumber(char32_t c) noexcept
Definition qtools_p.h:82
QTextStream & endl(QTextStream &stream)
Writes '\n' to the stream and flushes the stream.
Definition image.cpp:4
int qstrncmp(const char *str1, const char *str2, size_t len)
Q_CORE_EXPORT int qstrcmp(const char *str1, const char *str2)
constexpr QStaticByteArrayMatcher< N > qMakeStaticByteArrayMatcher(const char(&pattern)[N]) noexcept
std::optional< QRgb > qt_get_hex_rgb(const char *name)
Definition qcolor.cpp:90
typedef QByteArray(EGLAPIENTRYP PFNQGSGETDISPLAYSPROC)()
EGLOutputLayerEXT EGLint EGLAttrib value
[5]
#define forever
Definition qforeach.h:78
#define qCWarning(category,...)
#define Q_DECLARE_LOGGING_CATEGORY(name)
constexpr const T & qMax(const T &a, const T &b)
Definition qminmax.h:42
GLboolean GLboolean GLboolean b
GLint GLint GLint GLint GLint x
[0]
GLfloat GLfloat GLfloat w
[0]
GLint GLsizei GLsizei height
GLuint index
[2]
GLboolean r
[2]
GLuint GLuint end
GLint GLsizei GLsizei GLenum GLenum GLsizei void * data
GLenum GLuint buffer
GLint GLsizei width
GLuint color
[2]
GLenum GLuint GLenum GLsizei const GLchar * buf
GLuint start
GLenum GLuint GLintptr offset
GLuint name
GLint GLsizei GLsizei GLenum format
GLint y
GLfloat GLfloat GLfloat GLfloat h
GLsizei GLsizei GLchar * source
GLdouble s
[6]
Definition qopenglext.h:235
GLuint res
const GLubyte * c
GLfloat GLfloat p
[1]
GLuint GLenum option
QT_BEGIN_NAMESPACE typedef unsigned int QRgb
Definition qrgb.h:13
constexpr int qRed(QRgb rgb)
Definition qrgb.h:18
constexpr int qGreen(QRgb rgb)
Definition qrgb.h:21
constexpr int qBlue(QRgb rgb)
Definition qrgb.h:24
constexpr int qAlpha(QRgb rgb)
Definition qrgb.h:27
unsigned char uchar
Definition qtypes.h:32
unsigned long long quint64
Definition qtypes.h:61
unsigned int uint
Definition qtypes.h:34
long long qint64
Definition qtypes.h:60
static const struct XPMRGBData xpmRgbTbl[]
bool operator<(const char *name, const XPMRGBData &data)
static bool write_xpm_image(const QImage &sourceImage, QIODevice *device, const QString &fileName)
static bool is_xpm_color_spec_prefix(const QByteArray &prefix)
static std::optional< QRgb > qt_get_named_xpm_rgb(const char *name_no_space)
static bool read_xpm_string(QByteArray &buf, QIODevice *d, const char *const *source, int &index, QByteArray &state)
static const char * xpm_color_name(int cpp, int index, CharBuffer< 5 > &&returnable={})
bool qt_read_xpm_image_or_array(QIODevice *device, const char *const *source, QImage &image)
static bool read_xpm_header(QIODevice *device, const char *const *source, int &index, QByteArray &state, int *cpp, int *ncols, int *w, int *h)
static QString fbname(const QString &fileName)
static quint64 xpmHash(const QString &str)
static bool read_xpm_body(QIODevice *device, const char *const *source, int &index, QByteArray &state, int cpp, int ncols, int w, int h, QImage &image)
static const int xpmRgbTblSize
#define QRGB(r, g, b)
ASSERT failure in file div cpp
static const auto matcher
[0]
manager head(request, this, [this](QRestReply &reply) { if(reply.isSuccess()) })
[6]