调整图像大小而不失真保持纵横比,然后使用WideImage裁剪过量

时间:2011-06-15 09:56:45

标签: php image-resizing

我正在处理的网站上有一个区域,它会显示从外部源提取的用户个人资料图片(因此无法控制其原始大小)。

我要做的是拍摄一张图片(在此示例中为1000px x 800px并将其调整为200px x 150px。显然,这会产生宽高比差异。

我想要做的是调整原始图像的大小而不会失真,在这种情况下会产生200px x 160px图像。我当时想做的是从边缘裁剪任何多余的部分以产生正确的图像尺寸。因此,在这种情况下,从图像的顶部和底部裁剪5px,最终生成200px x 150px

我目前有WideImage库,想要使用它。我在SO上看到了几个类似的问题,但是我没想到的确实如此。

5 个答案:

答案 0 :(得分:27)

您可以尝试:

$img->resize(200, 150, 'outside')->crop('center', 'middle', 200, 150);

有些用户发布他们的计算版本...这也是我的版本:

$sourceWidth = 1000;
$sourceHeight = 250;

$targetWidth = 200;
$targetHeight = 150;

$sourceRatio = $sourceWidth / $sourceHeight;
$targetRatio = $targetWidth / $targetHeight;

if ( $sourceRatio < $targetRatio ) {
    $scale = $sourceWidth / $targetWidth;
} else {
    $scale = $sourceHeight / $targetHeight;
}

$resizeWidth = (int)($sourceWidth / $scale);
$resizeHeight = (int)($sourceHeight / $scale);

$cropLeft = (int)(($resizeWidth - $targetWidth) / 2);
$cropTop = (int)(($resizeHeight - $targetHeight) / 2);

var_dump($resizeWidth, $resizeHeight, $cropLeft, $cropTop);

答案 1 :(得分:8)

我尝试了所有的解决方案,每次都想出奇怪的数字。所以,这是我进行计算的方式。

基本比例公式:a / b = x / y - &gt; ay = bx然后求解未知值。 所以,让我们把它放到代码中。

此功能假设您已将图像打开为变量。如果您通过路径,则必须在函数中打开图像,进行数学运算,将其删除,返回值,然后在调整大小时再次打开图像,效率低...... < / p>

function resize_values($image){
    #for the purpose of this example, we'll set this here
    #to make this function more powerful, i'd pass these 
    #to the function on the fly
    $maxWidth = 200; 
    $maxHeight = 200;

    #get the size of the image you're resizing.
    $origHeight = imagesy($image);
    $origWidth = imagesx($image);

    #check for longest side, we'll be seeing that to the max value above
    if($origHeight > $origWidth){ #if height is more than width
         $newWidth = ($maxHeight * $origWidth) / $origHeight;

         $retval = array(width => $newWidth, height => $maxHeight);
    }else{
         $newHeight= ($maxWidth * $origHeight) / $origWidth;

         $retval = array(width => $maxWidth, height => $newHeight);
    }
return $retval;
}

上面的函数显然会返回一个数组。您可以将公式应用到您正在处理的任何脚本中。事实证明这是适合这项工作的公式。所以......取决于你是否使用它。 LATER!

答案 2 :(得分:2)

答案#1(已编辑)

看了thisthis之后,你可能需要这样做:

$img->resize(200, 150, 'outside')->crop("center", "center", 200, 150);

在调整大小时,它会调整图像的大小,使其完全适合框(框= 200x150)或其中一个尺寸适合,而另一个超出框。裁剪时,将剪裁在框外出血的图像部分。指定center智能坐标意味着将删除顶部+底部或左侧+右侧部分。

回答#2

如果您在计算要裁剪的内容时遇到问题,请尝试以下操作:

<?php
$target_wide = 200;
$target_tall = 150;

$test_case = array(
    array(1000, 800),
    array(800, 1000),
    array(1000, 750), // perfect fit
    array(750, 1000)
);

foreach($test_case as $test) {
    list(
        $source_wide,
        $source_tall
    ) = $test;
    $source_aspect_ratio = $source_wide / $source_tall;
    $target_aspect_ratio = $target_wide / $target_tall;
    if ($source_aspect_ratio > $target_aspect_ratio)
    {
        $output_tall = $target_tall;
        $output_wide = (int) ($target_tall * $source_aspect_ratio);
    }
    else
    {
        $output_wide = $target_wide;
        $output_tall = (int) ($target_wide / $source_aspect_ratio);
    }
    $output_crop_hori = (int) (($output_wide - $target_wide) / 2);
    $output_crop_vert = (int) (($output_tall - $target_tall) / 2);
    var_dump($source_wide, $source_tall, $output_wide, $output_tall, $output_crop_hori, $output_crop_vert);
    echo PHP_EOL;
}

输出:

int(1000)
int(800)
int(200)
int(160)
int(0)
int(5)

int(800)
int(1000)
int(200)
int(250)
int(0)
int(50)

int(1000)
int(750)
int(200)
int(150)
int(0)
int(0)

int(750)
int(1000)
int(200)
int(266)
int(0)
int(58)

答案 3 :(得分:2)

//Following code for keeping aspect ratio

$srcWidth=1000;
$srcHeight=800;

$targetWidth=200;
$targetHeight=150;

$diff=0;
//get difference

if( $srcWidth > $srcHeight ) // 1000 > 800
{
    $diff = $srcWidth - $srcHeight; // 1000 - 800 = 200
    $diff = ($diff / $srcWidth) * 100; // ( 200 / 1000 ) * 100 = 20
}
else
{
    $diff = $srcHeight - $srcWidth;
    $diff = ($diff / $srcHeight)*100;
}

if( $targetWidth > $targetHeight)
{
    $targetHeight = (( $targetHeight * $diff ) / 100) - $targetWidth; // (( 200 * 20 ) /  100) - 200 = 160
}
else
{
    $targetWidth = (( $targetWidth * $diff ) / 100) - $targetHeight;
}

答案 4 :(得分:1)

这就是你如何计算作物的大小:

$src_width = 1000;
$src_height = 500;
$src_ratio = $src_width/$src_height;

$width = 200;
$height = 150;
$ratio = $width/$height;

$crop_height = 0;
$crop_width = 0;

if ($src_height > $src_width)
{
    $new_height = $width/$src_ratio;
    $crop_height = $new_height-$height;
}
else
{
    $new_width = $height*$src_ratio;
    $crop_width = $new_width-$width;
}

print 'Crop height: '.$crop_height.PHP_EOL
      .'Crop width: '.$crop_width;