Skip to content

Commit 4f151c7

Browse files
committed
Fix Tiger app icon
png2icns produces a 128x128 icon that isn't compatible with OSX Tiger. Uses a patched version of png2icns to produce the correct icon. Here is the output of icns2png -l AppIcon_128.icns. Before: ``` Extracting icons from AppIcon_128.icns... Icon family size is 22560 bytes (including 8 byte header) Listing icon elements... 'is32' 16x16 32-bit icon (1024 bytes compressed to 710) 's8mk' 16x16 8-bit mask (256 bytes) 'il32' 32x32 32-bit icon (4096 bytes compressed to 2448) 'l8mk' 32x32 8-bit mask (1024 bytes) 'ih32' 48x48 32-bit icon (9216 bytes compressed to 4418) 'h8mk' 48x48 8-bit mask (2304 bytes) 'ic07' 128x128 32-bit icon (65536 bytes compressed to 11336) ``` After: ``` Extracting icons from AppIcon_128.icns... Icon family size is 42420 bytes (including 8 byte header) Listing icon elements... 'is32' 16x16 32-bit icon (1024 bytes compressed to 710) 's8mk' 16x16 8-bit mask (256 bytes) 'il32' 32x32 32-bit icon (4096 bytes compressed to 2300) 'l8mk' 32x32 8-bit mask (1024 bytes) 'it32' 128x128 32-bit icon (65536 bytes compressed to 21690) 't8mk' 128x128 8-bit mask (16384 bytes) ```
1 parent 08b4a81 commit 4f151c7

File tree

5 files changed

+422
-0
lines changed

5 files changed

+422
-0
lines changed

Packaging/apple/AppIcon_128.icns

19.4 KB
Binary file not shown.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
/build*/
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
# On Debian/Ubuntu, this requires libpng-dev and libicns-dev
2+
cmake_minimum_required(VERSION 3.22)
3+
project(png2icns_tiger LANGUAGES C)
4+
add_executable(png2icns_tiger png2icns_tiger.c)
5+
find_package(PkgConfig REQUIRED)
6+
pkg_check_modules(PNG REQUIRED IMPORTED_TARGET libpng)
7+
pkg_check_modules(ICNS REQUIRED IMPORTED_TARGET libicns)
8+
target_link_libraries(png2icns_tiger PRIVATE PkgConfig::PNG PkgConfig::ICNS)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,354 @@
1+
/*
2+
* png2icns_tiger - based on png2icns from the pngutils Debian package but
3+
* modified for Tiger support. See the section that says MODIFICATION.
4+
*
5+
* Copyright (C) 2008 Julien BLACHE <[email protected]>
6+
* Copyright (C) 2012 Mathew Eis <[email protected]>
7+
*
8+
* This program is free software; you can redistribute it and/or modify
9+
* it under the terms of the GNU General Public License as published by
10+
* the Free Software Foundation; either version 2 of the License, or
11+
* (at your option) any later version.
12+
*
13+
* This program is distributed in the hope that it will be useful,
14+
* but WITHOUT ANY WARRANTY; without even the implied warranty of
15+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16+
* GNU General Public License for more details.
17+
*
18+
* You should have received a copy of the GNU General Public License along
19+
* with this program; if not, write to the Free Software Foundation, Inc.,
20+
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
21+
*/
22+
#include <stdint.h>
23+
#include <stdio.h>
24+
#include <stdlib.h>
25+
#include <string.h>
26+
#include <unistd.h>
27+
28+
#include <errno.h>
29+
30+
#include <icns.h>
31+
#include <png.h>
32+
#include <setjmp.h>
33+
34+
#define FALSE 0
35+
#define TRUE 1
36+
37+
#if PNG_LIBPNG_VER >= 10209
38+
#define PNG2ICNS_EXPAND_GRAY 1
39+
#endif
40+
41+
static int read_png(FILE *fp, png_bytepp buffer, int32_t *bpp, int32_t *width,
42+
int32_t *height) {
43+
png_structp png_ptr;
44+
png_infop info;
45+
png_uint_32 w;
46+
png_uint_32 h;
47+
png_bytep *rows;
48+
49+
int bit_depth;
50+
int32_t color_type;
51+
52+
int row;
53+
int rowsize;
54+
55+
png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
56+
if (png_ptr == NULL)
57+
return FALSE;
58+
59+
info = png_create_info_struct(png_ptr);
60+
if (info == NULL) {
61+
png_destroy_read_struct(&png_ptr, NULL, NULL);
62+
return FALSE;
63+
}
64+
65+
if (setjmp(png_jmpbuf(png_ptr))) {
66+
png_destroy_read_struct(&png_ptr, &info, NULL);
67+
return FALSE;
68+
}
69+
70+
png_init_io(png_ptr, fp);
71+
72+
png_read_info(png_ptr, info);
73+
png_get_IHDR(png_ptr, info, &w, &h, &bit_depth, &color_type, NULL, NULL,
74+
NULL);
75+
76+
switch (color_type) {
77+
case PNG_COLOR_TYPE_GRAY:
78+
#ifdef PNG2ICNS_EXPAND_GRAY
79+
png_set_expand_gray_1_2_4_to_8(png_ptr);
80+
#else
81+
png_set_gray_1_2_4_to_8(png_ptr);
82+
#endif
83+
84+
if (bit_depth == 16) {
85+
png_set_strip_16(png_ptr);
86+
bit_depth = 8;
87+
}
88+
89+
png_set_gray_to_rgb(png_ptr);
90+
png_set_add_alpha(png_ptr, 0xff, PNG_FILLER_AFTER);
91+
break;
92+
93+
case PNG_COLOR_TYPE_GRAY_ALPHA:
94+
#ifdef PNG2ICNS_EXPAND_GRAY
95+
png_set_expand_gray_1_2_4_to_8(png_ptr);
96+
#else
97+
png_set_gray_1_2_4_to_8(png_ptr);
98+
#endif
99+
100+
if (bit_depth == 16) {
101+
png_set_strip_16(png_ptr);
102+
bit_depth = 8;
103+
}
104+
105+
png_set_gray_to_rgb(png_ptr);
106+
break;
107+
108+
case PNG_COLOR_TYPE_PALETTE:
109+
png_set_palette_to_rgb(png_ptr);
110+
111+
if (png_get_valid(png_ptr, info, PNG_INFO_tRNS))
112+
png_set_tRNS_to_alpha(png_ptr);
113+
else
114+
png_set_add_alpha(png_ptr, 0xff, PNG_FILLER_AFTER);
115+
break;
116+
117+
case PNG_COLOR_TYPE_RGB:
118+
if (bit_depth == 16) {
119+
png_set_strip_16(png_ptr);
120+
bit_depth = 8;
121+
}
122+
123+
png_set_add_alpha(png_ptr, 0xff, PNG_FILLER_AFTER);
124+
break;
125+
126+
case PNG_COLOR_TYPE_RGB_ALPHA:
127+
if (bit_depth == 16) {
128+
png_set_strip_16(png_ptr);
129+
bit_depth = 8;
130+
}
131+
132+
break;
133+
}
134+
135+
*width = w;
136+
*height = h;
137+
*bpp = bit_depth * 4;
138+
139+
png_set_interlace_handling(png_ptr);
140+
141+
png_read_update_info(png_ptr, info);
142+
143+
rowsize = png_get_rowbytes(png_ptr, info);
144+
rows = malloc(sizeof(png_bytep) * h);
145+
*buffer = malloc(rowsize * h + 8);
146+
147+
rows[0] = *buffer;
148+
for (row = 1; row < h; row++) {
149+
rows[row] = rows[row - 1] + rowsize;
150+
}
151+
152+
png_read_image(png_ptr, rows);
153+
png_destroy_read_struct(&png_ptr, &info, NULL);
154+
155+
free(rows);
156+
157+
return TRUE;
158+
}
159+
160+
static int add_png_to_family(icns_family_t **iconFamily, char *pngname) {
161+
FILE *pngfile;
162+
163+
int icnsErr = ICNS_STATUS_OK;
164+
icns_image_t icnsImage;
165+
icns_image_t icnsMask;
166+
icns_type_t iconType;
167+
icns_type_t maskType;
168+
icns_icon_info_t iconInfo;
169+
170+
icns_element_t *iconElement = NULL;
171+
icns_element_t *maskElement = NULL;
172+
char iconStr[5] = {0, 0, 0, 0, 0};
173+
char maskStr[5] = {0, 0, 0, 0, 0};
174+
int iconDataOffset = 0;
175+
int maskDataOffset = 0;
176+
177+
png_bytep buffer;
178+
int width, height, bpp;
179+
180+
pngfile = fopen(pngname, "rb");
181+
if (pngfile == NULL) {
182+
fprintf(stderr, "Could not open '%s' for reading: %s\n", pngname,
183+
strerror(errno));
184+
return FALSE;
185+
}
186+
187+
if (!read_png(pngfile, &buffer, &bpp, &width, &height)) {
188+
fprintf(stderr, "Failed to read PNG file\n");
189+
fclose(pngfile);
190+
191+
return FALSE;
192+
}
193+
194+
fclose(pngfile);
195+
196+
icnsImage.imageWidth = width;
197+
icnsImage.imageHeight = height;
198+
icnsImage.imageChannels = 4;
199+
icnsImage.imagePixelDepth = 8;
200+
icnsImage.imageDataSize = width * height * 4;
201+
icnsImage.imageData = buffer;
202+
203+
iconInfo.isImage = 1;
204+
iconInfo.iconWidth = icnsImage.imageWidth;
205+
iconInfo.iconHeight = icnsImage.imageHeight;
206+
iconInfo.iconBitDepth = bpp;
207+
iconInfo.iconChannels = (bpp == 32 ? 4 : 1);
208+
iconInfo.iconPixelDepth = bpp / iconInfo.iconChannels;
209+
210+
iconType = icns_get_type_from_image_info(iconInfo);
211+
maskType = icns_get_mask_type_for_icon_type(iconType);
212+
213+
/* MODIFICATION */
214+
if (iconType == ICNS_128x128_32BIT_ARGB_DATA) {
215+
/* libicns returns "ic07" for 128x128 icons but that doesn't work on Tiger
216+
*/
217+
iconType = ICNS_128X128_32BIT_DATA;
218+
maskType = ICNS_128X128_8BIT_MASK;
219+
}
220+
/* END OF MODIFICATION */
221+
222+
icns_type_str(iconType, iconStr);
223+
icns_type_str(maskType, maskStr);
224+
225+
/* Only convert the icons that match sizes icns supports */
226+
if (iconType == ICNS_NULL_TYPE) {
227+
fprintf(stderr, "Bad dimensions: PNG file '%s' is %dx%d\n", pngname, width,
228+
height);
229+
free(buffer);
230+
231+
return FALSE;
232+
}
233+
234+
if (bpp != 32) {
235+
fprintf(stderr, "Bit depth %d unsupported in '%s'\n", bpp, pngname);
236+
free(buffer);
237+
238+
return FALSE;
239+
}
240+
241+
icns_set_print_errors(0);
242+
if (icns_get_element_from_family(*iconFamily, iconType, &iconElement) ==
243+
ICNS_STATUS_OK) {
244+
icns_set_print_errors(1);
245+
246+
fprintf(stderr, "Duplicate icon element of type '%s' detected (%s)\n",
247+
iconStr, pngname);
248+
free(buffer);
249+
250+
return FALSE;
251+
}
252+
253+
icns_set_print_errors(1);
254+
255+
if ((iconType != ICNS_1024x1024_32BIT_ARGB_DATA) &&
256+
(iconType != ICNS_512x512_32BIT_ARGB_DATA) &&
257+
(iconType != ICNS_256x256_32BIT_ARGB_DATA) &&
258+
(iconType != ICNS_128x128_32BIT_ARGB_DATA)) {
259+
printf("Using icns type '%s', mask '%s' for '%s'\n", iconStr, maskStr,
260+
pngname);
261+
} else {
262+
printf("Using icns type '%s' (ARGB) for '%s'\n", iconStr, pngname);
263+
}
264+
265+
icnsErr = icns_new_element_from_image(&icnsImage, iconType, &iconElement);
266+
267+
if (iconElement != NULL) {
268+
if (icnsErr == ICNS_STATUS_OK) {
269+
icns_set_element_in_family(iconFamily, iconElement);
270+
}
271+
free(iconElement);
272+
}
273+
274+
if ((iconType != ICNS_1024x1024_32BIT_ARGB_DATA) &&
275+
(iconType != ICNS_512x512_32BIT_ARGB_DATA) &&
276+
(iconType != ICNS_256x256_32BIT_ARGB_DATA) &&
277+
(iconType != ICNS_128x128_32BIT_ARGB_DATA)) {
278+
icns_init_image_for_type(maskType, &icnsMask);
279+
280+
iconDataOffset = 0;
281+
maskDataOffset = 0;
282+
283+
while ((iconDataOffset < icnsImage.imageDataSize) &&
284+
(maskDataOffset < icnsMask.imageDataSize)) {
285+
icnsMask.imageData[maskDataOffset] =
286+
icnsImage.imageData[iconDataOffset + 3];
287+
iconDataOffset += 4; /* move to the next alpha byte */
288+
maskDataOffset += 1; /* move to the next byte */
289+
}
290+
291+
icnsErr = icns_new_element_from_mask(&icnsMask, maskType, &maskElement);
292+
293+
if (maskElement != NULL) {
294+
if (icnsErr == ICNS_STATUS_OK) {
295+
icns_set_element_in_family(iconFamily, maskElement);
296+
}
297+
free(maskElement);
298+
}
299+
300+
icns_free_image(&icnsMask);
301+
}
302+
303+
free(buffer);
304+
305+
return TRUE;
306+
}
307+
308+
int main(int argc, char **argv) {
309+
FILE *icnsfile;
310+
311+
icns_family_t *iconFamily;
312+
313+
int i;
314+
315+
if (argc < 3) {
316+
printf("Usage: png2icns file.icns file1.png file2.png ... filen.png\n");
317+
exit(1);
318+
}
319+
320+
icnsfile = fopen(argv[1], "wb+");
321+
if (icnsfile == NULL) {
322+
fprintf(stderr, "Could not open '%s' for writing: %s\n", argv[1],
323+
strerror(errno));
324+
exit(1);
325+
}
326+
327+
icns_set_print_errors(1);
328+
icns_create_family(&iconFamily);
329+
330+
for (i = 2; i < argc; i++) {
331+
if (!add_png_to_family(&iconFamily, argv[i])) {
332+
fclose(icnsfile);
333+
unlink(argv[1]);
334+
335+
exit(1);
336+
}
337+
}
338+
339+
if (icns_write_family_to_file(icnsfile, iconFamily) != ICNS_STATUS_OK) {
340+
fprintf(stderr, "Failed to write icns file\n");
341+
fclose(icnsfile);
342+
343+
exit(1);
344+
}
345+
346+
fclose(icnsfile);
347+
348+
printf("Saved icns file to %s\n", argv[1]);
349+
350+
if (iconFamily != NULL)
351+
free(iconFamily);
352+
353+
return 0;
354+
}

0 commit comments

Comments
 (0)