あなたの質問から、あなたは GD に慣れていないようです。私の経験をいくつか共有します。これは少し話題から外れているかもしれませんが、あなたのような GD に慣れていない人には役立つと思います。
ステップ 1、ファイルを検証します。次の関数を使用して、$_FILES['image']['tmp_name']
ファイルが有効なファイルかどうかを確認します。
function getContentsFromImage($image) {
if (@is_file($image) == true) {
return file_get_contents($image);
} else {
throw new \Exception('Invalid image');
}
}
$contents = getContentsFromImage($_FILES['image']['tmp_name']);
ステップ 2、ファイル形式の取得ファイル (コンテンツ) のファイル形式を確認するには、finfo 拡張子を付けて次の関数を試してください。$_FILES["image"]["type"]
ファイル形式を確認するために使用しないのはなぜですか?ファイルの内容ではなくファイル拡張子のみをチェックするため、もともと world.png と呼ばれていたファイルの名前をworld.jpgに変更すると、$_FILES["image"]["type"]
png ではなく jpeg$_FILES["image"]["type"]
が返され、間違った結果が返される可能性があります。
function getFormatFromContents($contents) {
$finfo = new \finfo();
$mimetype = $finfo->buffer($contents, FILEINFO_MIME_TYPE);
switch ($mimetype) {
case 'image/jpeg':
return 'jpeg';
break;
case 'image/png':
return 'png';
break;
case 'image/gif':
return 'gif';
break;
default:
throw new \Exception('Unknown or unsupported image format');
}
}
$format = getFormatFromContents($contents);
Step.3, GD リソースを取得する 以前のコンテンツから GD リソースを取得します。
function getGDResourceFromContents($contents) {
$resource = @imagecreatefromstring($contents);
if ($resource == false) {
throw new \Exception('Cannot process image');
}
return $resource;
}
$resource = getGDResourceFromContents($contents);
ステップ 4、画像の寸法を取得する これで、次の簡単なコードで画像の寸法を取得できます。
$width = imagesx($resource);
$height = imagesy($resource);
それでは、元の画像から取得した変数を見てみましょう。
$contents, $format, $resource, $width, $height
OK, lets move on
ステップ 5、サイズ変更された画像の引数を計算するこのステップはあなたの質問に関連しています。次の関数の目的は、GD 関数imagecopyresampled()
のサイズ変更引数を取得することです。コードは少し長いですが、うまく機能し、3 つのオプションもあります: ストレッチ、縮小、および塗りつぶします。
Stretch : 出力画像の寸法は、設定した新しい寸法と同じです。高さと幅の比率を維持しません。
縮小: 出力画像の寸法は、指定した新しい寸法を超えず、画像の高さと幅の比率を維持します。
fill : 出力画像の寸法は、指定した新しい寸法と同じになり、必要に応じて画像をトリミングおよびサイズ変更し、画像の高さと幅の比率を維持します。このオプションは、質問で必要なものです。
function getResizeArgs($width, $height, $newwidth, $newheight, $option) {
if ($option === 'stretch') {
if ($width === $newwidth && $height === $newheight) {
return false;
}
$dst_w = $newwidth;
$dst_h = $newheight;
$src_w = $width;
$src_h = $height;
$src_x = 0;
$src_y = 0;
} else if ($option === 'shrink') {
if ($width <= $newwidth && $height <= $newheight) {
return false;
} else if ($width / $height >= $newwidth / $newheight) {
$dst_w = $newwidth;
$dst_h = (int) round(($newwidth * $height) / $width);
} else {
$dst_w = (int) round(($newheight * $width) / $height);
$dst_h = $newheight;
}
$src_x = 0;
$src_y = 0;
$src_w = $width;
$src_h = $height;
} else if ($option === 'fill') {
if ($width === $newwidth && $height === $newheight) {
return false;
}
if ($width / $height >= $newwidth / $newheight) {
$src_w = (int) round(($newwidth * $height) / $newheight);
$src_h = $height;
$src_x = (int) round(($width - $src_w) / 2);
$src_y = 0;
} else {
$src_w = $width;
$src_h = (int) round(($width * $newheight) / $newwidth);
$src_x = 0;
$src_y = (int) round(($height - $src_h) / 2);
}
$dst_w = $newwidth;
$dst_h = $newheight;
}
if ($src_w < 1 || $src_h < 1) {
throw new \Exception('Image width or height is too small');
}
return array(
'dst_x' => 0,
'dst_y' => 0,
'src_x' => $src_x,
'src_y' => $src_y,
'dst_w' => $dst_w,
'dst_h' => $dst_h,
'src_w' => $src_w,
'src_h' => $src_h
);
}
$args = getResizeArgs($width, $height, 150, 170, 'fill');
ステップ 6、画像のサイズ変更上記で取得した$args
、$width
、$height
、および $resource を次の関数に使用し、サイズ変更された画像の新しいリソースを取得します。$format
function runResize($width, $height, $format, $resource, $args) {
if ($args === false) {
return; //if $args equal to false, this means no resize occurs;
}
$newimage = imagecreatetruecolor($args['dst_w'], $args['dst_h']);
if ($format === 'png') {
imagealphablending($newimage, false);
imagesavealpha($newimage, true);
$transparentindex = imagecolorallocatealpha($newimage, 255, 255, 255, 127);
imagefill($newimage, 0, 0, $transparentindex);
} else if ($format === 'gif') {
$transparentindex = imagecolorallocatealpha($newimage, 255, 255, 255, 127);
imagefill($newimage, 0, 0, $transparentindex);
imagecolortransparent($newimage, $transparentindex);
}
imagecopyresampled($newimage, $resource, $args['dst_x'], $args['dst_y'], $args['src_x'], $args['src_y'], $args['dst_w'], $args['dst_h'], $args['src_w'], $args['src_h']);
imagedestroy($resource);
return $newimage;
}
$newresource = runResize($width, $height, $format, $resource, $args);
ステップ 7、新しいコンテンツを取得する。次の関数を使用して、新しい GD リソースからコンテンツを取得します。
function getContentsFromGDResource($resource, $format) {
ob_start();
switch ($format) {
case 'gif':
imagegif($resource);
break;
case 'jpeg':
imagejpeg($resource, NULL, 100);
break;
case 'png':
imagepng($resource, NULL, 9);
}
$contents = ob_get_contents();
ob_end_clean();
return $contents;
}
$newcontents = getContentsFromGDResource($newresource, $format);
ステップ 8 拡張子を取得します。次の関数を使用して、画像形式から拡張子を取得します (注: 画像形式は画像拡張子とは異なります)。
function getExtensionFromFormat($format) {
switch ($format) {
case 'gif':
return 'gif';
break;
case 'jpeg':
return 'jpg';
break;
case 'png':
return 'png';
}
}
$extension = getExtensionFromFormat($format);
ステップ 9 画像の保存mike という名前のユーザーがいる場合、次のようにすると、この php スクリプトと同じフォルダーに保存されます。
$user_name = 'mike';
$filename = $user_name . '.' . $extension;
file_put_contents($filename, $newcontents);
Step 10 destroy resource GDリソースのdestroyを忘れずに!
imagedestroy($newresource);
または、すべてのコードをクラスに記述して、次を使用するだけです。
public function __destruct() {
@imagedestroy($this->resource);
}
チップ
ユーザーがアップロードするファイル形式を変換しないことをお勧めします。多くの問題が発生します。