2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# The Python Imaging Library.
|
|
|
|
#
|
|
|
|
# MSP file handling
|
|
|
|
#
|
|
|
|
# This is the format used by the Paint program in Windows 1 and 2.
|
|
|
|
#
|
|
|
|
# History:
|
|
|
|
# 95-09-05 fl Created
|
|
|
|
# 97-01-03 fl Read/write MSP images
|
2017-02-21 16:09:32 +03:00
|
|
|
# 17-02-21 es Fixed RLE interpretation
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# Copyright (c) Secret Labs AB 1997.
|
|
|
|
# Copyright (c) Fredrik Lundh 1995-97.
|
2017-02-21 16:09:32 +03:00
|
|
|
# Copyright (c) Eric Soroos 2017.
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# See the README file for information on usage and redistribution.
|
|
|
|
#
|
2017-01-29 23:28:41 +03:00
|
|
|
# More info on this format: https://archive.org/details/gg243631
|
|
|
|
# Page 313:
|
|
|
|
# Figure 205. Windows Paint Version 1: "DanM" Format
|
|
|
|
# Figure 206. Windows Paint Version 2: "LinS" Format. Used in Windows V2.03
|
|
|
|
#
|
2021-07-12 14:00:36 +03:00
|
|
|
# See also: https://www.fileformat.info/format/mspaint/egff.htm
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-04-20 14:14:23 +03:00
|
|
|
import io
|
2019-07-06 23:40:53 +03:00
|
|
|
import struct
|
|
|
|
|
|
|
|
from . import Image, ImageFile
|
2020-09-01 20:16:46 +03:00
|
|
|
from ._binary import i16le as i16
|
|
|
|
from ._binary import o16le as o16
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
#
|
|
|
|
# read MSP files
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
def _accept(prefix):
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
return prefix[:4] in [b"DanM", b"LinS"]
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
##
|
|
|
|
# Image plugin for Windows MSP images. This plugin supports both
|
|
|
|
# uncompressed (Windows 1.0).
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
class MspImageFile(ImageFile.ImageFile):
|
|
|
|
format = "MSP"
|
|
|
|
format_description = "Windows Paint"
|
|
|
|
|
|
|
|
def _open(self):
|
|
|
|
# Header
|
|
|
|
s = self.fp.read(32)
|
2020-06-23 10:41:13 +03:00
|
|
|
if not _accept(s):
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "not an MSP file"
|
|
|
|
raise SyntaxError(msg)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
# Header checksum
|
2015-04-02 11:45:24 +03:00
|
|
|
checksum = 0
|
2010-07-31 06:52:47 +04:00
|
|
|
for i in range(0, 32, 2):
|
2020-05-08 21:11:58 +03:00
|
|
|
checksum = checksum ^ i16(s, i)
|
2015-04-02 11:45:24 +03:00
|
|
|
if checksum != 0:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "bad MSP checksum"
|
|
|
|
raise SyntaxError(msg)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
self.mode = "1"
|
2020-05-08 21:11:58 +03:00
|
|
|
self._size = i16(s, 4), i16(s, 6)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
if s[:4] == b"DanM":
|
2019-03-21 16:28:20 +03:00
|
|
|
self.tile = [("raw", (0, 0) + self.size, 32, ("1", 0, 1))]
|
2010-07-31 06:52:47 +04:00
|
|
|
else:
|
2019-03-21 16:28:20 +03:00
|
|
|
self.tile = [("MSP", (0, 0) + self.size, 32, None)]
|
2017-02-21 16:09:32 +03:00
|
|
|
|
|
|
|
|
|
|
|
class MspDecoder(ImageFile.PyDecoder):
|
|
|
|
# The algo for the MSP decoder is from
|
2021-07-12 14:00:36 +03:00
|
|
|
# https://www.fileformat.info/format/mspaint/egff.htm
|
2017-02-21 16:09:32 +03:00
|
|
|
# cc-by-attribution -- That page references is taken from the
|
|
|
|
# Encyclopedia of Graphics File Formats and is licensed by
|
|
|
|
# O'Reilly under the Creative Common/Attribution license
|
|
|
|
#
|
|
|
|
# For RLE encoded files, the 32byte header is followed by a scan
|
|
|
|
# line map, encoded as one 16bit word of encoded byte length per
|
|
|
|
# line.
|
|
|
|
#
|
|
|
|
# NOTE: the encoded length of the line can be 0. This was not
|
|
|
|
# handled in the previous version of this encoder, and there's no
|
|
|
|
# mention of how to handle it in the documentation. From the few
|
|
|
|
# examples I've seen, I've assumed that it is a fill of the
|
|
|
|
# background color, in this case, white.
|
|
|
|
#
|
|
|
|
#
|
|
|
|
# Pseudocode of the decoder:
|
|
|
|
# Read a BYTE value as the RunType
|
2017-02-23 13:20:25 +03:00
|
|
|
# If the RunType value is zero
|
|
|
|
# Read next byte as the RunCount
|
|
|
|
# Read the next byte as the RunValue
|
|
|
|
# Write the RunValue byte RunCount times
|
|
|
|
# If the RunType value is non-zero
|
|
|
|
# Use this value as the RunCount
|
|
|
|
# Read and write the next RunCount bytes literally
|
2017-04-20 14:14:23 +03:00
|
|
|
#
|
2017-02-21 16:09:32 +03:00
|
|
|
# e.g.:
|
|
|
|
# 0x00 03 ff 05 00 01 02 03 04
|
|
|
|
# would yield the bytes:
|
|
|
|
# 0xff ff ff 00 01 02 03 04
|
|
|
|
#
|
|
|
|
# which are then interpreted as a bit packed mode '1' image
|
|
|
|
|
|
|
|
_pulls_fd = True
|
|
|
|
|
|
|
|
def decode(self, buffer):
|
|
|
|
img = io.BytesIO()
|
2019-03-21 16:28:20 +03:00
|
|
|
blank_line = bytearray((0xFF,) * ((self.state.xsize + 7) // 8))
|
2017-02-21 16:09:32 +03:00
|
|
|
try:
|
|
|
|
self.fd.seek(32)
|
2019-03-21 16:28:20 +03:00
|
|
|
rowmap = struct.unpack_from(
|
2020-07-16 12:43:29 +03:00
|
|
|
f"<{self.state.ysize}H", self.fd.read(self.state.ysize * 2)
|
2019-03-21 16:28:20 +03:00
|
|
|
)
|
2020-06-21 13:13:35 +03:00
|
|
|
except struct.error as e:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "Truncated MSP file in row map"
|
|
|
|
raise OSError(msg) from e
|
2017-02-21 16:09:32 +03:00
|
|
|
|
|
|
|
for x, rowlen in enumerate(rowmap):
|
|
|
|
try:
|
|
|
|
if rowlen == 0:
|
|
|
|
img.write(blank_line)
|
|
|
|
continue
|
|
|
|
row = self.fd.read(rowlen)
|
|
|
|
if len(row) != rowlen:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = f"Truncated MSP file, expected {rowlen} bytes on row {x}"
|
|
|
|
raise OSError(msg)
|
2017-02-21 16:09:32 +03:00
|
|
|
idx = 0
|
|
|
|
while idx < rowlen:
|
2020-05-08 19:48:02 +03:00
|
|
|
runtype = row[idx]
|
2017-02-21 16:09:32 +03:00
|
|
|
idx += 1
|
|
|
|
if runtype == 0:
|
2018-06-24 20:00:22 +03:00
|
|
|
(runcount, runval) = struct.unpack_from("Bc", row, idx)
|
2017-02-21 16:09:32 +03:00
|
|
|
img.write(runval * runcount)
|
|
|
|
idx += 2
|
|
|
|
else:
|
|
|
|
runcount = runtype
|
2019-03-21 16:28:20 +03:00
|
|
|
img.write(row[idx : idx + runcount])
|
2017-02-21 16:09:32 +03:00
|
|
|
idx += runcount
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2020-06-21 13:13:35 +03:00
|
|
|
except struct.error as e:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = f"Corrupted MSP file in row {x}"
|
|
|
|
raise OSError(msg) from e
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2017-02-21 16:09:32 +03:00
|
|
|
self.set_as_raw(img.getvalue(), ("1", 0, 1))
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2022-03-08 11:48:58 +03:00
|
|
|
return -1, 0
|
2017-02-21 16:09:32 +03:00
|
|
|
|
|
|
|
|
2019-03-21 16:28:20 +03:00
|
|
|
Image.register_decoder("MSP", MspDecoder)
|
2017-04-20 14:14:23 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
#
|
|
|
|
# write MSP files (uncompressed only)
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
def _save(im, fp, filename):
|
|
|
|
if im.mode != "1":
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = f"cannot write mode {im.mode} as MSP"
|
|
|
|
raise OSError(msg)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
# create MSP header
|
|
|
|
header = [0] * 16
|
|
|
|
|
2014-08-26 17:47:10 +04:00
|
|
|
header[0], header[1] = i16(b"Da"), i16(b"nM") # version 1
|
2010-07-31 06:52:47 +04:00
|
|
|
header[2], header[3] = im.size
|
|
|
|
header[4], header[5] = 1, 1
|
|
|
|
header[6], header[7] = 1, 1
|
|
|
|
header[8], header[9] = im.size
|
|
|
|
|
2015-04-02 11:45:24 +03:00
|
|
|
checksum = 0
|
2010-07-31 06:52:47 +04:00
|
|
|
for h in header:
|
2015-04-02 11:45:24 +03:00
|
|
|
checksum = checksum ^ h
|
|
|
|
header[12] = checksum # FIXME: is this the right field?
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
# header
|
|
|
|
for h in header:
|
|
|
|
fp.write(o16(h))
|
|
|
|
|
|
|
|
# image body
|
2019-03-21 16:28:20 +03:00
|
|
|
ImageFile._save(im, fp, [("raw", (0, 0) + im.size, 32, ("1", 0, 1))])
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-01-29 23:28:41 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# registry
|
|
|
|
|
2015-07-04 16:29:58 +03:00
|
|
|
Image.register_open(MspImageFile.format, MspImageFile, _accept)
|
|
|
|
Image.register_save(MspImageFile.format, _save)
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2015-07-04 16:29:58 +03:00
|
|
|
Image.register_extension(MspImageFile.format, ".msp")
|