证件照接口 #
接口详情 #
地址:https://www.clipimg.com/api/idphoto/make
请求方式:POST
数据类型:JSON
响应类型:JSON
计费:此接口不产生费用
接口描述:此接口为证件照制作接口,调用不产生费用,结果保留1小时,1小时后自动删除制作结果图片,请及时保存。重复下载同一张照片不会重复扣费。
参数[Header]
参数名 | 说明 | 是否必选 | 类型 |
X-API-Key | 秘钥,点此获取 | 必选 | string |
Content-Type | 请求数据类型 | 必选 | application/json |
参数[Body]:
参数名 | 说明 | 是否必选 | 类型 |
file | 图片的base64格式,不包含base64的头部数据 | 必选 | string |
spec_id | 规格id | 可选,width、height不传值时必传 | int |
width | 宽度px | 可选,spec_id不传值时必传 | int |
height | 高度px | 可选,spec_id不传值时必传 | int |
fair_level | 美颜级别 | 可选,0-1之间的小数,为0时不美颜 | float |
color | 背景色 | 可选,默认值蓝红白三种颜色 | list |
facepose | 人脸是否正对摄像头误差范围,检测人脸pitch、yaw、roll三项误差值,示例 | 可选,越小越严格,越大容错率越高,默认:20 | float |
eyeskew | 双眼倾斜误差范围 | 可选,越小越严格,越大容错率越高,默认:20 | float |
face_ratio_min | 头部最小比例 | 可选,默认0.55 | float |
face_ratio | 头部最大比例 | 可选,默认0.8 | float |
dpi | 返回的证件照分辨率,如果不设置该值,服务端会给出默认配置,一般返回300dpi的图片 | 可选 | int |
head_top_gap_ratio | 头顶发迹离上边缘距离 | 可选,默认0.08 | float |
eyes_to_botttom_ratio_min | 眼睛离下边缘最小距离 | 可选,默认0.46 | float |
enhance_image_quality | 是否对图像做清晰化处理 | 可选,默认为1,1表示是,0表示否 | int |
head_region | 头部的计算区域,第一种计算区域为整个头部(含头发、耳朵),示例;第二种计算区域为脸颊,示例; | 可选,默认0,0表示头部,1表示脸颊 | int |
align_face | 在eyeskew参数误差范围内是否进行人脸自动矫正 | 可选,默认0,0表示不矫正,1表示矫正 | int |
glasses_glare | 是否进行眼镜炫光检测 | 可选,默认0,0表示不检测,1表示检测 | int |
file_format | 文件生成类型,支持jpg,png。默认png | 可选,默认0,0表示png,1表示jpg | int |
head_height_ratio_max | 头部高度占比最大值 | 可选,默认1,示例值:0.7 | float |
head_height_ratio_min | 头部高度占比最小值 | 可选,默认0.01,示例值:0.5 | float |
eye_distance_ratio_max | 眼间距宽度占比最大值 | 可选,默认1,示例值:0.4 | float |
eye_distance_ratio_min | 眼间距宽度占比最小值 | 可选,默认0.01,示例值:0.3 | float |
color结构
参数名 | 说明 | 类型 |
name | 背景色名称,请使用纯字母组成此参数 | string |
start_color | 背景色值 | strng |
enc_color | 渐变色色值,可选,不传值时为纯色背景 | string |
返回:
参数 | 说明 | 示例 | 类型 |
code | 状态码 | 0 | int |
msg | 错误详情 | 成功 | string |
data | 结果 |
code取值
0 | 请求成功 |
402 | 点数已用完 |
407 | 未检测到人脸 |
408 | 包含多人 |
412 | 双眼倾斜,超过误差范围 |
413 | 人脸倾斜,超过误差范围 |
data结构体
参数名 | 说明 | 示例值 | 类型 |
image_id | 上传图片的标示 | d00a42c…0dc18946 | string |
filenames | 证件照列表 | [{‘preview_url’: ‘https://…’, ‘preview_img_name’: ‘d00a4…_white_preview’, ‘download_url’: ‘https://…’, ‘img_name’: ‘d00a4…_white’, ‘color_name’: ‘white’,’bg_color’: ‘#FFFFFF’}, …] | list |
filenames结构
参数名 | 说明 | 示例值 | 类型 |
preview_url | 预览图片地址 | https://… | string |
preview_img_name | 预览图名称,用于再次下载 | d00a4…_white | string |
download_url | 原图下载地址 | https://… | string |
img_name | 证件照名称,用于再次下载、换装 | d00a4…_white | string |
color_name | 背景色名称 | white | string |
bg_color | 背景色色值 | #FFFFFF | string |
print_img_name | 打印排版照原图名称 | d00a4…_print_white | string |
preview_print_img_name | 打印排版照预览图名称 | d00a4…_print_white | string |
print_url | 排版照原图下载地址 | https://… | string |
preview_print_url | 排版照预览图下载地址 | https://… | string |
调用示例 #
- PHP
- PYTHON
- JAVA
// curl示例
$base64_img = base64_encode(file_get_contents("idphoto.jpg"));//修改为您的图片地址,转换成base64格式
$post_data = array('file' => $base64_img,'width'=>413,'height'=>579,'color'=>array(array("name"=>"blue","start_color"=>"#FFFFFF")));
$data_string = json_encode($post_data,JSON_UNESCAPED_UNICODE);
$curl_con = curl_init();
curl_setopt($curl_con, CURLOPT_URL,'https://www.clipimg.com/api/idphoto/make');
curl_setopt($curl_con, CURLOPT_HEADER, true);
curl_setopt($curl_con, CURLOPT_POST, true);
curl_setopt($curl_con, CURLOPT_RETURNTRANSFER, TRUE);
curl_setopt($curl_con, CURLOPT_CONNECTTIMEOUT, 5);
curl_setopt($curl_con, CURLOPT_HTTPHEADER, array(
'Content-Type: application/json',
'X-API-KEY: apikey',
'Content-Length: ' . strlen($data_string))
);
curl_setopt($curl_con, CURLOPT_POSTFIELDS, $data_string);
$res = curl_exec($curl_con);
$status = curl_getinfo($curl_con);
curl_close($curl_con);
if (isset($status['http_code']) && $status['http_code'] == 200) {
print($res);
return $res;
} else {
return FALSE;
}
// GuzzleHttp示例
/**
* 制作证件照
* $img 上传的人像图片本地地址
* 参数说明参考文档https://clipimg.com/wp/api-doc/
*/
public function IdphotoMake($img,$color,$width,$height,$fair_level,$dpi){
$client = new Client();
$base64_img = base64_encode(file_get_contents($img));//转换成base64格式
$response = $client->request('POST', $this->api_url.$this->idphoto, [
'headers' => [
'X-API-Key' => $this->apiKey,
'Content-Type' => 'application/json'
],
'body'=>json_encode([
"file"=>$base64_img,
"width"=>$width,
'height'=>$height,
'dpi'=>$dpi,
"color"=>$color,
'fair_level'=>$fair_level]
)
]);
$json = json_decode($response->getBody()->getContents());
return $json;
}
# /**
# * 制作证件照
# * img 上传的人像图片本地地址
# * 参数说明参考文档https://clipimg.com/wp/api-doc/
# */
def IdphotoMake(self,img,spec_id,width,height,fair_level,color,
face_ratio_min,face_ratio,facepose,eyeskew,dpi,head_top_gap_ratio,
eyes_to_botttom_ratio_min,enhance_image_quality,head_region,align_face,glasses_glare,
file_format,head_height_ratio_max,head_height_ratio_min,
eye_distance_ratio_min,eye_distance_ratio_max):
with open(img, 'rb') as f:
pic = f.read()
base64_img = base64.b64encode(pic).decode()
response = requests.post(self.api_url + self.idphoto,
headers = {
'X-API-Key' : self.apiKey,
'Content-Type' : 'application/json'
},
data=json.dumps({"file":base64_img,
"spec_id":spec_id,"width":width,"height":height,
"fair_level":fair_level,"color":color,"face_ratio_min":face_ratio_min,"face_ratio":face_ratio,
"facepose":facepose,"eyeskew":eyeskew,"dpi":dpi,"head_top_gap_ratio":head_top_gap_ratio,
"eyes_to_botttom_ratio_min":eyes_to_botttom_ratio_min,"enhance_image_quality":enhance_image_quality,
"head_region":head_region,"align_face":align_face,'glasses_glare':glasses_glare,
"file_format":file_format,
'head_height_ratio_max':head_height_ratio_max,'head_height_ratio_min':head_height_ratio_min,
'eye_distance_ratio_min':eye_distance_ratio_min,'eye_distance_ratio_max':eye_distance_ratio_max
})
)
result = response.json()
return result
//证件照制作
public JSONObject IdphotoMake(String img,List<Map> colors,int width,int height,float fair_level,int dpi) throws IOException{
HashMap<String,Object> data = new LinkedHashMap<>();
String base64Str = fileToBase64(new File(img));
// colors组装示例
// HashMap<String,Object> color = new LinkedHashMap<>();
// color.put("name","blue");
// color.put("start_color","#FFFFFF");
// List<Map> colors = new LinkedList<Map>();
// colors.add(color);
data.put("file",base64Str);
data.put("color",colors);
data.put("dpi",dpi);
data.put("width",width);
data.put("height",height);
data.put("fair_level",fair_level);
StringEntity entity = new StringEntity(JSON.toJSONString(data), ContentType.APPLICATION_JSON, "utf-8", false);//解决中文乱码问题
Response response;
try {
response = Request.post("https://www.clipimg.com/api/idphoto/make")
.addHeader("X-API-Key", this.apiKey)
.body(entity).execute();
return JSON.parseObject(response.returnContent().asString());
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return null;
}
证件照预览 #
地址:https://www.clipimg.com/api/idphoto/preview/预览图片名称
请求方式:GET
数据类型:application/x-www-form-urlencoded
响应类型:Image
计费:此接口不产生费用
参数:
参数名 | 说明 | 是否必选 | 类型 |
text | 给证件照增加自定义文字,url拼接字符串形式拼接, eg:https://www.clipimg.com/api/idphoto/preview/d00a4…_white?text=这是自定义文字 | 可选 | string |
返回:httpcode为200时返回图片流
httpcode为404时表示图片不存在
证件照下载 #
地址:https://www.clipimg.com/api/idphoto/download/证件照图片名称
请求方式:POST
数据类型:JSON
响应类型:Image
计费:此接口调用成功扣除API账户30点余额
参数[Header]
参数名 | 说明 | 是否必选 | 类型 |
X-API-Key | 秘钥 | 必选 | string |
Content-Type | 请求数据类型 | 必选 | application/json |
参数[Body]:
参数名 | 说明 | 是否必选 | 类型 |
text | 给证件照增加自定义文字 | 可选 | string |
返回:httpcode为200时返回图片流
httpcode为404时表示图片不存在
httpcode为402时表示点数已用完
# /**
# * 预览证件照
# */
def IdphotoPreview(self,preview_img_name,text=None):
response = requests.get(self.api_url + self.idphoto_preview + '/' +
preview_img_name+("?text="+text if text is not None and len(text)>0 else ''),
headers = {
'X-API-Key' : self.apiKey,
'Content-Type' : 'application/json'
}
)
return response.content
# /**
# * 证件照下载
# */
def IdphotoDownload(self,img_name,text=None):
ext = 'png' if '_jpg' not in img_name else 'jpg'
path = self.GenSavePath('result',ext)
try:
body = {"text":text} if text is not None and len(text)>0 else {}
response = requests.post(self.api_url + self.idphoto_download + '/' + img_name,
headers = {
'X-API-Key' : self.apiKey,
'Content-Type' : 'application/json'
},data=json.dumps(body)
)
if response.status_code == 200:
with open(path, 'wb') as out:
out.write(response.content)
return path
except Exception as e:
status_code = response.status_code
return status_code
证件照排版预览 #
地址:https://www.clipimg.com/api/idphoto/print_preview/打印排版照预览图名称
请求方式:GET
数据类型:application/x-www-form-urlencoded
响应类型:Image
计费:此接口不产生费用
参数:
参数均使用url拼接字符串形式拼接,
eg:https://www.clipimg.com/api/idphoto/print_preview/d00a4..._white?text=这是自定义文字
参数名 | 说明 | 是否必选 | 类型 |
text | 给证件照增加自定义文字,url拼接字符串形式拼接, eg:https://www.clipimg.com/api/idphoto/print_preview/d00a4…_white?text=这是自定义文字 | 可选 | string |
layouttype | 排版方式, 0表示紧凑、1表示宽松,默认0 | 可选 | int |
orientation | 方向,0表示横板(图片宽比高长)、1表示纵版,默认0 | 可选 | int |
返回:httpcode为200时返回图片流
httpcode为404时表示图片不存在
证件照排版下载 #
地址:https://www.clipimg.com/api/idphoto/print_download/打印排版照原图名称
请求方式:POST
数据类型:JSON
响应类型:Image
计费:此接口调用成功扣除API账户50点余额,换装排版照扣100点余额
参数[Header]
参数名 | 说明 | 是否必选 | 类型 |
X-API-Key | 秘钥 | 必选 | string |
Content-Type | 请求数据类型 | 必选 | application/json |
参数[Body]:
参数均使用url拼接字符串形式拼接,
eg:https://www.clipimg.com/api/idphoto/print_preview/d00a4..._white?text=这是自定义文字
参数名 | 说明 | 是否必选 | 类型 |
text | 给证件照增加自定义文字 | 可选 | string |
layouttype | 排版方式, 0表示紧凑、1表示宽松,默认0 | 可选 | int |
orientation | 方向,0表示横板(图片宽比高长)、1表示纵版,默认0 | 可选 | int |
返回:httpcode为200时返回图片流
httpcode为404时表示图片不存在
httpcode为402时表示点数已用完
# /**
# * 排版照预览
# */
def PrintIdphotoPreview(self,preview_img_name,text='',layouttype = 0,orientation= 0):
params = []
if text is not None and len(text)>0:
params.append('text='+text)
if layouttype == 1:
params.append('layouttype='+str(layouttype))
if orientation == 1:
params.append('orientation='+str(orientation))
param = "&".join(params)
if param != '':
param = '?'+param
response = requests.get(self.api_url + self.print_idphoto_preview + '/' + preview_img_name+param,
headers = {
'X-API-Key' : self.apiKey,
'Content-Type' : 'application/json'
}
)
return response.content
# /**
# * 排版照下载
# */
def PrintIdphotoDownload(self,img_name,text='',layouttype = 0,orientation= 0):
ext = 'png' if '_jpg' not in img_name else 'jpg'
path = self.GenSavePath('result',ext)
try:
body = {}
if text is not None and len(text)>0:
body['text']=text
if layouttype == 1:
body['layouttype']=layouttype
if orientation == 1:
body['orientation']=orientation
response = requests.post(self.api_url + self.print_idphoto_download + '/' + img_name,
headers = {
'X-API-Key' : self.apiKey,
'Content-Type' : 'application/json'
},data=json.dumps(body)
)
if response.status_code == 200:
with open(path, 'wb') as out:
out.write(response.content)
return path
except Exception as e:
status_code = response.status_code
return status_code
您可以使用代码对下载后的证件照实现排版,以节省成本!
使用代码实现证件照排版 #
- Python
- JavaScript
- Java
- PHP
- C#
from PIL import Image, ImageDraw
# 6 寸相纸尺寸(英寸)
SIX_INCH_WIDTH = 6.0
SIX_INCH_HEIGHT = 4.0
# 默认分辨率(dpi)
DEFAULT_DPI = 300
# 照片间空隙(像素)
GAP = 10
# 裁剪线颜色
CUTLINE_COLOR = (0, 0, 0)
# 裁剪线宽度
CUTLINE_WIDTH = 1
# 转换为像素
SIX_INCH_PIXEL_WIDTH = int(SIX_INCH_WIDTH * DEFAULT_DPI)
SIX_INCH_PIXEL_HEIGHT = int(SIX_INCH_HEIGHT * DEFAULT_DPI)
def layout_photos(photo_path, output_path):
# 打开照片
photo = Image.open(photo_path)
photo_width, photo_height = photo.size
try:
photo_dpi = photo.info['dpi'][0]
if photo_dpi == 0:
photo_dpi = DEFAULT_DPI
except KeyError:
photo_dpi = DEFAULT_DPI
# 计算照片实际打印尺寸(英寸)
photo_print_width = photo_width / photo_dpi
photo_print_height = photo_height / photo_dpi
# 转换为像素(使用默认 dpi)
photo_pixel_width = int(photo_print_width * DEFAULT_DPI)
photo_pixel_height = int(photo_print_height * DEFAULT_DPI)
# 调整照片大小
resized_photo = photo.resize((photo_pixel_width, photo_pixel_height))
# 创建相纸
paper = Image.new('RGB', (SIX_INCH_PIXEL_WIDTH, SIX_INCH_PIXEL_HEIGHT), 'white')
draw = ImageDraw.Draw(paper)
# 计算每行和每列可放置的照片数量
cols = (SIX_INCH_PIXEL_WIDTH + GAP) // (photo_pixel_width + GAP)
rows = (SIX_INCH_PIXEL_HEIGHT + GAP) // (photo_pixel_height + GAP)
# 计算排版的总宽度和高度
total_width = cols * photo_pixel_width + (cols - 1) * GAP
total_height = rows * photo_pixel_height + (rows - 1) * GAP
# 计算居中偏移量
start_x = (SIX_INCH_PIXEL_WIDTH - total_width) // 2
start_y = (SIX_INCH_PIXEL_HEIGHT - total_height) // 2
# 粘贴照片
for i in range(rows):
for j in range(cols):
x = start_x + j * (photo_pixel_width + GAP)
y = start_y + i * (photo_pixel_height + GAP)
paper.paste(resized_photo, (x, y))
# 绘制垂直裁剪线(间隙两边)
for j in range(cols + 1):
if j == 0:
left_x1 = start_x
draw.line((left_x1, 0, left_x1, start_y), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
draw.line((left_x1, start_y + total_height, left_x1, SIX_INCH_PIXEL_HEIGHT), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
elif j == cols:
left_x2 = start_x + total_width
draw.line((left_x2, 0, left_x2, start_y), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
draw.line((left_x2, start_y + total_height, left_x2, SIX_INCH_PIXEL_HEIGHT), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
else:
left_x1 = start_x + (j - 1) * (photo_pixel_width + GAP) + photo_pixel_width
left_x2 = left_x1 + GAP
draw.line((left_x1, 0, left_x1, start_y), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
draw.line((left_x1, start_y + total_height, left_x1, SIX_INCH_PIXEL_HEIGHT), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
draw.line((left_x2, 0, left_x2, start_y), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
draw.line((left_x2, start_y + total_height, left_x2, SIX_INCH_PIXEL_HEIGHT), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
# 绘制水平裁剪线(间隙两边)
for i in range(rows + 1):
if i == 0:
top_y1 = start_y
draw.line((0, top_y1, start_x, top_y1), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
draw.line((start_x + total_width, top_y1, SIX_INCH_PIXEL_WIDTH, top_y1), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
elif i == rows:
top_y2 = start_y + total_height
draw.line((0, top_y2, start_x, top_y2), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
draw.line((start_x + total_width, top_y2, SIX_INCH_PIXEL_WIDTH, top_y2), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
else:
top_y1 = start_y + (i - 1) * (photo_pixel_height + GAP) + photo_pixel_height
top_y2 = top_y1 + GAP
draw.line((0, top_y1, start_x, top_y1), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
draw.line((start_x + total_width, top_y1, SIX_INCH_PIXEL_WIDTH, top_y1), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
draw.line((0, top_y2, start_x, top_y2), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
draw.line((start_x + total_width, top_y2, SIX_INCH_PIXEL_WIDTH, top_y2), fill=CUTLINE_COLOR, width=CUTLINE_WIDTH)
paper.save(output_path)
# 示例调用
layout_photos('your_photo.jpg', 'layout_photo.jpg')
使用 Node.js 和 Jimp 库
const Jimp = require('jimp');
// 6 寸相纸尺寸(英寸)
const SIX_INCH_WIDTH = 6.0;
const SIX_INCH_HEIGHT = 4.0;
// 默认分辨率(dpi)
const DEFAULT_DPI = 300;
// 照片间空隙(像素)
const GAP = 10;
// 裁剪线颜色(RGBA)
const CUTLINE_COLOR = 0x000000FF;
// 裁剪线宽度
const CUTLINE_WIDTH = 1;
// 转换为像素
const SIX_INCH_PIXEL_WIDTH = Math.round(SIX_INCH_WIDTH * DEFAULT_DPI);
const SIX_INCH_PIXEL_HEIGHT = Math.round(SIX_INCH_HEIGHT * DEFAULT_DPI);
async function layoutPhotos(photoPath, outputPath) {
try {
// 打开照片
const photo = await Jimp.read(photoPath);
const photoWidth = photo.getWidth();
const photoHeight = photo.getHeight();
const photoDpi = DEFAULT_DPI; // 假设默认 DPI
// 计算照片实际打印尺寸(英寸)
const photoPrintWidth = photoWidth / photoDpi;
const photoPrintHeight = photoHeight / photoDpi;
// 转换为像素(使用默认 dpi)
const photoPixelWidth = Math.round(photoPrintWidth * DEFAULT_DPI);
const photoPixelHeight = Math.round(photoPrintHeight * DEFAULT_DPI);
// 调整照片大小
const resizedPhoto = photo.resize(photoPixelWidth, photoPixelHeight);
// 创建相纸
const paper = new Jimp(SIX_INCH_PIXEL_WIDTH, SIX_INCH_PIXEL_HEIGHT, 0xFFFFFFFF);
// 计算每行和每列可放置的照片数量
const cols = Math.floor((SIX_INCH_PIXEL_WIDTH + GAP) / (photoPixelWidth + GAP));
const rows = Math.floor((SIX_INCH_PIXEL_HEIGHT + GAP) / (photoPixelHeight + GAP));
// 计算排版的总宽度和高度
const totalWidth = cols * photoPixelWidth + (cols - 1) * GAP;
const totalHeight = rows * photoPixelHeight + (rows - 1) * GAP;
// 计算居中偏移量
const startX = Math.floor((SIX_INCH_PIXEL_WIDTH - totalWidth) / 2);
const startY = Math.floor((SIX_INCH_PIXEL_HEIGHT - totalHeight) / 2);
// 粘贴照片
for (let i = 0; i < rows; i++) {
for (let j = 0; j < cols; j++) {
const x = startX + j * (photoPixelWidth + GAP);
const y = startY + i * (photoPixelHeight + GAP);
paper.composite(resizedPhoto, x, y);
}
}
// 绘制垂直裁剪线(间隙两边)
for (let j = 0; j <= cols; j++) {
let leftX1, leftX2;
if (j === 0) {
leftX1 = startX;
drawVerticalCutLine(paper, leftX1, 0, startY);
drawVerticalCutLine(paper, leftX1, startY + totalHeight, SIX_INCH_PIXEL_HEIGHT);
} else if (j === cols) {
leftX2 = startX + totalWidth;
drawVerticalCutLine(paper, leftX2, 0, startY);
drawVerticalCutLine(paper, leftX2, startY + totalHeight, SIX_INCH_PIXEL_HEIGHT);
} else {
leftX1 = startX + (j - 1) * (photoPixelWidth + GAP) + photoPixelWidth;
leftX2 = leftX1 + GAP;
drawVerticalCutLine(paper, leftX1, 0, startY);
drawVerticalCutLine(paper, leftX1, startY + totalHeight, SIX_INCH_PIXEL_HEIGHT);
drawVerticalCutLine(paper, leftX2, 0, startY);
drawVerticalCutLine(paper, leftX2, startY + totalHeight, SIX_INCH_PIXEL_HEIGHT);
}
}
// 绘制水平裁剪线(间隙两边)
for (let i = 0; i <= rows; i++) {
let topY1, topY2;
if (i === 0) {
topY1 = startY;
drawHorizontalCutLine(paper, 0, startX, topY1);
drawHorizontalCutLine(paper, startX + totalWidth, SIX_INCH_PIXEL_WIDTH, topY1);
} else if (i === rows) {
topY2 = startY + totalHeight;
drawHorizontalCutLine(paper, 0, startX, topY2);
drawHorizontalCutLine(paper, startX + totalWidth, SIX_INCH_PIXEL_WIDTH, topY2);
} else {
topY1 = startY + (i - 1) * (photoPixelHeight + GAP) + photoPixelHeight;
topY2 = topY1 + GAP;
drawHorizontalCutLine(paper, 0, startX, topY1);
drawHorizontalCutLine(paper, startX + totalWidth, SIX_INCH_PIXEL_WIDTH, topY1);
drawHorizontalCutLine(paper, 0, startX, topY2);
drawHorizontalCutLine(paper, startX + totalWidth, SIX_INCH_PIXEL_WIDTH, topY2);
}
}
// 保存相纸
await paper.writeAsync(outputPath);
} catch (error) {
console.error('Error:', error);
}
}
function drawVerticalCutLine(paper, x, startY, endY) {
for (let y = startY; y < endY; y++) {
for (let w = 0; w < CUTLINE_WIDTH; w++) {
paper.setPixelColor(CUTLINE_COLOR, x + w, y);
}
}
}
function drawHorizontalCutLine(paper, startX, endX, y) {
for (let x = startX; x < endX; x++) {
for (let w = 0; w < CUTLINE_WIDTH; w++) {
paper.setPixelColor(CUTLINE_COLOR, x, y + w);
}
}
}
// 示例调用
layoutPhotos('your_photo.jpg', 'layout_photo.jpg');
import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import javax.imageio.ImageIO;
public class PhotoLayout {
// 6 寸相纸尺寸(英寸)
private static final double SIX_INCH_WIDTH = 6.0;
private static final double SIX_INCH_HEIGHT = 4.0;
// 默认分辨率(dpi)
private static final int DEFAULT_DPI = 300;
// 照片间空隙(像素)
private static final int GAP = 10;
// 裁剪线颜色
private static final Color CUTLINE_COLOR = Color.BLACK;
// 裁剪线宽度
private static final int CUTLINE_WIDTH = 1;
public static void main(String[] args) {
String photoPath = "your_photo.jpg";
String outputPath = "layout_photo.jpg";
layoutPhotos(photoPath, outputPath);
}
public static void layoutPhotos(String photoPath, String outputPath) {
try {
// 读取照片
BufferedImage photo = ImageIO.read(new File(photoPath));
int photoWidth = photo.getWidth();
int photoHeight = photo.getHeight();
int photoDpi = DEFAULT_DPI; // 假设默认 DPI
// 计算照片实际打印尺寸(英寸)
double photoPrintWidth = (double) photoWidth / photoDpi;
double photoPrintHeight = (double) photoHeight / photoDpi;
// 转换为像素(使用默认 dpi)
int photoPixelWidth = (int) (photoPrintWidth * DEFAULT_DPI);
int photoPixelHeight = (int) (photoPrintHeight * DEFAULT_DPI);
// 调整照片大小
BufferedImage resizedPhoto = new BufferedImage(photoPixelWidth, photoPixelHeight, BufferedImage.TYPE_INT_RGB);
Graphics2D g2d = resizedPhoto.createGraphics();
g2d.drawImage(photo, 0, 0, photoPixelWidth, photoPixelHeight, null);
g2d.dispose();
// 相纸尺寸(像素)
int sixInchPixelWidth = (int) (SIX_INCH_WIDTH * DEFAULT_DPI);
int sixInchPixelHeight = (int) (SIX_INCH_HEIGHT * DEFAULT_DPI);
// 创建相纸
BufferedImage paper = new BufferedImage(sixInchPixelWidth, sixInchPixelHeight, BufferedImage.TYPE_INT_RGB);
Graphics2D paperG2d = paper.createGraphics();
paperG2d.setColor(Color.WHITE);
paperG2d.fillRect(0, 0, sixInchPixelWidth, sixInchPixelHeight);
// 计算每行和每列可放置的照片数量
int cols = (sixInchPixelWidth + GAP) / (photoPixelWidth + GAP);
int rows = (sixInchPixelHeight + GAP) / (photoPixelHeight + GAP);
// 计算排版的总宽度和高度
int totalWidth = cols * photoPixelWidth + (cols - 1) * GAP;
int totalHeight = rows * photoPixelHeight + (rows - 1) * GAP;
// 计算居中偏移量
int startX = (sixInchPixelWidth - totalWidth) / 2;
int startY = (sixInchPixelHeight - totalHeight) / 2;
// 粘贴照片
for (int i = 0; i < rows; i++) {
for (int j = 0; j < cols; j++) {
int x = startX + j * (photoPixelWidth + GAP);
int y = startY + i * (photoPixelHeight + GAP);
paperG2d.drawImage(resizedPhoto, x, y, null);
}
}
// 绘制垂直裁剪线(间隙两边)
for (int j = 0; j <= cols; j++) {
int leftX1, leftX2;
if (j == 0) {
leftX1 = startX;
drawVerticalCutLine(paperG2d, leftX1, 0, startY);
drawVerticalCutLine(paperG2d, leftX1, startY + totalHeight, sixInchPixelHeight);
} else if (j == cols) {
leftX2 = startX + totalWidth;
drawVerticalCutLine(paperG2d, leftX2, 0, startY);
drawVerticalCutLine(paperG2d, leftX2, startY + totalHeight, sixInchPixelHeight);
} else {
leftX1 = startX + (j - 1) * (photoPixelWidth + GAP) + photoPixelWidth;
leftX2 = leftX1 + GAP;
drawVerticalCutLine(paperG2d, leftX1, 0, startY);
drawVerticalCutLine(paperG2d, leftX1, startY + totalHeight, sixInchPixelHeight);
drawVerticalCutLine(paperG2d, leftX2, 0, startY);
drawVerticalCutLine(paperG2d, leftX2, startY + totalHeight, sixInchPixelHeight);
}
}
// 绘制水平裁剪线(间隙两边)
for (int i = 0; i <= rows; i++) {
int topY1, topY2;
if (i == 0) {
topY1 = startY;
drawHorizontalCutLine(paperG2d, 0, startX, topY1);
drawHorizontalCutLine(paperG2d, startX + totalWidth, sixInchPixelWidth, topY1);
} else if (i == rows) {
topY2 = startY + totalHeight;
drawHorizontalCutLine(paperG2d, 0, startX, topY2);
drawHorizontalCutLine(paperG2d, startX + totalWidth, sixInchPixelWidth, topY2);
} else {
topY1 = startY + (i - 1) * (photoPixelHeight + GAP) + photoPixelHeight;
topY2 = topY1 + GAP;
drawHorizontalCutLine(paperG2d, 0, startX, topY1);
drawHorizontalCutLine(paperG2d, startX + totalWidth, sixInchPixelWidth, topY1);
drawHorizontalCutLine(paperG2d, 0, startX, topY2);
drawHorizontalCutLine(paperG2d, startX + totalWidth, sixInchPixelWidth, topY2);
}
}
paperG2d.dispose();
// 保存相纸
ImageIO.write(paper, "jpg", new File(outputPath));
} catch (IOException e) {
e.printStackTrace();
}
}
private static void drawVerticalCutLine(Graphics2D g2d, int x, int startY, int endY) {
g2d.setColor(CUTLINE_COLOR);
for (int i = 0; i < CUTLINE_WIDTH; i++) {
g2d.drawLine(x + i, startY, x + i, endY);
}
}
private static void drawHorizontalCutLine(Graphics2D g2d, int startX, int endX, int y) {
g2d.setColor(CUTLINE_COLOR);
for (int i = 0; i < CUTLINE_WIDTH; i++) {
g2d.drawLine(startX, y + i, endX, y + i);
}
}
}
使用 GD 库
<?php
// 6 寸相纸尺寸(英寸)
define('SIX_INCH_WIDTH', 6.0);
define('SIX_INCH_HEIGHT', 4.0);
// 默认分辨率(dpi)
define('DEFAULT_DPI', 300);
// 照片间空隙(像素)
define('GAP', 10);
// 裁剪线颜色,这里使用黑色
define('CUTLINE_COLOR', imagecolorallocate($paper, 0, 0, 0));
// 裁剪线宽度
define('CUTLINE_WIDTH', 1);
// 转换为像素
$sixInchPixelWidth = intval(SIX_INCH_WIDTH * DEFAULT_DPI);
$sixInchPixelHeight = intval(SIX_INCH_HEIGHT * DEFAULT_DPI);
function layoutPhotos($photoPath, $outputPath) {
global $sixInchPixelWidth, $sixInchPixelHeight;
// 打开照片
$photo = imagecreatefromjpeg($photoPath);
if (!$photo) {
die("无法打开照片文件。");
}
$photoWidth = imagesx($photo);
$photoHeight = imagesy($photo);
$photoDpi = DEFAULT_DPI; // 假设默认 DPI
// 计算照片实际打印尺寸(英寸)
$photoPrintWidth = $photoWidth / $photoDpi;
$photoPrintHeight = $photoHeight / $photoDpi;
// 转换为像素(使用默认 dpi)
$photoPixelWidth = intval($photoPrintWidth * DEFAULT_DPI);
$photoPixelHeight = intval($photoPrintHeight * DEFAULT_DPI);
// 调整照片大小
$resizedPhoto = imagecreatetruecolor($photoPixelWidth, $photoPixelHeight);
imagecopyresampled($resizedPhoto, $photo, 0, 0, 0, 0, $photoPixelWidth, $photoPixelHeight, $photoWidth, $photoHeight);
// 创建相纸
$paper = imagecreatetruecolor($sixInchPixelWidth, $sixInchPixelHeight);
$white = imagecolorallocate($paper, 255, 255, 255);
imagefill($paper, 0, 0, $white);
// 计算每行和每列可放置的照片数量
$cols = intval(($sixInchPixelWidth + GAP) / ($photoPixelWidth + GAP));
$rows = intval(($sixInchPixelHeight + GAP) / ($photoPixelHeight + GAP));
// 计算排版的总宽度和高度
$totalWidth = $cols * $photoPixelWidth + ($cols - 1) * GAP;
$totalHeight = $rows * $photoPixelHeight + ($rows - 1) * GAP;
// 计算居中偏移量
$startX = intval(($sixInchPixelWidth - $totalWidth) / 2);
$startY = intval(($sixInchPixelHeight - $totalHeight) / 2);
// 粘贴照片
for ($i = 0; $i < $rows; $i++) {
for ($j = 0; $j < $cols; $j++) {
$x = $startX + $j * ($photoPixelWidth + GAP);
$y = $startY + $i * ($photoPixelHeight + GAP);
imagecopy($paper, $resizedPhoto, $x, $y, 0, 0, $photoPixelWidth, $photoPixelHeight);
}
}
// 绘制垂直裁剪线(间隙两边)
for ($j = 0; $j <= $cols; $j++) {
if ($j == 0) {
$leftX1 = $startX;
drawVerticalCutLine($paper, $leftX1, 0, $startY);
drawVerticalCutLine($paper, $leftX1, $startY + $totalHeight, $sixInchPixelHeight);
} elseif ($j == $cols) {
$leftX2 = $startX + $totalWidth;
drawVerticalCutLine($paper, $leftX2, 0, $startY);
drawVerticalCutLine($paper, $leftX2, $startY + $totalHeight, $sixInchPixelHeight);
} else {
$leftX1 = $startX + ($j - 1) * ($photoPixelWidth + GAP) + $photoPixelWidth;
$leftX2 = $leftX1 + GAP;
drawVerticalCutLine($paper, $leftX1, 0, $startY);
drawVerticalCutLine($paper, $leftX1, $startY + $totalHeight, $sixInchPixelHeight);
drawVerticalCutLine($paper, $leftX2, 0, $startY);
drawVerticalCutLine($paper, $leftX2, $startY + $totalHeight, $sixInchPixelHeight);
}
}
// 绘制水平裁剪线(间隙两边)
for ($i = 0; $i <= $rows; $i++) {
if ($i == 0) {
$topY1 = $startY;
drawHorizontalCutLine($paper, 0, $startX, $topY1);
drawHorizontalCutLine($paper, $startX + $totalWidth, $sixInchPixelWidth, $topY1);
} elseif ($i == $rows) {
$topY2 = $startY + $totalHeight;
drawHorizontalCutLine($paper, 0, $startX, $topY2);
drawHorizontalCutLine($paper, $startX + $totalWidth, $sixInchPixelWidth, $topY2);
} else {
$topY1 = $startY + ($i - 1) * ($photoPixelHeight + GAP) + $photoPixelHeight;
$topY2 = $topY1 + GAP;
drawHorizontalCutLine($paper, 0, $startX, $topY1);
drawHorizontalCutLine($paper, $startX + $totalWidth, $sixInchPixelWidth, $topY1);
drawHorizontalCutLine($paper, 0, $startX, $topY2);
drawHorizontalCutLine($paper, $startX + $totalWidth, $sixInchPixelWidth, $topY2);
}
}
// 保存相纸
imagejpeg($paper, $outputPath);
// 释放资源
imagedestroy($photo);
imagedestroy($resizedPhoto);
imagedestroy($paper);
}
function drawVerticalCutLine($image, $x, $startY, $endY) {
global $CUTLINE_COLOR, $CUTLINE_WIDTH;
for ($i = 0; $i < $CUTLINE_WIDTH; $i++) {
imageline($image, $x + $i, $startY, $x + $i, $endY, $CUTLINE_COLOR);
}
}
function drawHorizontalCutLine($image, $startX, $endX, $y) {
global $CUTLINE_COLOR, $CUTLINE_WIDTH;
for ($i = 0; $i < $CUTLINE_WIDTH; $i++) {
imageline($image, $startX, $y + $i, $endX, $y + $i, $CUTLINE_COLOR);
}
}
// 示例调用
layoutPhotos('your_photo.jpg', 'layout_photo.jpg');
?>
using System;
using System.Drawing;
using System.Drawing.Imaging;
class PhotoLayout
{
// 6 寸相纸尺寸(英寸)
private const double SIX_INCH_WIDTH = 6.0;
private const double SIX_INCH_HEIGHT = 4.0;
// 默认分辨率(dpi)
private const int DEFAULT_DPI = 300;
// 照片间空隙(像素)
private const int GAP = 10;
// 裁剪线颜色
private static readonly Color CUTLINE_COLOR = Color.Black;
// 裁剪线宽度
private const int CUTLINE_WIDTH = 1;
static void Main()
{
string photoPath = "your_photo.jpg";
string outputPath = "layout_photo.jpg";
LayoutPhotos(photoPath, outputPath);
}
static void LayoutPhotos(string photoPath, string outputPath)
{
try
{
// 读取照片
using (Image photo = Image.FromFile(photoPath))
{
int photoWidth = photo.Width;
int photoHeight = photo.Height;
int photoDpi = photo.HorizontalResolution;
if (photoDpi == 0)
{
photoDpi = DEFAULT_DPI;
}
// 计算照片实际打印尺寸(英寸)
double photoPrintWidth = (double)photoWidth / photoDpi;
double photoPrintHeight = (double)photoHeight / photoDpi;
// 转换为像素(使用默认 dpi)
int photoPixelWidth = (int)(photoPrintWidth * DEFAULT_DPI);
int photoPixelHeight = (int)(photoPrintHeight * DEFAULT_DPI);
// 调整照片大小
using (Bitmap resizedPhoto = new Bitmap(photo, photoPixelWidth, photoPixelHeight))
{
// 相纸尺寸(像素)
int sixInchPixelWidth = (int)(SIX_INCH_WIDTH * DEFAULT_DPI);
int sixInchPixelHeight = (int)(SIX_INCH_HEIGHT * DEFAULT_DPI);
// 创建相纸
using (Bitmap paper = new Bitmap(sixInchPixelWidth, sixInchPixelHeight))
{
using (Graphics g = Graphics.FromImage(paper))
{
g.Clear(Color.White);
// 计算每行和每列可放置的照片数量
int cols = (sixInchPixelWidth + GAP) / (photoPixelWidth + GAP);
int rows = (sixInchPixelHeight + GAP) / (photoPixelHeight + GAP);
// 计算排版的总宽度和高度
int totalWidth = cols * photoPixelWidth + (cols - 1) * GAP;
int totalHeight = rows * photoPixelHeight + (rows - 1) * GAP;
// 计算居中偏移量
int startX = (sixInchPixelWidth - totalWidth) / 2;
int startY = (sixInchPixelHeight - totalHeight) / 2;
// 粘贴照片
for (int i = 0; i < rows; i++)
{
for (int j = 0; j < cols; j++)
{
int x = startX + j * (photoPixelWidth + GAP);
int y = startY + i * (photoPixelHeight + GAP);
g.DrawImage(resizedPhoto, x, y);
}
}
// 绘制垂直裁剪线(间隙两边)
for (int j = 0; j <= cols; j++)
{
int leftX1, leftX2;
if (j == 0)
{
leftX1 = startX;
DrawVerticalCutLine(g, leftX1, 0, startY);
DrawVerticalCutLine(g, leftX1, startY + totalHeight, sixInchPixelHeight);
}
else if (j == cols)
{
leftX2 = startX + totalWidth;
DrawVerticalCutLine(g, leftX2, 0, startY);
DrawVerticalCutLine(g, leftX2, startY + totalHeight, sixInchPixelHeight);
}
else
{
leftX1 = startX + (j - 1) * (photoPixelWidth + GAP) + photoPixelWidth;
leftX2 = leftX1 + GAP;
DrawVerticalCutLine(g, leftX1, 0, startY);
DrawVerticalCutLine(g, leftX1, startY + totalHeight, sixInchPixelHeight);
DrawVerticalCutLine(g, leftX2, 0, startY);
DrawVerticalCutLine(g, leftX2, startY + totalHeight, sixInchPixelHeight);
}
}
// 绘制水平裁剪线(间隙两边)
for (int i = 0; i <= rows; i++)
{
int topY1, topY2;
if (i == 0)
{
topY1 = startY;
DrawHorizontalCutLine(g, 0, startX, topY1);
DrawHorizontalCutLine(g, startX + totalWidth, sixInchPixelWidth, topY1);
}
else if (i == rows)
{
topY2 = startY + totalHeight;
DrawHorizontalCutLine(g, 0, startX, topY2);
DrawHorizontalCutLine(g, startX + totalWidth, sixInchPixelWidth, topY2);
}
else
{
topY1 = startY + (i - 1) * (photoPixelHeight + GAP) + photoPixelHeight;
topY2 = topY1 + GAP;
DrawHorizontalCutLine(g, 0, startX, topY1);
DrawHorizontalCutLine(g, startX + totalWidth, sixInchPixelWidth, topY1);
DrawHorizontalCutLine(g, 0, startX, topY2);
DrawHorizontalCutLine(g, startX + totalWidth, sixInchPixelWidth, topY2);
}
}
}
// 保存相纸
paper.Save(outputPath, ImageFormat.Jpeg);
}
}
}
}
catch (Exception ex)
{
Console.WriteLine("处理过程中出现错误: " + ex.Message);
}
}
static void DrawVerticalCutLine(Graphics g, int x, int startY, int endY)
{
using (Pen pen = new Pen(CUTLINE_COLOR, CUTLINE_WIDTH))
{
g.DrawLine(pen, x, startY, x, endY);
}
}
static void DrawHorizontalCutLine(Graphics g, int startX, int endX, int y)
{
using (Pen pen = new Pen(CUTLINE_COLOR, CUTLINE_WIDTH))
{
g.DrawLine(pen, startX, y, endX, y);
}
}
}
你需要将 'your_photo.jpg' 替换为实际的照片文件路径,'layout_photo.jpg' 替换为你想要保存的排版后照片的文件路径。
证件照规格列表 #
如需全部规格列表数据,请咨询客服获取。
id | name |
1 | 一寸 |
2 | 小一寸 |
3 | 大一寸 |
4 | 二寸 |
5 | 小二寸 |
6 | 三寸 |
7 | 五寸 |
8 | 国考(二寸)(35*45mm) |
9 | 英语四六级考试(144×192,0~10kb)(12*16mm) |
10 | 全国计算机等级考试(12*16mm) |