在PHP中合并透明图像

时间:2012-07-02 10:27:54

标签: php image gd

我正在尝试将部分透明的PNG层叠在另一个不透明的PNG上。

有很多关于如何在这个网站和网络上做到这一点的例子,但是对于我尝试的每个版本,我似乎无法保持前景图像的透明度。

目前代码如下:

 $image = imagecreatefrompng($_GET['fg']);
 $frame = imagecreatefrompng($_GET['bg']);

 imagealphablending($frame,true);
 imagecopymerge($image, $frame, 0, 0, 0, 0, 0, 100, 100);

 # Save the image to a file
 $output_file = 'preview-' . time() . '.png';

 imagepng( $image, $_SERVER['DOCUMENT_ROOT'] . '/share/' . $output_file );

生成由前景图像组成的图像,透明部分为白色(或黑色)。

我也试过这个,如图像缩略图生成器TimThumb所示,它产生相同的输出:

$canvas= imagecreatefrompng($_GET['bg']);

$overlay_gd_image = imagecreatefrompng( $_GET['fg'] );
$overlay_width = imagesx( $overlay_gd_image );
$overlay_height = imagesy( $overlay_gd_image );
imagealphablending($canvas, true );
imagecopy( $canvas, $overlay_gd_image, 0, 0, 0, 0, $overlay_width, $overlay_height);
imagealphablending($canvas, false );
imagesavealpha($canvas , true); 

imagepng($canvas, 'new.png');

如果有人能够解决这个问题的话,我已经没有什么东西可以尝试了。如果有人能说清楚这个问题的话。

2 个答案:

答案 0 :(得分:1)

本地 imagecopymerge 在透明图像方面效果不是很好。如果你想将两个PNG合并在一起并且没有让alpha通道搞砸了,manual page上的这个功能对我来说就像一个魅力,并且实际上保留了透明度。我一直在将PNG与这个功能直接合并,并没有遇到任何问题所以我认为它可以解决你的问题。 (由“rodrigo dot polo at gmail dot com”发布,所以所有信用都归他/她所有。)

<?php 
/** 
 * PNG ALPHA CHANNEL SUPPORT for imagecopymerge(); 
 * This is a function like imagecopymerge but it handle alpha channel well!!! 
 **/ 

// A fix to get a function like imagecopymerge WITH ALPHA SUPPORT 
// Main script by aiden dot mail at freemail dot hu 
// Transformed to imagecopymerge_alpha() by rodrigo dot polo at gmail dot com 
function imagecopymerge_alpha($dst_im, $src_im, $dst_x, $dst_y, $src_x, $src_y, $src_w, $src_h, $pct){ 
    if(!isset($pct)){ 
        return false; 
    } 
    $pct /= 100; 
    // Get image width and height 
    $w = imagesx( $src_im ); 
    $h = imagesy( $src_im ); 
    // Turn alpha blending off 
    imagealphablending( $src_im, false ); 
    // Find the most opaque pixel in the image (the one with the smallest alpha value) 
    $minalpha = 127; 
    for( $x = 0; $x < $w; $x++ ) 
    for( $y = 0; $y < $h; $y++ ){ 
        $alpha = ( imagecolorat( $src_im, $x, $y ) >> 24 ) & 0xFF; 
        if( $alpha < $minalpha ){ 
            $minalpha = $alpha; 
        } 
    } 
    //loop through image pixels and modify alpha for each 
    for( $x = 0; $x < $w; $x++ ){ 
        for( $y = 0; $y < $h; $y++ ){ 
            //get current alpha value (represents the TANSPARENCY!) 
            $colorxy = imagecolorat( $src_im, $x, $y ); 
            $alpha = ( $colorxy >> 24 ) & 0xFF; 
            //calculate new alpha 
            if( $minalpha !== 127 ){ 
                $alpha = 127 + 127 * $pct * ( $alpha - 127 ) / ( 127 - $minalpha ); 
            } else { 
                $alpha += 127 * $pct; 
            } 
            //get the color index with new alpha 
            $alphacolorxy = imagecolorallocatealpha( $src_im, ( $colorxy >> 16 ) & 0xFF, ( $colorxy >> 8 ) & 0xFF, $colorxy & 0xFF, $alpha ); 
            //set pixel with the new color + opacity 
            if( !imagesetpixel( $src_im, $x, $y, $alphacolorxy ) ){ 
                return false; 
            } 
        } 
    } 
    // The image copy 
    imagecopy($dst_im, $src_im, $dst_x, $dst_y, $src_x, $src_y, $src_w, $src_h); 
} 

// USAGE EXAMPLE: 
$img_a = imagecreatefrompng('image1.png'); 
$img_b = imagecreatefrompng('wm2.png'); 

// SAME COMMANDS: 
imagecopymerge_alpha($img_a, $img_b, 10, 10, 0, 0, imagesx($img_b), imagesy($img_b),50); 

// OUTPUT IMAGE: 
header("Content-Type: image/png"); 
imagesavealpha($img_a, true); 
imagepng($img_a, NULL); 
?>

如果您使用该示例,您将看到$ img_a将低于$ img_b而不是有一个奇怪的黑盒子,$ img_b的透明度将被保留。

答案 1 :(得分:0)

我有完全相同的问题。我的解决方案是使用imagecopyresampled而不是imagecopymerge。这将保留两个图像的透明度。

要注意传递的参数略有不同