import os import webbrowser import customtkinter as ctk from typing import Callable, Tuple import cv2 from modules.gpu_processing import gpu_cvt_color, gpu_resize, gpu_flip from PIL import Image, ImageOps import time import json import queue import threading import numpy as np import modules.globals import modules.metadata from modules.face_analyser import ( get_one_face, get_many_faces, get_unique_faces_from_target_image, get_unique_faces_from_target_video, add_blank_map, has_valid_map, simplify_maps, ) from modules.capturer import get_video_frame, get_video_frame_total from modules.processors.frame.core import get_frame_processors_modules from modules.utilities import ( is_image, is_video, resolve_relative_path, has_image_extension, ) from modules.video_capture import VideoCapturer from modules.gettext import LanguageManager from modules.ui_tooltip import ToolTip from modules import globals import platform if platform.system() == "Windows": from pygrabber.dshow_graph import FilterGraph # --- Tk 9.0 compatibility patch --- # In Tk 9.0, Menu.index("end") returns "" instead of raising TclError # when the menu is empty. CustomTkinter's CTkOptionMenu doesn't handle # this, causing crashes. This patch adds the missing guard. try: from customtkinter.windows.widgets.core_widget_classes import DropdownMenu as _DropdownMenu _original_add_menu_commands = _DropdownMenu._add_menu_commands def _patched_add_menu_commands(self, *args, **kwargs): try: end_index = self._menu.index("end") if end_index == "" or end_index is None: return except Exception: pass _original_add_menu_commands(self, *args, **kwargs) _DropdownMenu._add_menu_commands = _patched_add_menu_commands except (ImportError, AttributeError): pass # CustomTkinter version doesn't have this class path # --- End Tk 9.0 patch --- ROOT = None POPUP = None POPUP_LIVE = None ROOT_HEIGHT = 800 ROOT_WIDTH = 600 PREVIEW = None PREVIEW_MAX_HEIGHT = 700 PREVIEW_MAX_WIDTH = 1200 PREVIEW_DEFAULT_WIDTH = 960 PREVIEW_DEFAULT_HEIGHT = 540 POPUP_WIDTH = 750 POPUP_HEIGHT = 810 POPUP_SCROLL_WIDTH = (740,) POPUP_SCROLL_HEIGHT = 700 POPUP_LIVE_WIDTH = 900 POPUP_LIVE_HEIGHT = 820 POPUP_LIVE_SCROLL_WIDTH = (890,) POPUP_LIVE_SCROLL_HEIGHT = 700 MAPPER_PREVIEW_MAX_HEIGHT = 100 MAPPER_PREVIEW_MAX_WIDTH = 100 DEFAULT_BUTTON_WIDTH = 200 DEFAULT_BUTTON_HEIGHT = 40 RECENT_DIRECTORY_SOURCE = None RECENT_DIRECTORY_TARGET = None RECENT_DIRECTORY_OUTPUT = None _ = None preview_label = None preview_slider = None source_label = None target_label = None status_label = None popup_status_label = None popup_status_label_live = None source_label_dict = {} source_label_dict_live = {} target_label_dict_live = {} img_ft, vid_ft = modules.globals.file_types def init(start: Callable[[], None], destroy: Callable[[], None], lang: str) -> ctk.CTk: global ROOT, PREVIEW, _ lang_manager = LanguageManager(lang) _ = lang_manager._ ROOT = create_root(start, destroy) PREVIEW = create_preview(ROOT) return ROOT def save_switch_states(): switch_states = { "keep_fps": modules.globals.keep_fps, "keep_audio": modules.globals.keep_audio, "keep_frames": modules.globals.keep_frames, "many_faces": modules.globals.many_faces, "map_faces": modules.globals.map_faces, "poisson_blend": modules.globals.poisson_blend, "color_correction": modules.globals.color_correction, "nsfw_filter": modules.globals.nsfw_filter, "live_mirror": modules.globals.live_mirror, "live_resizable": modules.globals.live_resizable, "fp_ui": modules.globals.fp_ui, "show_fps": modules.globals.show_fps, "mouth_mask": modules.globals.mouth_mask, "show_mouth_mask_box": modules.globals.show_mouth_mask_box, } with open("switch_states.json", "w") as f: json.dump(switch_states, f) def load_switch_states(): try: with open("switch_states.json", "r") as f: switch_states = json.load(f) modules.globals.keep_fps = switch_states.get("keep_fps", True) modules.globals.keep_audio = switch_states.get("keep_audio", True) modules.globals.keep_frames = switch_states.get("keep_frames", False) modules.globals.many_faces = switch_states.get("many_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.nsfw_filter = switch_states.get("nsfw_filter", False) modules.globals.live_mirror = switch_states.get("live_mirror", False) modules.globals.live_resizable = switch_states.get("live_resizable", False) modules.globals.fp_ui = switch_states.get("fp_ui", {"face_enhancer": False}) modules.globals.show_fps = switch_states.get("show_fps", False) modules.globals.mouth_mask = switch_states.get("mouth_mask", False) modules.globals.show_mouth_mask_box = switch_states.get( "show_mouth_mask_box", False ) except FileNotFoundError: # If the file doesn't exist, use default values pass def create_root(start: Callable[[], None], destroy: Callable[[], None]) -> ctk.CTk: global source_label, target_label, status_label, show_fps_switch load_switch_states() ctk.deactivate_automatic_dpi_awareness() ctk.set_appearance_mode("system") ctk.set_default_color_theme(resolve_relative_path("ui.json")) root = ctk.CTk() root.minsize(ROOT_WIDTH, ROOT_HEIGHT) root.title( f"{modules.metadata.name} {modules.metadata.version} {modules.metadata.edition}" ) root.configure() root.protocol("WM_DELETE_WINDOW", lambda: destroy()) source_label = ctk.CTkLabel(root, text=None) source_label.place(relx=0.1, rely=0.05, relwidth=0.275, relheight=0.225) target_label = ctk.CTkLabel(root, text=None) target_label.place(relx=0.6, rely=0.05, relwidth=0.275, relheight=0.225) select_face_button = ctk.CTkButton( root, text=_("Select a face"), cursor="hand2", command=lambda: select_source_path() ) select_face_button.place(relx=0.1, rely=0.30, relwidth=0.3, relheight=0.1) ToolTip(select_face_button, _("Choose the source face image to swap onto the target")) swap_faces_button = ctk.CTkButton( root, text="↔", cursor="hand2", command=lambda: swap_faces_paths() ) swap_faces_button.place(relx=0.45, rely=0.30, relwidth=0.1, relheight=0.1) ToolTip(swap_faces_button, _("Swap source and target images")) select_target_button = ctk.CTkButton( root, text=_("Select a target"), cursor="hand2", command=lambda: select_target_path(), ) select_target_button.place(relx=0.6, rely=0.30, relwidth=0.3, relheight=0.1) ToolTip(select_target_button, _("Choose the target image or video to apply face swap to")) keep_fps_value = ctk.BooleanVar(value=modules.globals.keep_fps) keep_fps_checkbox = ctk.CTkSwitch( root, text=_("Keep fps"), variable=keep_fps_value, cursor="hand2", command=lambda: ( setattr(modules.globals, "keep_fps", keep_fps_value.get()), save_switch_states(), ), ) keep_fps_checkbox.place(relx=0.1, rely=0.5) ToolTip(keep_fps_checkbox, _("Output video keeps the original frame rate")) keep_frames_value = ctk.BooleanVar(value=modules.globals.keep_frames) keep_frames_switch = ctk.CTkSwitch( root, text=_("Keep frames"), variable=keep_frames_value, cursor="hand2", command=lambda: ( setattr(modules.globals, "keep_frames", keep_frames_value.get()), save_switch_states(), ), ) keep_frames_switch.place(relx=0.1, rely=0.55) ToolTip(keep_frames_switch, _("Keep extracted frames on disk after processing")) enhancer_value = ctk.BooleanVar(value=modules.globals.fp_ui["face_enhancer"]) enhancer_switch = ctk.CTkSwitch( root, text=_("Face Enhancer"), variable=enhancer_value, cursor="hand2", command=lambda: ( update_tumbler("face_enhancer", enhancer_value.get()), save_switch_states(), ), ) enhancer_switch.place(relx=0.1, rely=0.6) ToolTip(enhancer_switch, _("Improve face quality using the GFPGAN restoration model")) gpen256_value = ctk.BooleanVar(value=modules.globals.fp_ui.get("face_enhancer_gpen256", False)) gpen256_switch = ctk.CTkSwitch( root, text=_("GPEN Enhancer 256"), variable=gpen256_value, cursor="hand2", command=lambda: ( update_tumbler("face_enhancer_gpen256", gpen256_value.get()), save_switch_states(), ), ) gpen256_switch.place(relx=0.1, rely=0.65) ToolTip(gpen256_switch, _("Use GPEN face enhancement model at 256px resolution (faster)")) gpen512_value = ctk.BooleanVar(value=modules.globals.fp_ui.get("face_enhancer_gpen512", False)) gpen512_switch = ctk.CTkSwitch( root, text=_("GPEN Enhancer 512"), variable=gpen512_value, cursor="hand2", command=lambda: ( update_tumbler("face_enhancer_gpen512", gpen512_value.get()), save_switch_states(), ), ) gpen512_switch.place(relx=0.1, rely=0.7) ToolTip(gpen512_switch, _("Use GPEN face enhancement model at 512px resolution (higher quality)")) keep_audio_value = ctk.BooleanVar(value=modules.globals.keep_audio) keep_audio_switch = ctk.CTkSwitch( root, text=_("Keep audio"), variable=keep_audio_value, cursor="hand2", command=lambda: ( setattr(modules.globals, "keep_audio", keep_audio_value.get()), save_switch_states(), ), ) keep_audio_switch.place(relx=0.6, rely=0.5) ToolTip(keep_audio_switch, _("Copy audio track from the source video to output")) many_faces_value = ctk.BooleanVar(value=modules.globals.many_faces) many_faces_switch = ctk.CTkSwitch( root, text=_("Many faces"), variable=many_faces_value, cursor="hand2", command=lambda: ( setattr(modules.globals, "many_faces", many_faces_value.get()), save_switch_states(), ), ) many_faces_switch.place(relx=0.6, rely=0.55) ToolTip(many_faces_switch, _("Swap every detected face, not just the primary one")) color_correction_value = ctk.BooleanVar(value=modules.globals.color_correction) color_correction_switch = ctk.CTkSwitch( root, text=_("Fix Blueish Cam"), variable=color_correction_value, cursor="hand2", command=lambda: ( setattr(modules.globals, "color_correction", color_correction_value.get()), save_switch_states(), ), ) color_correction_switch.place(relx=0.6, rely=0.6) ToolTip(color_correction_switch, _("Fix blue/green color cast from some webcams")) # nsfw_value = ctk.BooleanVar(value=modules.globals.nsfw_filter) # nsfw_switch = ctk.CTkSwitch(root, text='NSFW filter', variable=nsfw_value, cursor='hand2', command=lambda: setattr(modules.globals, 'nsfw_filter', nsfw_value.get())) # nsfw_switch.place(relx=0.6, rely=0.7) map_faces = ctk.BooleanVar(value=modules.globals.map_faces) map_faces_switch = ctk.CTkSwitch( root, text=_("Map faces"), variable=map_faces, cursor="hand2", command=lambda: ( setattr(modules.globals, "map_faces", map_faces.get()), save_switch_states(), close_mapper_window() if not map_faces.get() else None ), ) map_faces_switch.place(relx=0.1, rely=0.75) ToolTip(map_faces_switch, _("Manually assign which source face maps to which target face")) 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.8) ToolTip(poisson_blend_switch, _("Blend face edges smoothly using Poisson blending")) show_fps_value = ctk.BooleanVar(value=modules.globals.show_fps) show_fps_switch = ctk.CTkSwitch( root, text=_("Show FPS"), variable=show_fps_value, cursor="hand2", command=lambda: ( setattr(modules.globals, "show_fps", show_fps_value.get()), save_switch_states(), ), ) show_fps_switch.place(relx=0.6, rely=0.65) ToolTip(show_fps_switch, _("Display frames-per-second counter on the live preview")) mouth_mask_var = ctk.BooleanVar(value=modules.globals.mouth_mask) mouth_mask_switch = ctk.CTkSwitch( root, text=_("Mouth Mask"), variable=mouth_mask_var, cursor="hand2", command=lambda: setattr(modules.globals, "mouth_mask", mouth_mask_var.get()), ) mouth_mask_switch.place(relx=0.1, rely=0.45) ToolTip(mouth_mask_switch, _("Preserve original mouth movement in the swapped face")) show_mouth_mask_box_var = ctk.BooleanVar(value=modules.globals.show_mouth_mask_box) show_mouth_mask_box_switch = ctk.CTkSwitch( root, text=_("Show Mouth Mask Box"), variable=show_mouth_mask_box_var, cursor="hand2", command=lambda: setattr( modules.globals, "show_mouth_mask_box", show_mouth_mask_box_var.get() ), ) show_mouth_mask_box_switch.place(relx=0.6, rely=0.45) ToolTip(show_mouth_mask_box_switch, _("Display the mouth mask boundary for debugging")) start_button = ctk.CTkButton( root, text=_("Start"), cursor="hand2", command=lambda: analyze_target(start, root) ) start_button.place(relx=0.15, rely=0.86, relwidth=0.2, relheight=0.05) ToolTip(start_button, _("Begin processing the target image/video with selected face")) stop_button = ctk.CTkButton( root, text=_("Destroy"), cursor="hand2", command=lambda: destroy() ) stop_button.place(relx=0.4, rely=0.86, relwidth=0.2, relheight=0.05) ToolTip(stop_button, _("Stop processing and close the application")) preview_button = ctk.CTkButton( root, text=_("Preview"), cursor="hand2", command=lambda: toggle_preview() ) preview_button.place(relx=0.65, rely=0.86, relwidth=0.2, relheight=0.05) ToolTip(preview_button, _("Show/hide a preview of the processed output")) # --- Camera Selection --- camera_label = ctk.CTkLabel(root, text=_("Select Camera:")) camera_label.place(relx=0.1, rely=0.92, relwidth=0.2, relheight=0.05) available_cameras = get_available_cameras() camera_indices, camera_names = available_cameras if not camera_names or camera_names[0] == "No cameras found": camera_variable = ctk.StringVar(value="No cameras found") camera_optionmenu = ctk.CTkOptionMenu( root, variable=camera_variable, values=["No cameras found"], state="disabled", ) else: camera_variable = ctk.StringVar(value=camera_names[0]) camera_optionmenu = ctk.CTkOptionMenu( root, variable=camera_variable, values=camera_names ) camera_optionmenu.place(relx=0.35, rely=0.92, relwidth=0.25, relheight=0.05) ToolTip(camera_optionmenu, _("Select which camera to use for live mode")) live_button = ctk.CTkButton( root, text=_("Live"), cursor="hand2", command=lambda: webcam_preview( root, ( camera_indices[camera_names.index(camera_variable.get())] if camera_names and camera_names[0] != "No cameras found" else None ), ), state=( "normal" if camera_names and camera_names[0] != "No cameras found" else "disabled" ), ) live_button.place(relx=0.65, rely=0.92, relwidth=0.2, relheight=0.05) ToolTip(live_button, _("Start real-time face swap using webcam")) # --- End Camera Selection --- # 1) Define a DoubleVar for transparency (0 = fully transparent, 1 = fully opaque) transparency_var = ctk.DoubleVar(value=1.0) def on_transparency_change(value: float): # Convert slider value to float val = float(value) modules.globals.opacity = val # Set global opacity percentage = int(val * 100) if percentage == 0: modules.globals.fp_ui["face_enhancer"] = False update_status("Transparency set to 0% - Face swapping disabled.") elif percentage == 100: modules.globals.face_swapper_enabled = True update_status("Transparency set to 100%.") else: modules.globals.face_swapper_enabled = True update_status(f"Transparency set to {percentage}%") # 2) Transparency label and slider (placed ABOVE sharpness) transparency_label = ctk.CTkLabel(root, text="Transparency:") transparency_label.place(relx=0.15, rely=0.75, relwidth=0.2, relheight=0.05) transparency_slider = ctk.CTkSlider( root, from_=0.0, to=1.0, variable=transparency_var, command=on_transparency_change, fg_color="#E0E0E0", progress_color="#007BFF", button_color="#FFFFFF", button_hover_color="#CCCCCC", height=5, border_width=1, corner_radius=3, ) transparency_slider.place(relx=0.35, rely=0.77, relwidth=0.5, relheight=0.02) ToolTip(transparency_slider, _("Blend between original and swapped face (0% = original, 100% = fully swapped)")) # 3) Sharpness label & slider sharpness_var = ctk.DoubleVar(value=0.0) # start at 0.0 def on_sharpness_change(value: float): modules.globals.sharpness = float(value) update_status(f"Sharpness set to {value:.1f}") sharpness_label = ctk.CTkLabel(root, text="Sharpness:") sharpness_label.place(relx=0.15, rely=0.80, relwidth=0.2, relheight=0.05) sharpness_slider = ctk.CTkSlider( root, from_=0.0, to=5.0, variable=sharpness_var, command=on_sharpness_change, fg_color="#E0E0E0", progress_color="#007BFF", button_color="#FFFFFF", button_hover_color="#CCCCCC", height=5, border_width=1, corner_radius=3, ) sharpness_slider.place(relx=0.35, rely=0.82, relwidth=0.5, relheight=0.02) ToolTip(sharpness_slider, _("Sharpen the enhanced face output")) # Status and link at the bottom global status_label status_label = ctk.CTkLabel(root, text=None, justify="center") status_label.place(relx=0.1, rely=0.96, relwidth=0.8) donate_label = ctk.CTkLabel( root, text="Deep Live Cam", justify="center", cursor="hand2" ) donate_label.place(relx=0.1, rely=0.98, relwidth=0.8) donate_label.configure( text_color=ctk.ThemeManager.theme.get("URL").get("text_color") ) donate_label.bind( "