1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
|
#!/usr/bin/env python
# Copyright 2015 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Windows ICO file crusher.
Optimizes the PNG images within a Windows ICO icon file. This extracts all of
the sub-images within the file, runs any PNG-formatted images through
optimize-png-files.sh, then packs them back into an ICO file.
NOTE: ICO files can contain both raw uncompressed BMP files and PNG files. This
script does not touch the BMP files, which means if you have a huge uncompressed
image, it will not get smaller. 256x256 icons should be PNG-formatted first.
(Smaller icons should be BMPs for compatibility with Windows XP.)
"""
import argparse
import logging
import os
import StringIO
import sys
import ico_tools
def main(args=None):
if args is None:
args = sys.argv[1:]
parser = argparse.ArgumentParser(description='Crush Windows ICO files.')
parser.add_argument('files', metavar='ICO', type=argparse.FileType('r+b'),
nargs='+', help='.ico files to be crushed')
parser.add_argument('-o', dest='optimization_level', metavar='OPT', type=int,
help='optimization level')
parser.add_argument('-d', '--debug', dest='debug', action='store_true',
help='enable debug logging')
args = parser.parse_args()
if args.debug:
logging.getLogger().setLevel(logging.DEBUG)
for file in args.files:
buf = StringIO.StringIO()
file.seek(0, os.SEEK_END)
old_length = file.tell()
file.seek(0, os.SEEK_SET)
ico_tools.OptimizeIcoFile(file, buf, args.optimization_level)
new_length = len(buf.getvalue())
# Always write (even if file size not reduced), because we make other fixes
# such as regenerating the AND mask.
file.truncate(new_length)
file.seek(0)
file.write(buf.getvalue())
if new_length >= old_length:
logging.info('%s : Could not reduce file size.', file.name)
else:
saving = old_length - new_length
saving_percent = float(saving) / old_length
logging.info('%s : %d => %d (%d bytes : %d %%)', file.name, old_length,
new_length, saving, int(saving_percent * 100))
if __name__ == '__main__':
sys.exit(main())
|