mirror of
https://github.com/zebrajr/faceswap.git
synced 2026-01-15 12:15:15 +00:00
* FaceSwap 3 (#1515) * Update extract pipeline * Update requirements + setup for nvidia * Remove allow-growth option * tf.keras to keras updates * lib.model.losses - Port + fix all loss functions for Keras3 * lib.model - port initializers, layers. normalization to Keras3 * lib.model.autoclip to Keras 3 * Update mixed precision layer storage * model file to .keras format * Restructure nn_blocks to initialize layers in __init__ * Tensorboard - Trainer: Add Torch compatible Tensorboard callbacks - GUI event reader remove TF dependency * Loss logging - Flush TB logs on save - Replace TB live iterator for GUI * Backup models on total loss drop rather than per side * Update all models to Keras3 Compat * Remove lib.model.session * Update clip ViT to Keras 3 * plugins.extract.mask.unet-dfl - Fix for Keras3/Torch backend * Port AdaBelief to Keras 3 * setup.py: - Add --dev flag for dev tool install * Fix Keras 3 syntax * Fix LR Finder for Keras 3 * Fix mixed precision switching for Keras 3 * Add more optimizers + open up config setting * train: Remove updating FS1 weights to FS2 models * Alignments: Remove support for legacy .json files * tools.model: - Remove TF Saved Format saving - Fix Backup/Restore + Nan-Scan * Fix inference model creation for Keras 3 * Preview tool: Fix for Keras3 * setup.py: Configure keras backend * train: Migration of FS2 models to FS3 * Training: Default coverage to 100% * Remove DirectML backend * Update setup for MacOS * GUI: Force line reading to UTF-8 * Remove redundant Tensorflow references * Remove redundant code * Legacy model loading: Fix TFLamdaOp scalar ops and DepthwiseConv2D * Add vertical offset option for training * Github actions: Add more python versions * Add python version to workflow names * Github workflow: Exclude Python 3.12 for macOS * Implement custom training loop * Fs3 - Add RTX5xxx and ROCm 6.1-6.4 support (#1511) * setup.py: Add Cuda/ROCm version select options * bump minimum python version to 3.11 * Switch from setup.cgf to pyproject.toml * Documentation: Update all docs to use automodapi * Allow sysinfo to run with missing packages + correctly install tk under Linux * Bugfix: dot naming convention in clip models * lib.config: Centralise globally rather than passing as object - Add torch DataParallel for multi-gpu training - GUI: Group switches together when generating cli args - CLI: Remove deprecated multi-character argparse args - Refactor: - Centralise tensorboard reading/writing + unit tests - Create trainer plugin interfaces + add original + distributed * Update installers
197 lines
7.0 KiB
Python
197 lines
7.0 KiB
Python
#!/usr/bin python3
|
|
""" The optional GUI for faceswap """
|
|
|
|
import logging
|
|
import sys
|
|
import tkinter as tk
|
|
from tkinter import messagebox, ttk
|
|
|
|
from lib.gui import (TaskBar, CliOptions, CommandNotebook, ConsoleOut, DisplayNotebook,
|
|
get_images, gui_config as cfg, initialize_images, initialize_config,
|
|
LastSession, MainMenuBar, preview_trigger, ProcessWrapper, StatusBar)
|
|
from lib.utils import get_module_objects
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
class FaceswapGui(tk.Tk):
|
|
""" The Graphical User Interface
|
|
|
|
Launch the Faceswap GUI
|
|
|
|
Parameters
|
|
----------
|
|
debug : bool
|
|
Output to the terminal rather than to Faceswap's internal console
|
|
config_file : str | None
|
|
Path to a custom .ini configuration file. ``None`` to use the default config file
|
|
"""
|
|
|
|
def __init__(self, debug, config_file):
|
|
logger.debug("Initializing %s", self.__class__.__name__)
|
|
super().__init__()
|
|
cfg.load_config(config_file)
|
|
|
|
self._init_args = {"debug": debug}
|
|
self._config = self.initialize_globals()
|
|
self.set_fonts()
|
|
self._config.set_geometry(1200, 640, cfg.fullscreen())
|
|
|
|
self.wrapper = ProcessWrapper()
|
|
self.objects = {}
|
|
|
|
get_images().delete_preview()
|
|
preview_trigger().clear(trigger_type=None)
|
|
self.protocol("WM_DELETE_WINDOW", self.close_app)
|
|
self.build_gui()
|
|
self._last_session = LastSession(self._config)
|
|
logger.debug("Initialized %s", self.__class__.__name__)
|
|
|
|
def initialize_globals(self):
|
|
""" Initialize config and images global constants """
|
|
cliopts = CliOptions()
|
|
statusbar = StatusBar(self)
|
|
config = initialize_config(self, cliopts, statusbar)
|
|
initialize_images()
|
|
return config
|
|
|
|
def set_fonts(self):
|
|
""" Set global default font """
|
|
tk.font.nametofont("TkFixedFont").configure(size=self._config.default_font[1])
|
|
for font in ("TkDefaultFont", "TkHeadingFont", "TkMenuFont"):
|
|
tk.font.nametofont(font).configure(family=self._config.default_font[0],
|
|
size=self._config.default_font[1])
|
|
|
|
def build_gui(self, rebuild=False):
|
|
""" Build the GUI """
|
|
logger.debug("Building GUI")
|
|
if not rebuild:
|
|
self.tk.call('wm', 'iconphoto', self._w, get_images().icons["favicon"])
|
|
self.configure(menu=MainMenuBar(self))
|
|
|
|
if rebuild:
|
|
objects = list(self.objects.keys())
|
|
for obj in objects:
|
|
self.objects[obj].destroy()
|
|
del self.objects[obj]
|
|
|
|
self.objects["taskbar"] = TaskBar(self)
|
|
self.add_containers()
|
|
|
|
self.objects["command"] = CommandNotebook(self.objects["container_top"])
|
|
self.objects["display"] = DisplayNotebook(self.objects["container_top"])
|
|
self.objects["console"] = ConsoleOut(self.objects["container_bottom"],
|
|
self._init_args["debug"])
|
|
self.set_initial_focus()
|
|
self.set_layout()
|
|
self._config.set_default_options()
|
|
logger.debug("Built GUI")
|
|
|
|
def add_containers(self):
|
|
""" Add the paned window containers that
|
|
hold each main area of the gui """
|
|
logger.debug("Adding containers")
|
|
maincontainer = ttk.PanedWindow(self,
|
|
orient=tk.VERTICAL,
|
|
name="pw_main")
|
|
maincontainer.pack(fill=tk.BOTH, expand=True)
|
|
|
|
topcontainer = ttk.PanedWindow(maincontainer,
|
|
orient=tk.HORIZONTAL,
|
|
name="pw_top")
|
|
maincontainer.add(topcontainer)
|
|
|
|
bottomcontainer = ttk.Frame(maincontainer, name="frame_bottom")
|
|
maincontainer.add(bottomcontainer)
|
|
self.objects["container_main"] = maincontainer
|
|
self.objects["container_top"] = topcontainer
|
|
self.objects["container_bottom"] = bottomcontainer
|
|
|
|
logger.debug("Added containers")
|
|
|
|
def set_initial_focus(self):
|
|
""" Set the tab focus from settings """
|
|
tab = cfg.tab()
|
|
logger.debug("Setting focus for tab: %s", tab)
|
|
self._config.set_active_tab_by_name(tab)
|
|
logger.debug("Focus set to: %s", tab)
|
|
|
|
def set_layout(self):
|
|
""" Set initial layout """
|
|
self.update_idletasks()
|
|
r_width = self.winfo_width()
|
|
r_height = self.winfo_height()
|
|
w_ratio = cfg.options_panel_width() / 100.0
|
|
h_ratio = 1 - (cfg.console_panel_height() / 100.0)
|
|
width = round(r_width * w_ratio)
|
|
height = round(r_height * h_ratio)
|
|
logger.debug("Setting Initial Layout: (root_width: %s, root_height: %s, width_ratio: %s, "
|
|
"height_ratio: %s, width: %s, height: %s", r_width, r_height, w_ratio,
|
|
h_ratio, width, height)
|
|
self.objects["container_top"].sashpos(0, width)
|
|
self.objects["container_main"].sashpos(0, height)
|
|
self.update_idletasks()
|
|
|
|
def rebuild(self):
|
|
""" Rebuild the GUI on config change """
|
|
logger.debug("Redrawing GUI")
|
|
session_state = self._last_session.to_dict()
|
|
get_images().__init__() # pylint:disable=unnecessary-dunder-call
|
|
self.set_fonts()
|
|
self.build_gui(rebuild=True)
|
|
if session_state is not None:
|
|
self._last_session.from_dict(session_state)
|
|
logger.debug("GUI Redrawn")
|
|
|
|
def close_app(self, *args): # pylint:disable=unused-argument
|
|
""" Close Python. This is here because the graph
|
|
animation function continues to run even when
|
|
tkinter has gone away """
|
|
logger.debug("Close Requested")
|
|
|
|
if not self._confirm_close_on_running_task():
|
|
return
|
|
if not self._config.project.confirm_close():
|
|
return
|
|
|
|
if self._config.tk_vars.running_task.get():
|
|
self.wrapper.task.terminate()
|
|
|
|
self._last_session.save()
|
|
get_images().delete_preview()
|
|
preview_trigger().clear(trigger_type=None)
|
|
self.quit()
|
|
logger.debug("Closed GUI")
|
|
sys.exit(0)
|
|
|
|
def _confirm_close_on_running_task(self):
|
|
""" Pop a confirmation box to close the GUI if a task is running
|
|
|
|
Returns
|
|
-------
|
|
bool: ``True`` if user confirms close, ``False`` if user cancels close
|
|
"""
|
|
if not self._config.tk_vars.running_task.get():
|
|
logger.debug("No tasks currently running")
|
|
return True
|
|
|
|
confirmtxt = "Processes are still running.\n\nAre you sure you want to exit?"
|
|
if not messagebox.askokcancel("Close", confirmtxt, default="cancel", icon="warning"):
|
|
logger.debug("Close Cancelled")
|
|
return False
|
|
logger.debug("Close confirmed")
|
|
return True
|
|
|
|
|
|
class Gui():
|
|
""" The GUI process. """
|
|
def __init__(self, arguments):
|
|
self.root = FaceswapGui(arguments.debug, arguments.configfile)
|
|
|
|
def process(self):
|
|
""" Builds the GUI """
|
|
self.root.mainloop()
|
|
|
|
|
|
__all__ = get_module_objects(__name__)
|