| # | 
| # The Python Imaging Library | 
| # $Id$ | 
| # | 
| # drawing interface operations | 
| # | 
| # History: | 
| # 1996-04-13 fl   Created (experimental) | 
| # 1996-08-07 fl   Filled polygons, ellipses. | 
| # 1996-08-13 fl   Added text support | 
| # 1998-06-28 fl   Handle I and F images | 
| # 1998-12-29 fl   Added arc; use arc primitive to draw ellipses | 
| # 1999-01-10 fl   Added shape stuff (experimental) | 
| # 1999-02-06 fl   Added bitmap support | 
| # 1999-02-11 fl   Changed all primitives to take options | 
| # 1999-02-20 fl   Fixed backwards compatibility | 
| # 2000-10-12 fl   Copy on write, when necessary | 
| # 2001-02-18 fl   Use default ink for bitmap/text also in fill mode | 
| # 2002-10-24 fl   Added support for CSS-style color strings | 
| # 2002-12-10 fl   Added experimental support for RGBA-on-RGB drawing | 
| # 2002-12-11 fl   Refactored low-level drawing API (work in progress) | 
| # 2004-08-26 fl   Made Draw() a factory function, added getdraw() support | 
| # 2004-09-04 fl   Added width support to line primitive | 
| # 2004-09-10 fl   Added font mode handling | 
| # 2006-06-19 fl   Added font bearing support (getmask2) | 
| # | 
| # Copyright (c) 1997-2006 by Secret Labs AB | 
| # Copyright (c) 1996-2006 by Fredrik Lundh | 
| # | 
| # See the README file for information on usage and redistribution. | 
| # | 
|   | 
| import Image, ImageColor | 
|   | 
| try: | 
|     import warnings | 
| except ImportError: | 
|     warnings = None | 
|   | 
| ## | 
| # A simple 2D drawing interface for PIL images. | 
| # <p> | 
| # Application code should use the <b>Draw</b> factory, instead of | 
| # directly. | 
|   | 
| class ImageDraw: | 
|   | 
|     ## | 
|     # Create a drawing instance. | 
|     # | 
|     # @param im The image to draw in. | 
|     # @param mode Optional mode to use for color values.  For RGB | 
|     #    images, this argument can be RGB or RGBA (to blend the | 
|     #    drawing into the image).  For all other modes, this argument | 
|     #    must be the same as the image mode.  If omitted, the mode | 
|     #    defaults to the mode of the image. | 
|   | 
|     def __init__(self, im, mode=None): | 
|         im.load() | 
|         if im.readonly: | 
|             im._copy() # make it writable | 
|         blend = 0 | 
|         if mode is None: | 
|             mode = im.mode | 
|         if mode != im.mode: | 
|             if mode == "RGBA" and im.mode == "RGB": | 
|                 blend = 1 | 
|             else: | 
|                 raise ValueError("mode mismatch") | 
|         if mode == "P": | 
|             self.palette = im.palette | 
|         else: | 
|             self.palette = None | 
|         self.im = im.im | 
|         self.draw = Image.core.draw(self.im, blend) | 
|         self.mode = mode | 
|         if mode in ("I", "F"): | 
|             self.ink = self.draw.draw_ink(1, mode) | 
|         else: | 
|             self.ink = self.draw.draw_ink(-1, mode) | 
|         if mode in ("1", "P", "I", "F"): | 
|             # FIXME: fix Fill2 to properly support matte for I+F images | 
|             self.fontmode = "1" | 
|         else: | 
|             self.fontmode = "L" # aliasing is okay for other modes | 
|         self.fill = 0 | 
|         self.font = None | 
|   | 
|     ## | 
|     # Set the default pen color. | 
|   | 
|     def setink(self, ink): | 
|         # compatibility | 
|         if warnings: | 
|             warnings.warn( | 
|                 "'setink' is deprecated; use keyword arguments instead", | 
|                 DeprecationWarning, stacklevel=2 | 
|                 ) | 
|         if Image.isStringType(ink): | 
|             ink = ImageColor.getcolor(ink, self.mode) | 
|         if self.palette and not Image.isNumberType(ink): | 
|             ink = self.palette.getcolor(ink) | 
|         self.ink = self.draw.draw_ink(ink, self.mode) | 
|   | 
|     ## | 
|     # Set the default background color. | 
|   | 
|     def setfill(self, onoff): | 
|         # compatibility | 
|         if warnings: | 
|             warnings.warn( | 
|                 "'setfill' is deprecated; use keyword arguments instead", | 
|                 DeprecationWarning, stacklevel=2 | 
|                 ) | 
|         self.fill = onoff | 
|   | 
|     ## | 
|     # Set the default font. | 
|   | 
|     def setfont(self, font): | 
|         # compatibility | 
|         self.font = font | 
|   | 
|     ## | 
|     # Get the current default font. | 
|   | 
|     def getfont(self): | 
|         if not self.font: | 
|             # FIXME: should add a font repository | 
|             import ImageFont | 
|             self.font = ImageFont.load_default() | 
|         return self.font | 
|   | 
|     def _getink(self, ink, fill=None): | 
|         if ink is None and fill is None: | 
|             if self.fill: | 
|                 fill = self.ink | 
|             else: | 
|                 ink = self.ink | 
|         else: | 
|             if ink is not None: | 
|                 if Image.isStringType(ink): | 
|                     ink = ImageColor.getcolor(ink, self.mode) | 
|                 if self.palette and not Image.isNumberType(ink): | 
|                     ink = self.palette.getcolor(ink) | 
|                 ink = self.draw.draw_ink(ink, self.mode) | 
|             if fill is not None: | 
|                 if Image.isStringType(fill): | 
|                     fill = ImageColor.getcolor(fill, self.mode) | 
|                 if self.palette and not Image.isNumberType(fill): | 
|                     fill = self.palette.getcolor(fill) | 
|                 fill = self.draw.draw_ink(fill, self.mode) | 
|         return ink, fill | 
|   | 
|     ## | 
|     # Draw an arc. | 
|   | 
|     def arc(self, xy, start, end, fill=None): | 
|         ink, fill = self._getink(fill) | 
|         if ink is not None: | 
|             self.draw.draw_arc(xy, start, end, ink) | 
|   | 
|     ## | 
|     # Draw a bitmap. | 
|   | 
|     def bitmap(self, xy, bitmap, fill=None): | 
|         bitmap.load() | 
|         ink, fill = self._getink(fill) | 
|         if ink is None: | 
|             ink = fill | 
|         if ink is not None: | 
|             self.draw.draw_bitmap(xy, bitmap.im, ink) | 
|   | 
|     ## | 
|     # Draw a chord. | 
|   | 
|     def chord(self, xy, start, end, fill=None, outline=None): | 
|         ink, fill = self._getink(outline, fill) | 
|         if fill is not None: | 
|             self.draw.draw_chord(xy, start, end, fill, 1) | 
|         if ink is not None: | 
|             self.draw.draw_chord(xy, start, end, ink, 0) | 
|   | 
|     ## | 
|     # Draw an ellipse. | 
|   | 
|     def ellipse(self, xy, fill=None, outline=None): | 
|         ink, fill = self._getink(outline, fill) | 
|         if fill is not None: | 
|             self.draw.draw_ellipse(xy, fill, 1) | 
|         if ink is not None: | 
|             self.draw.draw_ellipse(xy, ink, 0) | 
|   | 
|     ## | 
|     # Draw a line, or a connected sequence of line segments. | 
|   | 
|     def line(self, xy, fill=None, width=0): | 
|         ink, fill = self._getink(fill) | 
|         if ink is not None: | 
|             self.draw.draw_lines(xy, ink, width) | 
|   | 
|     ## | 
|     # (Experimental) Draw a shape. | 
|   | 
|     def shape(self, shape, fill=None, outline=None): | 
|         # experimental | 
|         shape.close() | 
|         ink, fill = self._getink(outline, fill) | 
|         if fill is not None: | 
|             self.draw.draw_outline(shape, fill, 1) | 
|         if ink is not None: | 
|             self.draw.draw_outline(shape, ink, 0) | 
|   | 
|     ## | 
|     # Draw a pieslice. | 
|   | 
|     def pieslice(self, xy, start, end, fill=None, outline=None): | 
|         ink, fill = self._getink(outline, fill) | 
|         if fill is not None: | 
|             self.draw.draw_pieslice(xy, start, end, fill, 1) | 
|         if ink is not None: | 
|             self.draw.draw_pieslice(xy, start, end, ink, 0) | 
|   | 
|     ## | 
|     # Draw one or more individual pixels. | 
|   | 
|     def point(self, xy, fill=None): | 
|         ink, fill = self._getink(fill) | 
|         if ink is not None: | 
|             self.draw.draw_points(xy, ink) | 
|   | 
|     ## | 
|     # Draw a polygon. | 
|   | 
|     def polygon(self, xy, fill=None, outline=None): | 
|         ink, fill = self._getink(outline, fill) | 
|         if fill is not None: | 
|             self.draw.draw_polygon(xy, fill, 1) | 
|         if ink is not None: | 
|             self.draw.draw_polygon(xy, ink, 0) | 
|   | 
|     ## | 
|     # Draw a rectangle. | 
|   | 
|     def rectangle(self, xy, fill=None, outline=None): | 
|         ink, fill = self._getink(outline, fill) | 
|         if fill is not None: | 
|             self.draw.draw_rectangle(xy, fill, 1) | 
|         if ink is not None: | 
|             self.draw.draw_rectangle(xy, ink, 0) | 
|   | 
|     ## | 
|     # Draw text. | 
|   | 
|     def text(self, xy, text, fill=None, font=None, anchor=None): | 
|         ink, fill = self._getink(fill) | 
|         if font is None: | 
|             font = self.getfont() | 
|         if ink is None: | 
|             ink = fill | 
|         if ink is not None: | 
|             try: | 
|                 mask, offset = font.getmask2(text, self.fontmode) | 
|                 xy = xy[0] + offset[0], xy[1] + offset[1] | 
|             except AttributeError: | 
|                 try: | 
|                     mask = font.getmask(text, self.fontmode) | 
|                 except TypeError: | 
|                     mask = font.getmask(text) | 
|             self.draw.draw_bitmap(xy, mask, ink) | 
|   | 
|     ## | 
|     # Get the size of a given string, in pixels. | 
|   | 
|     def textsize(self, text, font=None): | 
|         if font is None: | 
|             font = self.getfont() | 
|         return font.getsize(text) | 
|   | 
| ## | 
| # A simple 2D drawing interface for PIL images. | 
| # | 
| # @param im The image to draw in. | 
| # @param mode Optional mode to use for color values.  For RGB | 
| #    images, this argument can be RGB or RGBA (to blend the | 
| #    drawing into the image).  For all other modes, this argument | 
| #    must be the same as the image mode.  If omitted, the mode | 
| #    defaults to the mode of the image. | 
|   | 
| def Draw(im, mode=None): | 
|     try: | 
|         return im.getdraw(mode) | 
|     except AttributeError: | 
|         return ImageDraw(im, mode) | 
|   | 
| # experimental access to the outline API | 
| try: | 
|     Outline = Image.core.outline | 
| except: | 
|     Outline = None | 
|   | 
| ## | 
| # (Experimental) A more advanced 2D drawing interface for PIL images, | 
| # based on the WCK interface. | 
| # | 
| # @param im The image to draw in. | 
| # @param hints An optional list of hints. | 
| # @return A (drawing context, drawing resource factory) tuple. | 
|   | 
| def getdraw(im=None, hints=None): | 
|     # FIXME: this needs more work! | 
|     # FIXME: come up with a better 'hints' scheme. | 
|     handler = None | 
|     if not hints or "nicest" in hints: | 
|         try: | 
|             import _imagingagg | 
|             handler = _imagingagg | 
|         except ImportError: | 
|             pass | 
|     if handler is None: | 
|         import ImageDraw2 | 
|         handler = ImageDraw2 | 
|     if im: | 
|         im = handler.Draw(im) | 
|     return im, handler | 
|   | 
| ## | 
| # (experimental) Fills a bounded region with a given color. | 
| # | 
| # @param image Target image. | 
| # @param xy Seed position (a 2-item coordinate tuple). | 
| # @param value Fill color. | 
| # @param border Optional border value.  If given, the region consists of | 
| #     pixels with a color different from the border color.  If not given, | 
| #     the region consists of pixels having the same color as the seed | 
| #     pixel. | 
|   | 
| def floodfill(image, xy, value, border=None): | 
|     "Fill bounded region." | 
|     # based on an implementation by Eric S. Raymond | 
|     pixel = image.load() | 
|     x, y = xy | 
|     try: | 
|         background = pixel[x, y] | 
|         if background == value: | 
|             return # seed point already has fill color | 
|         pixel[x, y] = value | 
|     except IndexError: | 
|         return # seed point outside image | 
|     edge = [(x, y)] | 
|     if border is None: | 
|         while edge: | 
|             newedge = [] | 
|             for (x, y) in edge: | 
|                 for (s, t) in ((x+1, y), (x-1, y), (x, y+1), (x, y-1)): | 
|                     try: | 
|                         p = pixel[s, t] | 
|                     except IndexError: | 
|                         pass | 
|                     else: | 
|                         if p == background: | 
|                             pixel[s, t] = value | 
|                             newedge.append((s, t)) | 
|             edge = newedge | 
|     else: | 
|         while edge: | 
|             newedge = [] | 
|             for (x, y) in edge: | 
|                 for (s, t) in ((x+1, y), (x-1, y), (x, y+1), (x, y-1)): | 
|                     try: | 
|                         p = pixel[s, t] | 
|                     except IndexError: | 
|                         pass | 
|                     else: | 
|                         if p != value and p != border: | 
|                             pixel[s, t] = value | 
|                             newedge.append((s, t)) | 
|             edge = newedge |