From 4e6aa7d6d3733629b31ac46fae96681c79822faf Mon Sep 17 00:00:00 2001 From: Andrew Murray Date: Mon, 8 Apr 2019 00:27:16 +1000 Subject: [PATCH] Fixed loading font with non-Unicode path on Windows --- Tests/test_imagefont.py | 10 ++++++++++ src/PIL/ImageFont.py | 19 ++++++++++++++++--- 2 files changed, 26 insertions(+), 3 deletions(-) diff --git a/Tests/test_imagefont.py b/Tests/test_imagefont.py index 33f37916c..91ff44c13 100644 --- a/Tests/test_imagefont.py +++ b/Tests/test_imagefont.py @@ -7,6 +7,7 @@ import os import sys import copy import re +import shutil import distutils.version FONT_PATH = "Tests/fonts/FreeMono.ttf" @@ -131,6 +132,15 @@ class TestImageFont(PillowTestCase): with open(FONT_PATH, 'rb') as f: self._render(f) + def test_non_unicode_path(self): + try: + tempfile = self.tempfile("temp_"+chr(128)+".ttf") + except UnicodeEncodeError: + self.skipTest("Unicode path could not be created") + shutil.copy(FONT_PATH, tempfile) + + ImageFont.truetype(tempfile, FONT_SIZE) + def _render(self, font): txt = "Hello World!" ttf = ImageFont.truetype(font, FONT_SIZE, diff --git a/src/PIL/ImageFont.py b/src/PIL/ImageFont.py index 580aa8744..145278e6c 100644 --- a/src/PIL/ImageFont.py +++ b/src/PIL/ImageFont.py @@ -140,13 +140,26 @@ class FreeTypeFont(object): self.layout_engine = layout_engine + def load_from_bytes(f): + self.font_bytes = f.read() + self.font = core.getfont( + "", size, index, encoding, self.font_bytes, layout_engine) + if isPath(font): + if sys.platform == "win32": + font_bytes_path = font if isinstance(font, bytes) else font.encode() + try: + font_bytes_path.decode('ascii') + except UnicodeDecodeError: + # FreeType cannot load fonts with non-ASCII characters on Windows + # So load it into memory first + with open(font, 'rb') as f: + load_from_bytes(f) + return self.font = core.getfont(font, size, index, encoding, layout_engine=layout_engine) else: - self.font_bytes = font.read() - self.font = core.getfont( - "", size, index, encoding, self.font_bytes, layout_engine) + load_from_bytes(font) def _multiline_split(self, text): split_character = "\n" if isinstance(text, str) else b"\n"