# 
 | 
# The Python Imaging Library. 
 | 
# $Id$ 
 | 
# 
 | 
# standard channel operations 
 | 
# 
 | 
# History: 
 | 
# 1996-03-24 fl   Created 
 | 
# 1996-08-13 fl   Added logical operations (for "1" images) 
 | 
# 2000-10-12 fl   Added offset method (from Image.py) 
 | 
# 
 | 
# Copyright (c) 1997-2000 by Secret Labs AB 
 | 
# Copyright (c) 1996-2000 by Fredrik Lundh 
 | 
# 
 | 
# See the README file for information on usage and redistribution. 
 | 
# 
 | 
  
 | 
import Image 
 | 
  
 | 
## 
 | 
# The <b>ImageChops</b> module contains a number of arithmetical image 
 | 
# operations, called <i>channel operations</i> ("chops"). These can be 
 | 
# used for various purposes, including special effects, image 
 | 
# compositions, algorithmic painting, and more. 
 | 
# <p> 
 | 
# At this time, channel operations are only implemented for 8-bit 
 | 
# images (e.g. "L" and "RGB"). 
 | 
# <p> 
 | 
# Most channel operations take one or two image arguments and returns 
 | 
# a new image.  Unless otherwise noted, the result of a channel 
 | 
# operation is always clipped to the range 0 to MAX (which is 255 for 
 | 
# all modes supported by the operations in this module). 
 | 
## 
 | 
  
 | 
## 
 | 
# Return an image with the same size as the given image, but filled 
 | 
# with the given pixel value. 
 | 
# 
 | 
# @param image Reference image. 
 | 
# @param value Pixel value. 
 | 
# @return An image object. 
 | 
  
 | 
def constant(image, value): 
 | 
    "Fill a channel with a given grey level" 
 | 
  
 | 
    return Image.new("L", image.size, value) 
 | 
  
 | 
## 
 | 
# Copy image. 
 | 
# 
 | 
# @param image Source image. 
 | 
# @return A copy of the source image. 
 | 
  
 | 
def duplicate(image): 
 | 
    "Create a copy of a channel" 
 | 
  
 | 
    return image.copy() 
 | 
  
 | 
## 
 | 
# Inverts an image 
 | 
# (MAX - image). 
 | 
# 
 | 
# @param image Source image. 
 | 
# @return An image object. 
 | 
  
 | 
def invert(image): 
 | 
    "Invert a channel" 
 | 
  
 | 
    image.load() 
 | 
    return image._new(image.im.chop_invert()) 
 | 
  
 | 
## 
 | 
# Compare images, and return lighter pixel value 
 | 
# (max(image1, image2)). 
 | 
# <p> 
 | 
# Compares the two images, pixel by pixel, and returns a new image 
 | 
# containing the lighter values. 
 | 
# 
 | 
# @param image1 First image. 
 | 
# @param image1 Second image. 
 | 
# @return An image object. 
 | 
  
 | 
def lighter(image1, image2): 
 | 
    "Select the lighter pixels from each image" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_lighter(image2.im)) 
 | 
  
 | 
## 
 | 
# Compare images, and return darker pixel value 
 | 
# (min(image1, image2)). 
 | 
# <p> 
 | 
# Compares the two images, pixel by pixel, and returns a new image 
 | 
# containing the darker values. 
 | 
# 
 | 
# @param image1 First image. 
 | 
# @param image1 Second image. 
 | 
# @return An image object. 
 | 
  
 | 
def darker(image1, image2): 
 | 
    "Select the darker pixels from each image" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_darker(image2.im)) 
 | 
  
 | 
## 
 | 
# Calculate absolute difference 
 | 
# (abs(image1 - image2)). 
 | 
# <p> 
 | 
# Returns the absolute value of the difference between the two images. 
 | 
# 
 | 
# @param image1 First image. 
 | 
# @param image1 Second image. 
 | 
# @return An image object. 
 | 
  
 | 
def difference(image1, image2): 
 | 
    "Subtract one image from another" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_difference(image2.im)) 
 | 
  
 | 
## 
 | 
# Superimpose positive images 
 | 
# (image1 * image2 / MAX). 
 | 
# <p> 
 | 
# Superimposes two images on top of each other. If you multiply an 
 | 
# image with a solid black image, the result is black. If you multiply 
 | 
# with a solid white image, the image is unaffected. 
 | 
# 
 | 
# @param image1 First image. 
 | 
# @param image1 Second image. 
 | 
# @return An image object. 
 | 
  
 | 
def multiply(image1, image2): 
 | 
    "Superimpose two positive images" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_multiply(image2.im)) 
 | 
  
 | 
## 
 | 
# Superimpose negative images 
 | 
# (MAX - ((MAX - image1) * (MAX - image2) / MAX)). 
 | 
# <p> 
 | 
# Superimposes two inverted images on top of each other. 
 | 
# 
 | 
# @param image1 First image. 
 | 
# @param image1 Second image. 
 | 
# @return An image object. 
 | 
  
 | 
def screen(image1, image2): 
 | 
    "Superimpose two negative images" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_screen(image2.im)) 
 | 
  
 | 
## 
 | 
# Add images 
 | 
# ((image1 + image2) / scale + offset). 
 | 
# <p> 
 | 
# Adds two images, dividing the result by scale and adding the 
 | 
# offset. If omitted, scale defaults to 1.0, and offset to 0.0. 
 | 
# 
 | 
# @param image1 First image. 
 | 
# @param image1 Second image. 
 | 
# @return An image object. 
 | 
  
 | 
def add(image1, image2, scale=1.0, offset=0): 
 | 
    "Add two images" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_add(image2.im, scale, offset)) 
 | 
  
 | 
## 
 | 
# Subtract images 
 | 
# ((image1 - image2) / scale + offset). 
 | 
# <p> 
 | 
# Subtracts two images, dividing the result by scale and adding the 
 | 
# offset. If omitted, scale defaults to 1.0, and offset to 0.0. 
 | 
# 
 | 
# @param image1 First image. 
 | 
# @param image1 Second image. 
 | 
# @return An image object. 
 | 
  
 | 
def subtract(image1, image2, scale=1.0, offset=0): 
 | 
    "Subtract two images" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_subtract(image2.im, scale, offset)) 
 | 
  
 | 
## 
 | 
# Add images without clipping 
 | 
# ((image1 + image2) % MAX). 
 | 
# <p> 
 | 
# Adds two images, without clipping the result. 
 | 
# 
 | 
# @param image1 First image. 
 | 
# @param image1 Second image. 
 | 
# @return An image object. 
 | 
  
 | 
def add_modulo(image1, image2): 
 | 
    "Add two images without clipping" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_add_modulo(image2.im)) 
 | 
  
 | 
## 
 | 
# Subtract images without clipping 
 | 
# ((image1 - image2) % MAX). 
 | 
# <p> 
 | 
# Subtracts two images, without clipping the result. 
 | 
# 
 | 
# @param image1 First image. 
 | 
# @param image1 Second image. 
 | 
# @return An image object. 
 | 
  
 | 
def subtract_modulo(image1, image2): 
 | 
    "Subtract two images without clipping" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_subtract_modulo(image2.im)) 
 | 
  
 | 
## 
 | 
# Logical AND 
 | 
# (image1 and image2). 
 | 
  
 | 
def logical_and(image1, image2): 
 | 
    "Logical and between two images" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_and(image2.im)) 
 | 
  
 | 
## 
 | 
# Logical OR 
 | 
# (image1 or image2). 
 | 
  
 | 
def logical_or(image1, image2): 
 | 
    "Logical or between two images" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_or(image2.im)) 
 | 
  
 | 
## 
 | 
# Logical XOR 
 | 
# (image1 xor image2). 
 | 
  
 | 
def logical_xor(image1, image2): 
 | 
    "Logical xor between two images" 
 | 
  
 | 
    image1.load() 
 | 
    image2.load() 
 | 
    return image1._new(image1.im.chop_xor(image2.im)) 
 | 
  
 | 
## 
 | 
# Blend images using constant transparency weight. 
 | 
# <p> 
 | 
# Same as the <b>blend</b> function in the <b>Image</b> module. 
 | 
  
 | 
def blend(image1, image2, alpha): 
 | 
    "Blend two images using a constant transparency weight" 
 | 
  
 | 
    return Image.blend(image1, image2, alpha) 
 | 
  
 | 
## 
 | 
# Create composite using transparency mask. 
 | 
# <p> 
 | 
# Same as the <b>composite</b> function in the <b>Image</b> module. 
 | 
  
 | 
def composite(image1, image2, mask): 
 | 
    "Create composite image by blending images using a transparency mask" 
 | 
  
 | 
    return Image.composite(image1, image2, mask) 
 | 
  
 | 
## 
 | 
# Offset image data. 
 | 
# <p> 
 | 
# Returns a copy of the image where data has been offset by the given 
 | 
# distances.  Data wraps around the edges.  If yoffset is omitted, it 
 | 
# is assumed to be equal to xoffset. 
 | 
# 
 | 
# @param image Source image. 
 | 
# @param xoffset The horizontal distance. 
 | 
# @param yoffset The vertical distance.  If omitted, both 
 | 
#    distances are set to the same value. 
 | 
# @return An Image object. 
 | 
  
 | 
def offset(image, xoffset, yoffset=None): 
 | 
    "Offset image in horizontal and/or vertical direction" 
 | 
    if yoffset is None: 
 | 
        yoffset = xoffset 
 | 
    image.load() 
 | 
    return image._new(image.im.offset(xoffset, yoffset)) 
 |