-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathbmp_handler.py
More file actions
172 lines (141 loc) · 6.96 KB
/
bmp_handler.py
File metadata and controls
172 lines (141 loc) · 6.96 KB
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
"""
Python class for handling BMP files.
V5 BMP specifically (created by GIMP), since it supports transparency.
"""
import struct
BITMAPFILEHEADER = {
"Signature": b'BM',
"File Size": 0, # image_size + header_size
"Reserved1": 0,
"Offset to PixelArray": 0 # header_size
}
BITMAPV5HEADER = {
"DIB Header Size": 124,
"Image Width": 0,
"Image Height": 0,
"Planes": 1, # Always 1. 2 bytes
"BPP": 16, # 2 bytes
"Compression": 3, # 0 - BI_RGB, 3 - BI_BITMAP
"Image Size": 1,
"X Pixels Per Meter": 0,
"Y Pixels Per Meter": 0,
"Color Count": 0,
"Important Colors": 0,
"R Bitmask": 0b1111100000000000, # 63488
"G Bitmask": 0b0000011111100000, # 2016
"B Bitmask": 0b0000000000011111, # 31
"A Bitmask": 0b0000000000000000,
"Color Space Type": b'BGRs',
#"Color Space Endpoints": 0, # 36 bytes, no idea what it does
"R Gamma": 0,
"G Gamma": 0,
"B Gamma": 0,
"Intent": 2,
"ICC Profile Data": 0,
"ICC Profile Size": 0,
"Reserved2": 0,
}
def fill_dict_from_bytes_by_formatting(dictionary: dict, source_bytes: bytes, formatting: str) -> dict:
value_list = struct.unpack(formatting, source_bytes)
dictionary = dict(zip(dictionary.keys(), value_list))
return dictionary
class BMPv5:
def __init__(self):
# Header structure
self.FILE_HEADER_SIZE = 14
self.BMP_FILE_HEADER_FORMATTING = "<2s3I" # 14
self.file_header = dict.copy(BITMAPFILEHEADER)
self.IMAGE_HEADER_SIZE = 124
self.BMP_IMAGE_HEADER_FORMATTING = "<2Ii2hI" + "I2i2I" + "4I" + "4s36x4I" + "3I" # 124
self.image_header = dict.copy(BITMAPV5HEADER)
self.color_table_bytes = b''
self.pixel_bytes = b''
self.update_file_size_and_pixel_offset()
def set_resolution(self, image_width, image_height):
self.image_header["Image Width"] = image_width
self.image_header["Image Height"] = - image_height
self.image_header["X Pixels Per Meter"] = image_width
self.image_header["Y Pixels Per Meter"] = - image_height
def check_color_format(self):
if self.image_header["Compression"] == 0:
if self.image_header["BPP"] <= 8:
return "Palette"
elif self.image_header["BPP"] == 24:
return "RGB888"
elif self.image_header["BPP"] == 32:
return "ARGB8888"
else:
return "Other"
elif self.image_header["Compression"] == 3:
if self.image_header["BPP"] == 32:
return "ARGB8888"
if self.image_header["R Bitmask"] == 0b1111100000000000 \
and self.image_header["G Bitmask"] == 0b0000011111100000 \
and self.image_header["B Bitmask"] == 0b0000000000011111 \
and self.image_header["A Bitmask"] == 0b0000000000000000:
return "RGB565"
elif self.image_header["R Bitmask"] == 0b0111110000000000 \
and self.image_header["G Bitmask"] == 0b0000001111100000 \
and self.image_header["B Bitmask"] == 0b0000000000011111 \
and self.image_header["A Bitmask"] == 0b1000000000000000:
return "ARGB1555"
elif self.image_header["R Bitmask"] == 0b0000111100000000 \
and self.image_header["G Bitmask"] == 0b0000000011110000 \
and self.image_header["B Bitmask"] == 0b0000000000001111 \
and self.image_header["A Bitmask"] == 0b1111000000000000:
return "ARGB4444"
else:
return "Other"
def update_file_size_and_pixel_offset(self):
self.file_header["File Size"] = (self.FILE_HEADER_SIZE
+ self.IMAGE_HEADER_SIZE
+ len(self.color_table_bytes)
+ len(self.pixel_bytes))
self.file_header["Offset to PixelArray"] = (self.FILE_HEADER_SIZE
+ self.IMAGE_HEADER_SIZE
+ len(self.color_table_bytes))
self.image_header["Image Size"] = len(self.pixel_bytes)
def swapColorFormat(self, color_format):
if color_format == "RGB565":
self.image_header["A Bitmask"] = 0b0000000000000000
self.image_header["R Bitmask"] = 0b1111100000000000
self.image_header["G Bitmask"] = 0b0000011111100000
self.image_header["B Bitmask"] = 0b0000000000011111
elif color_format == "ARGB1555":
self.image_header["A Bitmask"] = 0b1000000000000000
self.image_header["R Bitmask"] = 0b0111110000000000
self.image_header["G Bitmask"] = 0b0000001111100000
self.image_header["B Bitmask"] = 0b0000000000011111
elif color_format == "ARGB4444":
self.image_header["A Bitmask"] = 0b1111000000000000
self.image_header["R Bitmask"] = 0b0000111100000000
self.image_header["G Bitmask"] = 0b0000000011110000
self.image_header["B Bitmask"] = 0b0000000000001111
else:
raise ValueError("Incorrect BMP Color Format")
def unpack_from_bytes(self, bitmap_bytes):
self.file_header = fill_dict_from_bytes_by_formatting(self.file_header,
bitmap_bytes[:self.FILE_HEADER_SIZE],
self.BMP_FILE_HEADER_FORMATTING)
self.image_header = fill_dict_from_bytes_by_formatting(self.image_header,
bitmap_bytes[self.FILE_HEADER_SIZE:self.FILE_HEADER_SIZE + self.IMAGE_HEADER_SIZE],
self.BMP_IMAGE_HEADER_FORMATTING)
# Read palette if present
if self.file_header["Offset to PixelArray"] - (self.FILE_HEADER_SIZE + self.IMAGE_HEADER_SIZE) > 0:
self.color_table_bytes = bitmap_bytes[self.FILE_HEADER_SIZE + self.IMAGE_HEADER_SIZE:self.file_header["Offset to PixelArray"]]
self.pixel_bytes = bitmap_bytes[self.file_header["Offset to PixelArray"]:]
def unpack_from_file(self, bitmap_path: str):
bitmap_file = open(bitmap_path, "r+b")
bitmap_bytes = bitmap_file.read()
self.unpack_from_bytes(bitmap_bytes)
bitmap_file.close()
def pack_and_return(self):
self.update_file_size_and_pixel_offset()
file_header_bytes = struct.pack(self.BMP_FILE_HEADER_FORMATTING, *list(self.file_header.values()))
image_header_bytes = struct.pack(self.BMP_IMAGE_HEADER_FORMATTING, *list(self.image_header.values()))
return file_header_bytes + image_header_bytes + self.color_table_bytes + self.pixel_bytes
def save(self, output_path):
bin_data = self.pack_and_return()
output_file = open(output_path, "w+b")
output_file.write(bin_data)
output_file.close()