forked from lzimann/IconDiffBot
-
Notifications
You must be signed in to change notification settings - Fork 1
/
icons.py
179 lines (159 loc) · 5.99 KB
/
icons.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
import re
import hashlib
from collections import OrderedDict
from PIL import ImageChops
import PIL.Image
import numpy as np
PARSE_REGEX = re.compile(r'\t(.*) = (.*)')
def int_or_float(value):
try:
value = int(value)
except ValueError:
value = float(value)
return value
def parse_metadata(img):
"""
Parses a DMI metadata,
returning an tuple array(icon_state, state dict).
img is a PIL.Image object
"""
img_dict = img.info
info = img_dict['Description'].split('\n')
if not 'version = 4.0' in info:
return None
meta_info = []
current_key = None
for entry in info:
if entry in ["# BEGIN DMI", "# END DMI", ""]:
continue
if '\t' not in entry:
current_key = entry.replace('state = ', '').replace('\"', '')
meta_info.append((current_key, {}))
else:
this_info = PARSE_REGEX.search(entry)
if this_info:
grp_1 = this_info.group(1)
grp_2 = this_info.group(2)
if grp_1 in ['delay', 'hotspot']:
entries = grp_2.split(',')
grp_2 = []
for thing in entries:
grp_2.append(int_or_float(thing))
else:
grp_2 = int_or_float(grp_2)
dict_to_add = {grp_1 : grp_2}
meta_info[len(meta_info) - 1][1].update(dict_to_add)
return meta_info
def get_icon_hash(fp, fp_name):
"""Returns a file's hashed, fp is passed as a string"""
sha1 = hashlib.sha1(fp)
return sha1.hexdigest()
def generate_icon_states(filename, save_each=False):
"""Generates every icon state into an Image object. Returning a dict with {name : (object, icon hash)}"""
img = PIL.Image.open(filename)
meta_data = parse_metadata(img)
if meta_data is None:
print("Failed to retreive metadata.")
return
image_width = img.width
image_height = img.height
icon_width = meta_data[0][1]['width']
icon_height = meta_data[0][1]['height']
meta_data = meta_data[1:] #We don't need the version info anymore
icons_per_line = int(image_width / icon_width)
total_lines = int(image_height / icon_height)
if img.mode != "RGBA":
img = img.convert("RGBA")
data = np.asarray(img)
img.close()
icon_count = 0
skip_naming = 0
name_count = 1
icons = {}
for line in range(0, total_lines):
icon = 0
while icon < icons_per_line:
this_icon = PIL.Image.new('RGBA', (icon_width, icon_height))
try:
state_tuple = meta_data[icon_count] # (name, {'dirs' : 1, 'frames' : 1})
name = state_tuple[0]
if skip_naming:
if name_count > 0:
name += "[{}]".format(str(name_count))
name_count += 1
skip_naming -= 1
if not skip_naming:
icon_count += 1
else:
amt_dirs = state_tuple[1]['dirs']
amt_frames = state_tuple[1]['frames']
skip_naming = (amt_dirs * amt_frames) - 1
if not skip_naming:
icon_count += 1
else:
name_count = 1
except IndexError:
break #IndexError means blank icon
icon_start_w = icon * icon_width
icon_end_w = icon_start_w + icon_width
icon_start_h = line * icon_height
icon_end_h = icon_start_h + icon_height
this_state_x = 0
for i in range(icon_start_w, icon_end_w):
this_state_y = 0
for j in range(icon_start_h, icon_end_h):
this_icon.putpixel((this_state_x, this_state_y), tuple(data[j, i]))
this_state_y += 1
this_state_x += 1
icon += 1
icons[name] = this_icon
if save_each:
this_icon.save("icon_dump/{}.png".format(name))
return icons
def check_icon_state_diff(image_a, image_b):
"""Compares two icons(passed as an Image object), returning True if the icons are equal, False in case of a difference."""
return ImageChops.difference(image_a, image_b).getbbox() is None
def compare_two_icon_files(file_a, file_b):
"""
Compares every icon state of two icons, returning a dict with the icon state status:
{state name : {
status : no_check/modified/created,
img_a : Image obj,
img_a_hash: sha1 of Image a obj,
img_b : Image obj,
img_b_hash: sha1 of Image b obj
}
}
"""
if file_a:
file_a_dict = generate_icon_states(file_a)
else:
file_a_dict = {}
file_b_dict = generate_icon_states(file_b)
final_dict = OrderedDict()
for key in file_a_dict:
final_dict[key] = {}
final_dict[key]["img_a_hash"] = get_icon_hash(file_a_dict[key].tobytes(), key)
if not file_b_dict.get(key):
final_dict[key]["status"] = "Removed"
final_dict[key]["img_a"] = file_a_dict[key]
elif check_icon_state_diff(file_a_dict[key], file_b_dict[key]):
final_dict[key]["status"] = "Equal"
file_a_dict[key].close()
file_b_dict[key].close()
else:
final_dict[key]["status"] = "Modified"
final_dict[key]["img_a"] = file_a_dict[key]
final_dict[key]["img_b"] = file_b_dict[key]
final_dict[key]["img_b_hash"] = get_icon_hash(file_b_dict[key].tobytes(), key)
for key in file_b_dict:
if not file_a_dict.get(key):
final_dict[key] = {
"status" : "Created",
"img_b" : file_b_dict[key],
"img_b_hash" : get_icon_hash(file_b_dict[key].tobytes(), key)
}
return final_dict
if __name__ == '__main__':
with open("./icon_dump/new_unary_devices.dmi", 'rb') as f:
generate_icon_states(f)