Adds Poisson blending
- adds poisson blending on the face to make a seamless blending of the face and the swapped image removing the "frame" - adds the switch on the UI Advance Merry Christmas everyone!
This commit is contained in:
@@ -27,6 +27,7 @@ keep_audio: bool = True
|
|||||||
keep_frames: bool = False
|
keep_frames: bool = False
|
||||||
many_faces: bool = False # Process all detected faces with default source
|
many_faces: bool = False # Process all detected faces with default source
|
||||||
map_faces: bool = False # Use source_target_map or simple_map for specific swaps
|
map_faces: bool = False # Use source_target_map or simple_map for specific swaps
|
||||||
|
poisson_blend: bool = False # Enable Poisson Blending for smoother face swaps
|
||||||
color_correction: bool = False # Enable color correction (implementation specific)
|
color_correction: bool = False # Enable color correction (implementation specific)
|
||||||
nsfw_filter: bool = False
|
nsfw_filter: bool = False
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,3 @@
|
|||||||
name = 'Deep-Live-Cam'
|
name = 'Deep-Live-Cam'
|
||||||
version = '2.0c'
|
version = '2.0.1c'
|
||||||
edition = 'GitHub Edition'
|
edition = 'GitHub Edition'
|
||||||
|
|||||||
@@ -194,6 +194,35 @@ def swap_face(source_face: Face, target_face: Face, temp_frame: Frame) -> Frame:
|
|||||||
swapped_frame, target_face, mouth_mask_data
|
swapped_frame, target_face, mouth_mask_data
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# --- Poisson Blending ---
|
||||||
|
if getattr(modules.globals, "poisson_blend", False):
|
||||||
|
face_mask = create_face_mask(target_face, temp_frame)
|
||||||
|
if face_mask is not None:
|
||||||
|
# Find bounding box of the mask
|
||||||
|
y_indices, x_indices = np.where(face_mask > 0)
|
||||||
|
if len(x_indices) > 0 and len(y_indices) > 0:
|
||||||
|
x_min, x_max = np.min(x_indices), np.max(x_indices)
|
||||||
|
y_min, y_max = np.min(y_indices), np.max(y_indices)
|
||||||
|
|
||||||
|
# Calculate center
|
||||||
|
center = (int((x_min + x_max) / 2), int((y_min + y_max) / 2))
|
||||||
|
|
||||||
|
# Crop src and mask
|
||||||
|
src_crop = swapped_frame[y_min : y_max + 1, x_min : x_max + 1]
|
||||||
|
mask_crop = face_mask[y_min : y_max + 1, x_min : x_max + 1]
|
||||||
|
|
||||||
|
try:
|
||||||
|
# Use original_frame as destination to blend the swapped face onto it
|
||||||
|
swapped_frame = cv2.seamlessClone(
|
||||||
|
src_crop,
|
||||||
|
original_frame,
|
||||||
|
mask_crop,
|
||||||
|
center,
|
||||||
|
cv2.NORMAL_CLONE,
|
||||||
|
)
|
||||||
|
except Exception as e:
|
||||||
|
print(f"Poisson blending failed: {e}")
|
||||||
|
|
||||||
# Apply opacity blend between the original frame and the swapped frame
|
# Apply opacity blend between the original frame and the swapped frame
|
||||||
opacity = getattr(modules.globals, "opacity", 1.0)
|
opacity = getattr(modules.globals, "opacity", 1.0)
|
||||||
# Ensure opacity is within valid range [0.0, 1.0]
|
# Ensure opacity is within valid range [0.0, 1.0]
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ if platform.system() == "Windows":
|
|||||||
ROOT = None
|
ROOT = None
|
||||||
POPUP = None
|
POPUP = None
|
||||||
POPUP_LIVE = None
|
POPUP_LIVE = None
|
||||||
ROOT_HEIGHT = 750
|
ROOT_HEIGHT = 800
|
||||||
ROOT_WIDTH = 600
|
ROOT_WIDTH = 600
|
||||||
|
|
||||||
PREVIEW = None
|
PREVIEW = None
|
||||||
@@ -98,6 +98,7 @@ def save_switch_states():
|
|||||||
"keep_frames": modules.globals.keep_frames,
|
"keep_frames": modules.globals.keep_frames,
|
||||||
"many_faces": modules.globals.many_faces,
|
"many_faces": modules.globals.many_faces,
|
||||||
"map_faces": modules.globals.map_faces,
|
"map_faces": modules.globals.map_faces,
|
||||||
|
"poisson_blend": modules.globals.poisson_blend,
|
||||||
"color_correction": modules.globals.color_correction,
|
"color_correction": modules.globals.color_correction,
|
||||||
"nsfw_filter": modules.globals.nsfw_filter,
|
"nsfw_filter": modules.globals.nsfw_filter,
|
||||||
"live_mirror": modules.globals.live_mirror,
|
"live_mirror": modules.globals.live_mirror,
|
||||||
@@ -120,6 +121,7 @@ def load_switch_states():
|
|||||||
modules.globals.keep_frames = switch_states.get("keep_frames", False)
|
modules.globals.keep_frames = switch_states.get("keep_frames", False)
|
||||||
modules.globals.many_faces = switch_states.get("many_faces", False)
|
modules.globals.many_faces = switch_states.get("many_faces", False)
|
||||||
modules.globals.map_faces = switch_states.get("map_faces", False)
|
modules.globals.map_faces = switch_states.get("map_faces", False)
|
||||||
|
modules.globals.poisson_blend = switch_states.get("poisson_blend", False)
|
||||||
modules.globals.color_correction = switch_states.get("color_correction", False)
|
modules.globals.color_correction = switch_states.get("color_correction", False)
|
||||||
modules.globals.nsfw_filter = switch_states.get("nsfw_filter", False)
|
modules.globals.nsfw_filter = switch_states.get("nsfw_filter", False)
|
||||||
modules.globals.live_mirror = switch_states.get("live_mirror", False)
|
modules.globals.live_mirror = switch_states.get("live_mirror", False)
|
||||||
@@ -272,6 +274,19 @@ def create_root(start: Callable[[], None], destroy: Callable[[], None]) -> ctk.C
|
|||||||
)
|
)
|
||||||
map_faces_switch.place(relx=0.1, rely=0.65)
|
map_faces_switch.place(relx=0.1, rely=0.65)
|
||||||
|
|
||||||
|
poisson_blend_value = ctk.BooleanVar(value=modules.globals.poisson_blend)
|
||||||
|
poisson_blend_switch = ctk.CTkSwitch(
|
||||||
|
root,
|
||||||
|
text=_("Poisson Blend"),
|
||||||
|
variable=poisson_blend_value,
|
||||||
|
cursor="hand2",
|
||||||
|
command=lambda: (
|
||||||
|
setattr(modules.globals, "poisson_blend", poisson_blend_value.get()),
|
||||||
|
save_switch_states(),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
poisson_blend_switch.place(relx=0.1, rely=0.7)
|
||||||
|
|
||||||
show_fps_value = ctk.BooleanVar(value=modules.globals.show_fps)
|
show_fps_value = ctk.BooleanVar(value=modules.globals.show_fps)
|
||||||
show_fps_switch = ctk.CTkSwitch(
|
show_fps_switch = ctk.CTkSwitch(
|
||||||
root,
|
root,
|
||||||
@@ -310,21 +325,21 @@ def create_root(start: Callable[[], None], destroy: Callable[[], None]) -> ctk.C
|
|||||||
start_button = ctk.CTkButton(
|
start_button = ctk.CTkButton(
|
||||||
root, text=_("Start"), cursor="hand2", command=lambda: analyze_target(start, root)
|
root, text=_("Start"), cursor="hand2", command=lambda: analyze_target(start, root)
|
||||||
)
|
)
|
||||||
start_button.place(relx=0.15, rely=0.80, relwidth=0.2, relheight=0.05)
|
start_button.place(relx=0.15, rely=0.86, relwidth=0.2, relheight=0.05)
|
||||||
|
|
||||||
stop_button = ctk.CTkButton(
|
stop_button = ctk.CTkButton(
|
||||||
root, text=_("Destroy"), cursor="hand2", command=lambda: destroy()
|
root, text=_("Destroy"), cursor="hand2", command=lambda: destroy()
|
||||||
)
|
)
|
||||||
stop_button.place(relx=0.4, rely=0.80, relwidth=0.2, relheight=0.05)
|
stop_button.place(relx=0.4, rely=0.86, relwidth=0.2, relheight=0.05)
|
||||||
|
|
||||||
preview_button = ctk.CTkButton(
|
preview_button = ctk.CTkButton(
|
||||||
root, text=_("Preview"), cursor="hand2", command=lambda: toggle_preview()
|
root, text=_("Preview"), cursor="hand2", command=lambda: toggle_preview()
|
||||||
)
|
)
|
||||||
preview_button.place(relx=0.65, rely=0.80, relwidth=0.2, relheight=0.05)
|
preview_button.place(relx=0.65, rely=0.86, relwidth=0.2, relheight=0.05)
|
||||||
|
|
||||||
# --- Camera Selection ---
|
# --- Camera Selection ---
|
||||||
camera_label = ctk.CTkLabel(root, text=_("Select Camera:"))
|
camera_label = ctk.CTkLabel(root, text=_("Select Camera:"))
|
||||||
camera_label.place(relx=0.1, rely=0.86, relwidth=0.2, relheight=0.05)
|
camera_label.place(relx=0.1, rely=0.92, relwidth=0.2, relheight=0.05)
|
||||||
|
|
||||||
available_cameras = get_available_cameras()
|
available_cameras = get_available_cameras()
|
||||||
camera_indices, camera_names = available_cameras
|
camera_indices, camera_names = available_cameras
|
||||||
@@ -343,7 +358,7 @@ def create_root(start: Callable[[], None], destroy: Callable[[], None]) -> ctk.C
|
|||||||
root, variable=camera_variable, values=camera_names
|
root, variable=camera_variable, values=camera_names
|
||||||
)
|
)
|
||||||
|
|
||||||
camera_optionmenu.place(relx=0.35, rely=0.86, relwidth=0.25, relheight=0.05)
|
camera_optionmenu.place(relx=0.35, rely=0.92, relwidth=0.25, relheight=0.05)
|
||||||
|
|
||||||
live_button = ctk.CTkButton(
|
live_button = ctk.CTkButton(
|
||||||
root,
|
root,
|
||||||
@@ -363,7 +378,7 @@ def create_root(start: Callable[[], None], destroy: Callable[[], None]) -> ctk.C
|
|||||||
else "disabled"
|
else "disabled"
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
live_button.place(relx=0.65, rely=0.86, relwidth=0.2, relheight=0.05)
|
live_button.place(relx=0.65, rely=0.92, relwidth=0.2, relheight=0.05)
|
||||||
# --- End Camera Selection ---
|
# --- End Camera Selection ---
|
||||||
|
|
||||||
# 1) Define a DoubleVar for transparency (0 = fully transparent, 1 = fully opaque)
|
# 1) Define a DoubleVar for transparency (0 = fully transparent, 1 = fully opaque)
|
||||||
@@ -387,7 +402,7 @@ def create_root(start: Callable[[], None], destroy: Callable[[], None]) -> ctk.C
|
|||||||
|
|
||||||
# 2) Transparency label and slider (placed ABOVE sharpness)
|
# 2) Transparency label and slider (placed ABOVE sharpness)
|
||||||
transparency_label = ctk.CTkLabel(root, text="Transparency:")
|
transparency_label = ctk.CTkLabel(root, text="Transparency:")
|
||||||
transparency_label.place(relx=0.15, rely=0.69, relwidth=0.2, relheight=0.05)
|
transparency_label.place(relx=0.15, rely=0.75, relwidth=0.2, relheight=0.05)
|
||||||
|
|
||||||
transparency_slider = ctk.CTkSlider(
|
transparency_slider = ctk.CTkSlider(
|
||||||
root,
|
root,
|
||||||
@@ -403,7 +418,7 @@ def create_root(start: Callable[[], None], destroy: Callable[[], None]) -> ctk.C
|
|||||||
border_width=1,
|
border_width=1,
|
||||||
corner_radius=3,
|
corner_radius=3,
|
||||||
)
|
)
|
||||||
transparency_slider.place(relx=0.35, rely=0.71, relwidth=0.5, relheight=0.02)
|
transparency_slider.place(relx=0.35, rely=0.77, relwidth=0.5, relheight=0.02)
|
||||||
|
|
||||||
# 3) Sharpness label & slider
|
# 3) Sharpness label & slider
|
||||||
sharpness_var = ctk.DoubleVar(value=0.0) # start at 0.0
|
sharpness_var = ctk.DoubleVar(value=0.0) # start at 0.0
|
||||||
@@ -412,7 +427,7 @@ def create_root(start: Callable[[], None], destroy: Callable[[], None]) -> ctk.C
|
|||||||
update_status(f"Sharpness set to {value:.1f}")
|
update_status(f"Sharpness set to {value:.1f}")
|
||||||
|
|
||||||
sharpness_label = ctk.CTkLabel(root, text="Sharpness:")
|
sharpness_label = ctk.CTkLabel(root, text="Sharpness:")
|
||||||
sharpness_label.place(relx=0.15, rely=0.74, relwidth=0.2, relheight=0.05)
|
sharpness_label.place(relx=0.15, rely=0.80, relwidth=0.2, relheight=0.05)
|
||||||
|
|
||||||
sharpness_slider = ctk.CTkSlider(
|
sharpness_slider = ctk.CTkSlider(
|
||||||
root,
|
root,
|
||||||
@@ -428,17 +443,17 @@ def create_root(start: Callable[[], None], destroy: Callable[[], None]) -> ctk.C
|
|||||||
border_width=1,
|
border_width=1,
|
||||||
corner_radius=3,
|
corner_radius=3,
|
||||||
)
|
)
|
||||||
sharpness_slider.place(relx=0.35, rely=0.76, relwidth=0.5, relheight=0.02)
|
sharpness_slider.place(relx=0.35, rely=0.82, relwidth=0.5, relheight=0.02)
|
||||||
|
|
||||||
# Status and link at the bottom
|
# Status and link at the bottom
|
||||||
global status_label
|
global status_label
|
||||||
status_label = ctk.CTkLabel(root, text=None, justify="center")
|
status_label = ctk.CTkLabel(root, text=None, justify="center")
|
||||||
status_label.place(relx=0.1, rely=0.9, relwidth=0.8)
|
status_label.place(relx=0.1, rely=0.96, relwidth=0.8)
|
||||||
|
|
||||||
donate_label = ctk.CTkLabel(
|
donate_label = ctk.CTkLabel(
|
||||||
root, text="Deep Live Cam", justify="center", cursor="hand2"
|
root, text="Deep Live Cam", justify="center", cursor="hand2"
|
||||||
)
|
)
|
||||||
donate_label.place(relx=0.1, rely=0.95, relwidth=0.8)
|
donate_label.place(relx=0.1, rely=0.98, relwidth=0.8)
|
||||||
donate_label.configure(
|
donate_label.configure(
|
||||||
text_color=ctk.ThemeManager.theme.get("URL").get("text_color")
|
text_color=ctk.ThemeManager.theme.get("URL").get("text_color")
|
||||||
)
|
)
|
||||||
|
|||||||
Reference in New Issue
Block a user