|
|
import math |
|
|
import time |
|
|
|
|
|
import cv2 |
|
|
import numpy as np |
|
|
|
|
|
import onnx |
|
|
import onnxruntime |
|
|
|
|
|
def create_onnx_session(onnx_path, providers) -> onnxruntime.InferenceSession: |
|
|
start = time.perf_counter() |
|
|
onnx_model = onnx.load(onnx_path) |
|
|
onnx.checker.check_model(onnx_model) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sess = onnxruntime.InferenceSession(onnx_path, providers=providers) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return sess |
|
|
|
|
|
|
|
|
def smoothing_factor(t_e, cutoff): |
|
|
r = 2 * math.pi * cutoff * t_e |
|
|
return r / (r + 1) |
|
|
|
|
|
|
|
|
def exponential_smoothing(a, x, x_prev): |
|
|
return a * x + (1 - a) * x_prev |
|
|
|
|
|
|
|
|
class OneEuroFilter: |
|
|
def __init__(self, dx0=0.0, d_cutoff=1.0): |
|
|
"""Initialize the one euro filter.""" |
|
|
|
|
|
|
|
|
self.d_cutoff = float(d_cutoff) |
|
|
self.dx_prev = float(dx0) |
|
|
|
|
|
|
|
|
def __call__(self, x, x_prev, fcmin=1.0, min_cutoff=1.0, beta=0.0): |
|
|
if x_prev is None: |
|
|
return x |
|
|
|
|
|
a_d = smoothing_factor(fcmin, self.d_cutoff) |
|
|
dx = (x - x_prev) / fcmin |
|
|
dx_hat = exponential_smoothing(a_d, dx, self.dx_prev) |
|
|
cutoff = min_cutoff + beta * abs(dx_hat) |
|
|
a = smoothing_factor(fcmin, cutoff) |
|
|
x_hat = exponential_smoothing(a, x, x_prev) |
|
|
self.dx_prev = dx_hat |
|
|
return x_hat |
|
|
|
|
|
|
|
|
def get_warp_mat_bbox( |
|
|
face_bbox, base_angle, dst_size=128, expand_ratio=0.15, aug_angle=0.0, aug_scale=1.0 |
|
|
): |
|
|
face_x_min, face_y_min, face_x_max, face_y_max = face_bbox |
|
|
face_x_center = (face_x_min + face_x_max) / 2 |
|
|
face_y_center = (face_y_min + face_y_max) / 2 |
|
|
face_width = face_x_max - face_x_min |
|
|
face_height = face_y_max - face_y_min |
|
|
scale = dst_size / max(face_width, face_height) * (1 - expand_ratio) * aug_scale |
|
|
M = cv2.getRotationMatrix2D( |
|
|
(face_x_center, face_y_center), angle=base_angle + aug_angle, scale=scale |
|
|
) |
|
|
offset = [dst_size / 2 - face_x_center, dst_size / 2 - face_y_center] |
|
|
M[:, 2] += offset |
|
|
return M |
|
|
|
|
|
|
|
|
def transform_points(points, mat, invert=False): |
|
|
if invert: |
|
|
mat = cv2.invertAffineTransform(mat) |
|
|
points = np.expand_dims(points, axis=1) |
|
|
points = cv2.transform(points, mat, points.shape) |
|
|
points = np.squeeze(points) |
|
|
return points |
|
|
|
|
|
|
|
|
def get_warp_mat_bbox_by_gt_pts_float( |
|
|
gt_pts, base_angle=0.0, dst_size=128, expand_ratio=0.15, return_info=False |
|
|
): |
|
|
|
|
|
face_x_min, face_x_max = np.min(gt_pts[:, 0]), np.max(gt_pts[:, 0]) |
|
|
face_y_min, face_y_max = np.min(gt_pts[:, 1]), np.max(gt_pts[:, 1]) |
|
|
face_x_center = (face_x_min + face_x_max) / 2 |
|
|
face_y_center = (face_y_min + face_y_max) / 2 |
|
|
M_step_1 = cv2.getRotationMatrix2D( |
|
|
(face_x_center, face_y_center), angle=base_angle, scale=1.0 |
|
|
) |
|
|
pts_step_1 = transform_points(gt_pts, M_step_1) |
|
|
face_x_min_step_1, face_x_max_step_1 = np.min(pts_step_1[:, 0]), np.max( |
|
|
pts_step_1[:, 0] |
|
|
) |
|
|
face_y_min_step_1, face_y_max_step_1 = np.min(pts_step_1[:, 1]), np.max( |
|
|
pts_step_1[:, 1] |
|
|
) |
|
|
|
|
|
face_width = face_x_max_step_1 - face_x_min_step_1 |
|
|
face_height = face_y_max_step_1 - face_y_min_step_1 |
|
|
scale = dst_size / max(face_width, face_height) * (1 - expand_ratio) |
|
|
M_step_2 = cv2.getRotationMatrix2D( |
|
|
(face_x_center, face_y_center), angle=base_angle, scale=scale |
|
|
) |
|
|
pts_step_2 = transform_points(gt_pts, M_step_2) |
|
|
face_x_min_step_2, face_x_max_step_2 = np.min(pts_step_2[:, 0]), np.max( |
|
|
pts_step_2[:, 0] |
|
|
) |
|
|
face_y_min_step_2, face_y_max_step_2 = np.min(pts_step_2[:, 1]), np.max( |
|
|
pts_step_2[:, 1] |
|
|
) |
|
|
face_x_center_step_2 = (face_x_min_step_2 + face_x_max_step_2) / 2 |
|
|
face_y_center_step_2 = (face_y_min_step_2 + face_y_max_step_2) / 2 |
|
|
|
|
|
M = cv2.getRotationMatrix2D( |
|
|
(face_x_center, face_y_center), angle=base_angle, scale=scale |
|
|
) |
|
|
offset = [dst_size / 2 - face_x_center_step_2, dst_size / 2 - face_y_center_step_2] |
|
|
M[:, 2] += offset |
|
|
|
|
|
if not return_info: |
|
|
return M |
|
|
else: |
|
|
transform_info = { |
|
|
"M": M, |
|
|
"center_x": face_x_center, |
|
|
"center_y": face_y_center, |
|
|
"rotate_angle": base_angle, |
|
|
"scale": scale, |
|
|
} |
|
|
return transform_info |
|
|
|