File size: 7,527 Bytes
ca46a75
 
 
 
 
 
 
 
 
 
 
 
 
 
7173af9
f8cafb8
ca46a75
f8cafb8
434720c
ca46a75
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1c25fe3
f8cafb8
ca46a75
 
 
 
 
 
 
 
7173af9
ca46a75
 
 
7173af9
 
f8cafb8
 
 
 
9b2289b
ca46a75
 
 
 
7173af9
 
ca46a75
 
 
 
7173af9
 
f8cafb8
 
 
9b2289b
 
ca46a75
 
 
 
 
 
 
 
 
7173af9
 
 
f8cafb8
 
 
 
9b2289b
ca46a75
f8cafb8
434720c
 
 
 
ca46a75
 
 
 
 
 
 
 
7173af9
9b2289b
434720c
7173af9
 
434720c
 
7173af9
434720c
 
7173af9
434720c
7173af9
 
 
434720c
9b2289b
434720c
 
f8cafb8
434720c
 
f8cafb8
7173af9
ca46a75
 
 
 
7173af9
ca46a75
 
7173af9
ca46a75
 
 
7173af9
9b2289b
434720c
 
ca46a75
434720c
 
ca46a75
7173af9
9b2289b
 
434720c
 
9b2289b
 
 
88e96c2
9b2289b
88e96c2
 
9b2289b
 
 
 
 
 
434720c
 
9b2289b
 
434720c
 
ca46a75
 
 
434720c
 
7173af9
9b2289b
ca46a75
 
 
7173af9
ca46a75
 
7173af9
ca46a75
 
9b2289b
434720c
 
 
 
ca46a75
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
#!/usr/bin/env python
# -*- coding: utf-8 -*-
r"""
@DATE: 2024/9/5 16:45
@File: __init__.py
@IDE: pycharm
@Description:
    创建证件照
"""
import numpy as np
from typing import Tuple
import hivision.creator.utils as U
from .context import Context, ContextHandler, Params, Result
from .human_matting import extract_human
from .face_detector import detect_face_mtcnn
from hivision.plugin.beauty.handler import beauty_face
from .photo_adjuster import adjust_photo
import cv2
import time


class IDCreator:
    """
    证件照创建类,包含完整的证件照流程
    """

    def __init__(self):
        # 回调时机
        self.before_all: ContextHandler = None
        """
        在所有处理之前,此时图像已经被 resize 到最大边长为 2000
        """
        self.after_matting: ContextHandler = None
        """
        在抠图之后,ctx.matting_image 被赋值
        """
        self.after_detect: ContextHandler = None
        """
        在人脸检测之后,ctx.face 被赋值,如果为仅换底,则不会执行此回调
        """
        self.after_all: ContextHandler = None
        """
        在所有处理之后,此时 ctx.result 被赋值
        """
        # 处理者
        self.matting_handler: ContextHandler = extract_human
        self.detection_handler: ContextHandler = detect_face_mtcnn
        self.beauty_handler: ContextHandler = beauty_face
        # 上下文
        self.ctx = None

    def __call__(
        self,
        image: np.ndarray,
        size: Tuple[int, int] = (413, 295),
        change_bg_only: bool = False,
        crop_only: bool = False,
        head_measure_ratio: float = 0.2,
        head_height_ratio: float = 0.45,
        head_top_range: float = (0.12, 0.1),
        face: Tuple[int, int, int, int] = None,
        whitening_strength: int = 0,
        brightness_strength: int = 0,
        contrast_strength: int = 0,
        sharpen_strength: int = 0,
        saturation_strength: int = 0,
        face_alignment: bool = False,
    ) -> Result:
        """
        证件照处理函数
        :param image: 输入图像
        :param change_bg_only: 是否只需要抠图
        :param crop_only: 是否只需要裁剪
        :param size: 输出的图像大小(h,w)
        :param head_measure_ratio: 人脸面积与全图面积的期望比值
        :param head_height_ratio: 人脸中心处在全图高度的比例期望值
        :param head_top_range: 头距离顶部的比例(max,min)
        :param face: 人脸坐标
        :param whitening_strength: 美白强度
        :param brightness_strength: 亮度强度
        :param contrast_strength: 对比度强度
        :param sharpen_strength: 锐化强度
        :param align_face: 是否需要人脸矫正

        :return: 返回处理后的证件照和一系列参数
        """
        # 0.初始化上下文
        params = Params(
            size=size,
            change_bg_only=change_bg_only,
            head_measure_ratio=head_measure_ratio,
            head_height_ratio=head_height_ratio,
            head_top_range=head_top_range,
            crop_only=crop_only,
            face=face,
            whitening_strength=whitening_strength,
            brightness_strength=brightness_strength,
            contrast_strength=contrast_strength,
            sharpen_strength=sharpen_strength,
            saturation_strength=saturation_strength,
            face_alignment=face_alignment,
        )


        # 总的开始时间
        total_start_time = time.time()
        
        self.ctx = Context(params)
        ctx = self.ctx
        ctx.processing_image = image
        ctx.processing_image = U.resize_image_esp(
            ctx.processing_image, 2000
        )  # 将输入图片 resize 到最大边长为 2000
        ctx.origin_image = ctx.processing_image.copy()
        self.before_all and self.before_all(ctx)

        # 1. ------------------人像抠图------------------
        # 如果仅裁剪,则不进行抠图
        if not ctx.params.crop_only:
            # 调用抠图工作流
            print("[1]  Start Human Matting...")
            start_matting_time = time.time()
            self.matting_handler(ctx)
            end_matting_time = time.time()
            print(f"[1]  Human Matting Time: {end_matting_time - start_matting_time:.3f}s")
            self.after_matting and self.after_matting(ctx)
        # 如果进行抠图
        else:
            ctx.matting_image = ctx.processing_image


        # 2. ------------------美颜------------------
        print("[2]  Start Beauty...")
        start_beauty_time = time.time()
        self.beauty_handler(ctx)
        end_beauty_time = time.time()
        print(f"[2]  Beauty Time: {end_beauty_time - start_beauty_time:.3f}s")

        # 如果仅换底,则直接返回抠图结果
        if ctx.params.change_bg_only:
            ctx.result = Result(
                standard=ctx.matting_image,
                hd=ctx.matting_image,
                matting=ctx.matting_image,
                clothing_params=None,
                typography_params=None,
                face=None,
            )
            self.after_all and self.after_all(ctx)
            return ctx.result

        # 3. ------------------人脸检测------------------
        print("[3]  Start Face Detection...")
        start_detection_time = time.time()
        self.detection_handler(ctx)
        end_detection_time = time.time()
        print(f"[3]  Face Detection Time: {end_detection_time - start_detection_time:.3f}s")
        self.after_detect and self.after_detect(ctx)

        # 3.1 ------------------人脸对齐------------------
        if ctx.params.face_alignment and abs(ctx.face["roll_angle"]) > 2:
            print("[3.1]  Start Face Alignment...")
            start_alignment_time = time.time()
            from hivision.creator.rotation_adjust import rotate_bound_4channels

            # 根据角度旋转原图和抠图
            b, g, r, a = cv2.split(ctx.matting_image)
            ctx.origin_image, ctx.matting_image, _, _, _, _ = rotate_bound_4channels(
                cv2.merge((b, g, r)),
                a,
                -1 * ctx.face["roll_angle"],
            )

            # 旋转后再执行一遍人脸检测
            self.detection_handler(ctx)
            self.after_detect and self.after_detect(ctx)
            end_alignment_time = time.time()
            print(f"[3.1]  Face Alignment Time: {end_alignment_time - start_alignment_time:.3f}s")

        # 4. ------------------图像调整------------------
        print("[4]  Start Image Post-Adjustment...")
        start_adjust_time = time.time()
        result_image_hd, result_image_standard, clothing_params, typography_params = (
            adjust_photo(ctx)
        )
        end_adjust_time = time.time()
        print(f"[4]  Image Post-Adjustment Time: {end_adjust_time - start_adjust_time:.3f}s")

        # 5. ------------------返回结果------------------
        ctx.result = Result(
            standard=result_image_standard,
            hd=result_image_hd,
            matting=ctx.matting_image,
            clothing_params=clothing_params,
            typography_params=typography_params,
            face=ctx.face,
        )
        self.after_all and self.after_all(ctx)

        # 总的结束时间
        total_end_time = time.time()
        print(f"[Total]  Total Time: {total_end_time - total_start_time:.3f}s")

        return ctx.result