HivisionIDPhotos / demo /processor.py
TheEeeeLin's picture
update
ec5a4d3
import numpy as np
from hivision import IDCreator
from hivision.error import FaceError, APIError
from hivision.utils import (
add_background,
add_background_with_image,
resize_image_to_kb,
add_watermark,
save_image_dpi_to_bytes,
)
from hivision.creator.layout_calculator import (
generate_layout_array,
generate_layout_image,
)
from hivision.creator.choose_handler import choose_handler
from hivision.plugin.template.template_calculator import generte_template_photo
from demo.utils import range_check
import gradio as gr
import os
import cv2
import time
from demo.locales import LOCALES
base_path = os.path.dirname(os.path.abspath(__file__))
class IDPhotoProcessor:
def process(
self,
input_image,
mode_option,
size_list_option,
color_option,
render_option,
image_kb_options,
custom_color_R,
custom_color_G,
custom_color_B,
custom_color_hex_value,
custom_size_height,
custom_size_width,
custom_size_height_mm,
custom_size_width_mm,
custom_image_kb,
language,
matting_model_option,
watermark_option,
watermark_text,
watermark_text_color,
watermark_text_size,
watermark_text_opacity,
watermark_text_angle,
watermark_text_space,
face_detect_option,
head_measure_ratio=0.2,
top_distance_max=0.12,
whitening_strength=0,
image_dpi_option=False,
custom_image_dpi=None,
brightness_strength=0,
contrast_strength=0,
sharpen_strength=0,
saturation_strength=0,
plugin_option=[],
):
# 初始化参数
top_distance_min = top_distance_max - 0.02
# 得到render_option在LOCALES["render_mode"][language]["choices"]中的索引
render_option_index = LOCALES["render_mode"][language]["choices"].index(
render_option
)
# 读取插件选项
# 人脸对齐选项
if LOCALES["plugin"][language]["choices"][0] in plugin_option:
face_alignment_option = True
else:
face_alignment_option = False
# 排版裁剪线选项
if LOCALES["plugin"][language]["choices"][1] in plugin_option:
layout_photo_crop_line_option = True
else:
layout_photo_crop_line_option = False
# JPEG格式选项
if LOCALES["plugin"][language]["choices"][2] in plugin_option:
jpeg_format_option = True
else:
jpeg_format_option = False
# 五寸相纸选项
if LOCALES["plugin"][language]["choices"][3] in plugin_option:
five_inch_option = True
else:
five_inch_option = False
idphoto_json = self._initialize_idphoto_json(
mode_option, color_option, render_option_index, image_kb_options, layout_photo_crop_line_option, jpeg_format_option, five_inch_option
)
# 处理尺寸模式
size_result = self._process_size_mode(
idphoto_json,
language,
size_list_option,
custom_size_height,
custom_size_width,
custom_size_height_mm,
custom_size_width_mm,
)
if isinstance(size_result, list):
return size_result # 返回错误信息
# 处理颜色模式
self._process_color_mode(
idphoto_json,
language,
color_option,
custom_color_R,
custom_color_G,
custom_color_B,
custom_color_hex_value,
)
# 如果设置了自定义KB大小
if (
idphoto_json["image_kb_mode"]
== LOCALES["image_kb"][language]["choices"][-1]
):
idphoto_json["custom_image_kb"] = custom_image_kb
# 如果设置了自定义DPI大小
if image_dpi_option == LOCALES["image_dpi"][language]["choices"][-1]:
idphoto_json["custom_image_dpi"] = custom_image_dpi
# 创建IDCreator实例并设置处理器
creator = IDCreator()
choose_handler(creator, matting_model_option, face_detect_option)
# 生成证件照
try:
result = self._generate_id_photo(
creator,
input_image,
idphoto_json,
language,
head_measure_ratio,
top_distance_max,
top_distance_min,
whitening_strength,
brightness_strength,
contrast_strength,
sharpen_strength,
saturation_strength,
face_alignment_option,
)
except (FaceError, APIError):
return self._handle_photo_generation_error(language)
# 后处理生成的照片
return self._process_generated_photo(
result,
idphoto_json,
language,
watermark_option,
watermark_text,
watermark_text_size,
watermark_text_opacity,
watermark_text_angle,
watermark_text_space,
watermark_text_color,
)
# 初始化idphoto_json字典
def _initialize_idphoto_json(
self,
mode_option,
color_option,
render_option,
image_kb_options,
layout_photo_crop_line_option,
jpeg_format_option,
five_inch_option,
):
"""初始化idphoto_json字典"""
return {
"size_mode": mode_option,
"color_mode": color_option,
"render_mode": render_option,
"image_kb_mode": image_kb_options,
"custom_image_kb": None,
"custom_image_dpi": None,
"layout_photo_crop_line_option": layout_photo_crop_line_option,
"jpeg_format_option": jpeg_format_option,
"five_inch_option": five_inch_option,
}
# 处理尺寸模式
def _process_size_mode(
self,
idphoto_json,
language,
size_list_option,
custom_size_height,
custom_size_width,
custom_size_height_mm,
custom_size_width_mm,
):
"""处理尺寸模式"""
# 如果选择了尺寸列表
if idphoto_json["size_mode"] == LOCALES["size_mode"][language]["choices"][0]:
idphoto_json["size"] = LOCALES["size_list"][language]["develop"][
size_list_option
]
# 如果选择了自定义尺寸(px或mm)
elif (
idphoto_json["size_mode"] == LOCALES["size_mode"][language]["choices"][2]
or idphoto_json["size_mode"] == LOCALES["size_mode"][language]["choices"][3]
):
# 如果选择了自定义尺寸(px)
if (
idphoto_json["size_mode"]
== LOCALES["size_mode"][language]["choices"][2]
):
id_height, id_width = int(custom_size_height), int(custom_size_width)
# 如果选择了自定义尺寸(mm)
else:
# 将mm转换为px
id_height = int(custom_size_height_mm / 25.4 * 300)
id_width = int(custom_size_width_mm / 25.4 * 300)
# 检查尺寸像素是否在100到1800之间
if (
id_height < id_width
or min(id_height, id_width) < 100
or max(id_height, id_width) > 1800
):
return self._create_error_response(language)
idphoto_json["size"] = (id_height, id_width)
# 如果选择了只换底
else:
idphoto_json["size"] = (None, None)
# 处理颜色模式
def _process_color_mode(
self,
idphoto_json,
language,
color_option,
custom_color_R,
custom_color_G,
custom_color_B,
custom_color_hex_value,
):
"""处理颜色模式"""
# 如果选择了自定义颜色BGR
if idphoto_json["color_mode"] == LOCALES["bg_color"][language]["choices"][-2]:
idphoto_json["color_bgr"] = tuple(
map(range_check, [custom_color_R, custom_color_G, custom_color_B])
)
# 如果选择了自定义颜色HEX
elif idphoto_json["color_mode"] == LOCALES["bg_color"][language]["choices"][-1]:
hex_color = custom_color_hex_value
# 将十六进制颜色转换为RGB颜色,如果长度为6,则直接转换,如果长度为7,则去掉#号再转换
if len(hex_color) == 6:
idphoto_json["color_bgr"] = tuple(
int(hex_color[i : i + 2], 16) for i in (0, 2, 4)
)
elif len(hex_color) == 7:
hex_color = hex_color[1:]
idphoto_json["color_bgr"] = tuple(
int(hex_color[i : i + 2], 16) for i in (0, 2, 4)
)
else:
raise ValueError(
"Invalid hex color. You can only use 6 or 7 characters. For example: #FFFFFF or FFFFFF"
)
# 如果选择了美式证件照
elif idphoto_json["color_mode"] == LOCALES["bg_color"][language]["choices"][-3]:
idphoto_json["color_bgr"] = (255, 255, 255)
else:
hex_color = LOCALES["bg_color"][language]["develop"][color_option]
idphoto_json["color_bgr"] = tuple(
int(hex_color[i : i + 2], 16) for i in (0, 2, 4)
)
# 生成证件照
def _generate_id_photo(
self,
creator: IDCreator,
input_image,
idphoto_json,
language,
head_measure_ratio,
top_distance_max,
top_distance_min,
whitening_strength,
brightness_strength,
contrast_strength,
sharpen_strength,
saturation_strength,
face_alignment_option,
):
"""生成证件照"""
change_bg_only = (
idphoto_json["size_mode"] in LOCALES["size_mode"][language]["choices"][1]
)
return creator(
input_image,
change_bg_only=change_bg_only,
size=idphoto_json["size"],
head_measure_ratio=head_measure_ratio,
head_top_range=(top_distance_max, top_distance_min),
whitening_strength=whitening_strength,
brightness_strength=brightness_strength,
contrast_strength=contrast_strength,
sharpen_strength=sharpen_strength,
saturation_strength=saturation_strength,
face_alignment=face_alignment_option,
)
# 处理照片生成错误
def _handle_photo_generation_error(self, language):
"""处理照片生成错误"""
return [gr.update(value=None) for _ in range(4)] + [
gr.update(visible=False),
gr.update(
value=LOCALES["notification"][language]["face_error"], visible=True
),
None,
]
# 处理生成的照片
def _process_generated_photo(
self,
result,
idphoto_json,
language,
watermark_option,
watermark_text,
watermark_text_size,
watermark_text_opacity,
watermark_text_angle,
watermark_text_space,
watermark_text_color,
):
"""处理生成的照片"""
result_image_standard, result_image_hd, _, _, _, _ = result
result_image_standard_png = np.uint8(result_image_standard)
result_image_hd_png = np.uint8(result_image_hd)
# 渲染背景
result_image_standard, result_image_hd = self._render_background(
result_image_standard, result_image_hd, idphoto_json, language
)
# 添加水印
if watermark_option == LOCALES["watermark_switch"][language]["choices"][1]:
result_image_standard, result_image_hd = self._add_watermark(
result_image_standard,
result_image_hd,
watermark_text,
watermark_text_size,
watermark_text_opacity,
watermark_text_angle,
watermark_text_space,
watermark_text_color,
)
# 生成排版照片
result_image_layout, result_image_layout_visible = self._generate_image_layout(
idphoto_json,
result_image_standard,
language,
)
# 生成模板照片
result_image_template, result_image_template_visible = self._generate_image_template(
idphoto_json,
result_image_hd,
language,
)
# 调整图片大小
output_image_path_dict = self._save_image(
result_image_standard,
result_image_hd,
result_image_layout,
idphoto_json,
format="jpeg" if idphoto_json["jpeg_format_option"] else "png",
)
# 返回
if result_image_layout is not None:
result_image_layout = output_image_path_dict["layout"]["path"]
return self._create_response(
output_image_path_dict["standard"]["path"],
output_image_path_dict["hd"]["path"],
result_image_standard_png,
result_image_hd_png,
gr.update(value=result_image_layout, visible=result_image_layout_visible),
gr.update(value=result_image_template, visible=result_image_template_visible),
gr.update(visible = result_image_template_visible),
)
# 渲染背景
def _render_background(self, result_image_standard, result_image_hd, idphoto_json, language):
"""渲染背景"""
render_modes = {0: "pure_color", 1: "updown_gradient", 2: "center_gradient"}
render_mode = render_modes[idphoto_json["render_mode"]]
if idphoto_json["color_mode"] != LOCALES["bg_color"][language]["choices"][-3]:
result_image_standard = np.uint8(
add_background(
result_image_standard, bgr=idphoto_json["color_bgr"], mode=render_mode
)
)
result_image_hd = np.uint8(
add_background(
result_image_hd, bgr=idphoto_json["color_bgr"], mode=render_mode
)
)
# 如果选择了美式证件照
else:
result_image_standard = np.uint8(
add_background_with_image(
result_image_standard,
background_image=cv2.imread(os.path.join(base_path, "assets", "american-style.png"))
)
)
result_image_hd = np.uint8(
add_background_with_image(
result_image_hd,
background_image=cv2.imread(os.path.join(base_path, "assets", "american-style.png"))
)
)
return result_image_standard, result_image_hd
# 生成排版照片
def _generate_image_layout(
self,
idphoto_json,
result_image_standard,
language,
):
"""生成排版照片"""
# 如果选择了只换底,则不生成排版照片
if idphoto_json["size_mode"] in LOCALES["size_mode"][language]["choices"][1]:
return None, False
typography_arr, typography_rotate = generate_layout_array(
input_height=idphoto_json["size"][0],
input_width=idphoto_json["size"][1],
LAYOUT_HEIGHT= 1205 if not idphoto_json["five_inch_option"] else 1051,
LAYOUT_WIDTH= 1795 if not idphoto_json["five_inch_option"] else 1500,
)
result_image_layout = generate_layout_image(
result_image_standard,
typography_arr,
typography_rotate,
height=idphoto_json["size"][0],
width=idphoto_json["size"][1],
crop_line=idphoto_json["layout_photo_crop_line_option"],
LAYOUT_HEIGHT=1205 if not idphoto_json["five_inch_option"] else 1051,
LAYOUT_WIDTH=1795 if not idphoto_json["five_inch_option"] else 1500,
)
return result_image_layout, True
# 生成模板照片
def _generate_image_template(
self,
idphoto_json,
result_image_hd,
language,
):
# 如果选择了只换底,则不生成模板照片
if idphoto_json["size_mode"] in LOCALES["size_mode"][language]["choices"][1]:
return None, False
TEMPLATE_NAME_LIST = ["template_1", "template_2"]
"""生成模板照片"""
result_image_template_list = []
for template_name in TEMPLATE_NAME_LIST:
result_image_template = generte_template_photo(
template_name=template_name,
input_image=result_image_hd,
)
result_image_template_list.append(result_image_template)
return result_image_template_list, True
# 添加水印
def _add_watermark(
self,
result_image_standard,
result_image_hd,
watermark_text,
watermark_text_size,
watermark_text_opacity,
watermark_text_angle,
watermark_text_space,
watermark_text_color,
):
"""添加水印"""
watermark_params = {
"text": watermark_text,
"size": watermark_text_size,
"opacity": watermark_text_opacity,
"angle": watermark_text_angle,
"space": watermark_text_space,
"color": watermark_text_color,
}
result_image_standard = add_watermark(
image=result_image_standard, **watermark_params
)
result_image_hd = add_watermark(image=result_image_hd, **watermark_params)
return result_image_standard, result_image_hd
def _save_image(
self,
result_image_standard,
result_image_hd,
result_image_layout,
idphoto_json,
format="png",
):
# 设置输出路径(临时目录)
import tempfile
base_path = tempfile.mkdtemp()
timestamp = int(time.time())
output_paths = {
"standard": {
"path": f"{base_path}/{timestamp}_standard",
"processed": False,
},
"hd": {"path": f"{base_path}/{timestamp}_hd", "processed": False},
"layout": {"path": f"{base_path}/{timestamp}_layout", "processed": False},
}
# 获取自定义的KB和DPI值
custom_kb = idphoto_json.get("custom_image_kb")
custom_dpi = idphoto_json.get("custom_image_dpi", 300)
# 处理同时有自定义KB和DPI的情况
if custom_kb and custom_dpi:
# 为所有输出路径添加DPI信息
for key in output_paths:
output_paths[key]["path"] += f"_{custom_dpi}dpi.{format}"
# 为标准图像添加KB信息
output_paths["standard"]["path"] = output_paths["standard"]["path"].replace(
f".{format}", f"_{custom_kb}kb.{format}"
)
# 调整标准图像大小并保存
resize_image_to_kb(
result_image_standard,
output_paths["standard"]["path"],
custom_kb,
dpi=custom_dpi,
)
# 保存高清图像和排版图像
save_image_dpi_to_bytes(
result_image_hd, output_paths["hd"]["path"], dpi=custom_dpi
)
if result_image_layout is not None:
save_image_dpi_to_bytes(
result_image_layout, output_paths["layout"]["path"], dpi=custom_dpi
)
return output_paths
# 只有自定义DPI的情况
elif custom_dpi:
for key in output_paths:
# 保存所有图像,使用自定义DPI
# 如果只换底,则不保存排版图像
if key == "layout" and result_image_layout is None:
continue
output_paths[key]["path"] += f"_{custom_dpi}dpi.{format}"
save_image_dpi_to_bytes(
locals()[f"result_image_{key}"],
output_paths[key]["path"],
dpi=custom_dpi,
)
return output_paths
# 只有自定义KB的情况
elif custom_kb:
output_paths["standard"]["path"] += f"_{custom_kb}kb.{format}"
output_paths["hd"]["path"] += f".{format}"
if not (key == "layout" and result_image_layout is None):
output_paths[key]["path"] += f".{format}"
# 只调整标准图像大小
resize_image_to_kb(
result_image_standard,
output_paths["standard"]["path"],
custom_kb,
dpi=300,
)
# 保存高清图像和排版图像
save_image_dpi_to_bytes(
result_image_hd, output_paths["hd"]["path"], dpi=300
)
if result_image_layout is not None:
save_image_dpi_to_bytes(
result_image_layout, output_paths["layout"]["path"], dpi=300
)
return output_paths
# 没有自定义设置
else:
output_paths["standard"]["path"] += f".{format}"
output_paths["hd"]["path"] += f".{format}"
output_paths["layout"]["path"] += f".{format}"
# 保存所有图像
save_image_dpi_to_bytes(
result_image_standard, output_paths["standard"]["path"], dpi=300
)
save_image_dpi_to_bytes(
result_image_hd, output_paths["hd"]["path"], dpi=300
)
if result_image_layout is not None:
save_image_dpi_to_bytes(
result_image_layout, output_paths["layout"]["path"], dpi=300
)
return output_paths
def _create_response(
self,
result_image_standard,
result_image_hd,
result_image_standard_png,
result_image_hd_png,
result_layout_image_gr,
result_image_template_gr,
result_image_template_accordion_gr,
):
"""创建响应"""
response = [
result_image_standard,
result_image_hd,
result_image_standard_png,
result_image_hd_png,
result_layout_image_gr,
result_image_template_gr,
result_image_template_accordion_gr,
gr.update(visible=False),
]
return response
def _create_error_response(self, language):
"""创建错误响应"""
return [gr.update(value=None) for _ in range(4)] + [
None,
gr.update(
value=LOCALES["size_mode"][language]["custom_size_eror"], visible=True
),
None,
]