With the Python Imaging Library (PIL), how does one compose an image with an alpha channel over another image?

后端 未结 3 555
执笔经年
执笔经年 2020-11-28 06:20

I have two images, both with alpha channels. I want to put one image over the other, resulting in a new image with an alpha channel, just as would occur if they were render

相关标签:
3条回答
  • 2020-11-28 06:45

    Pillow 2.0 now contains an alpha_composite function that does this.

    img3 = Image.alpha_composite(img1, img2)
    
    0 讨论(0)
  • 2020-11-28 06:48

    This appears to do the trick:

    from PIL import Image
    bottom = Image.open("a.png")
    top = Image.open("b.png")
    
    r, g, b, a = top.split()
    top = Image.merge("RGB", (r, g, b))
    mask = Image.merge("L", (a,))
    bottom.paste(top, (0, 0), mask)
    bottom.save("over.png")
    
    0 讨论(0)
  • 2020-11-28 07:01

    I couldn't find an alpha composite function in PIL, so here is my attempt at implementing it with numpy:

    import numpy as np
    from PIL import Image
    
    def alpha_composite(src, dst):
        '''
        Return the alpha composite of src and dst.
    
        Parameters:
        src -- PIL RGBA Image object
        dst -- PIL RGBA Image object
    
        The algorithm comes from http://en.wikipedia.org/wiki/Alpha_compositing
        '''
        # http://stackoverflow.com/a/3375291/190597
        # http://stackoverflow.com/a/9166671/190597
        src = np.asarray(src)
        dst = np.asarray(dst)
        out = np.empty(src.shape, dtype = 'float')
        alpha = np.index_exp[:, :, 3:]
        rgb = np.index_exp[:, :, :3]
        src_a = src[alpha]/255.0
        dst_a = dst[alpha]/255.0
        out[alpha] = src_a+dst_a*(1-src_a)
        old_setting = np.seterr(invalid = 'ignore')
        out[rgb] = (src[rgb]*src_a + dst[rgb]*dst_a*(1-src_a))/out[alpha]
        np.seterr(**old_setting)    
        out[alpha] *= 255
        np.clip(out,0,255)
        # astype('uint8') maps np.nan (and np.inf) to 0
        out = out.astype('uint8')
        out = Image.fromarray(out, 'RGBA')
        return out
    

    For example given these two images,

    img1 = Image.new('RGBA', size = (100, 100), color = (255, 0, 0, 255))
    draw = ImageDraw.Draw(img1)
    draw.rectangle((33, 0, 66, 100), fill = (255, 0, 0, 128))
    draw.rectangle((67, 0, 100, 100), fill = (255, 0, 0, 0))
    img1.save('/tmp/img1.png')
    

    enter image description here

    img2 = Image.new('RGBA', size = (100, 100), color = (0, 255, 0, 255))
    draw = ImageDraw.Draw(img2)
    draw.rectangle((0, 33, 100, 66), fill = (0, 255, 0, 128))
    draw.rectangle((0, 67, 100, 100), fill = (0, 255, 0, 0))
    img2.save('/tmp/img2.png')
    

    enter image description here

    alpha_composite produces:

    img3 = alpha_composite(img1, img2)
    img3.save('/tmp/img3.png')
    

    enter image description here

    0 讨论(0)
提交回复
热议问题