123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194 |
- import io
- import struct
- from . import Image, ImageFile
- from ._binary import i16le as i16
- from ._binary import o16le as o16
- def _accept(prefix):
- return prefix[:4] in [b"DanM", b"LinS"]
- class MspImageFile(ImageFile.ImageFile):
- format = "MSP"
- format_description = "Windows Paint"
- def _open(self):
-
- s = self.fp.read(32)
- if not _accept(s):
- raise SyntaxError("not an MSP file")
-
- checksum = 0
- for i in range(0, 32, 2):
- checksum = checksum ^ i16(s, i)
- if checksum != 0:
- raise SyntaxError("bad MSP checksum")
- self.mode = "1"
- self._size = i16(s, 4), i16(s, 6)
- if s[:4] == b"DanM":
- self.tile = [("raw", (0, 0) + self.size, 32, ("1", 0, 1))]
- else:
- self.tile = [("MSP", (0, 0) + self.size, 32, None)]
- class MspDecoder(ImageFile.PyDecoder):
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- _pulls_fd = True
- def decode(self, buffer):
- img = io.BytesIO()
- blank_line = bytearray((0xFF,) * ((self.state.xsize + 7) // 8))
- try:
- self.fd.seek(32)
- rowmap = struct.unpack_from(
- f"<{self.state.ysize}H", self.fd.read(self.state.ysize * 2)
- )
- except struct.error as e:
- raise OSError("Truncated MSP file in row map") from e
- for x, rowlen in enumerate(rowmap):
- try:
- if rowlen == 0:
- img.write(blank_line)
- continue
- row = self.fd.read(rowlen)
- if len(row) != rowlen:
- raise OSError(
- "Truncated MSP file, expected %d bytes on row %s", (rowlen, x)
- )
- idx = 0
- while idx < rowlen:
- runtype = row[idx]
- idx += 1
- if runtype == 0:
- (runcount, runval) = struct.unpack_from("Bc", row, idx)
- img.write(runval * runcount)
- idx += 2
- else:
- runcount = runtype
- img.write(row[idx : idx + runcount])
- idx += runcount
- except struct.error as e:
- raise OSError(f"Corrupted MSP file in row {x}") from e
- self.set_as_raw(img.getvalue(), ("1", 0, 1))
- return 0, 0
- Image.register_decoder("MSP", MspDecoder)
- def _save(im, fp, filename):
- if im.mode != "1":
- raise OSError(f"cannot write mode {im.mode} as MSP")
-
- header = [0] * 16
- header[0], header[1] = i16(b"Da"), i16(b"nM")
- header[2], header[3] = im.size
- header[4], header[5] = 1, 1
- header[6], header[7] = 1, 1
- header[8], header[9] = im.size
- checksum = 0
- for h in header:
- checksum = checksum ^ h
- header[12] = checksum
-
- for h in header:
- fp.write(o16(h))
-
- ImageFile._save(im, fp, [("raw", (0, 0) + im.size, 32, ("1", 0, 1))])
- Image.register_open(MspImageFile.format, MspImageFile, _accept)
- Image.register_save(MspImageFile.format, _save)
- Image.register_extension(MspImageFile.format, ".msp")
|