Jeśli nie dbają o prędkości zbyt wiele, można to zrobić za pomocą skład: tekst
- rysować koloru halo na pustym
RGBA
obrazu
- plama to
- wyciągnąć go ponownie kolor tekstu
- inwertowany ten obraz, aby uzyskać skład maskować
- „scalić” z oryginalnego obrazu
Na przykład:
import sys
import Image, ImageChops, ImageDraw, ImageFont, ImageFilter
def draw_text_with_halo(img, position, text, font, col, halo_col):
halo = Image.new('RGBA', img.size, (0, 0, 0, 0))
ImageDraw.Draw(halo).text(position, text, font = font, fill = halo_col)
blurred_halo = halo.filter(ImageFilter.BLUR)
ImageDraw.Draw(blurred_halo).text(position, text, font = font, fill = col)
return Image.composite(img, blurred_halo, ImageChops.invert(blurred_halo))
if __name__ == '__main__':
i = Image.open(sys.argv[1])
font = ImageFont.load_default()
txt = 'Example 1234'
text_col = (0, 255, 0) # bright green
halo_col = (0, 0, 0) # black
i2 = draw_text_with_halo(i, (20, 20), txt, font, text_col, halo_col)
i2.save('halo.png')
Ma wiele zalet:
- efektem jest gładka i wygląda ładnie
- można wybrać inny filtr zamiast
BLUR
dostać inny "halo"
- działa nawet przy bardzo dużych czcionek i nadal wygląda świetnie
Aby uzyskać grubsze halo, można użyć filtra tak:
kernel = [
0, 1, 2, 1, 0,
1, 2, 4, 2, 1,
2, 4, 8, 4, 1,
1, 2, 4, 2, 1,
0, 1, 2, 1, 0]
kernelsum = sum(kernel)
myfilter = ImageFilter.Kernel((5, 5), kernel, scale = 0.1 * sum(kernel))
blurred_halo = halo.filter(myfilter)
Część scale = 0.1 * sum(kernel)
sprawia aureolę grubszą (małe wartości) lub ściemniacza (duże wartości).
wydrukować tekst trzy razy. Pierwsze dwa z kolorem zarys i przesunięciami (-1, -1) oraz (1, 1), a trzeci z pierwotnego koloru w pozycji wyjściowej. – Matthias
http://mail.python.org/pipermail/image-sig/2009-May/005681.html http://stackoverflow.com/a/8050556/442650 –
PIL.ImageDraw ma opcję "konspektu", ale nie dotyczy tekstu. –