Compare commits
	
		
			11 Commits 
		
	
	
		
			58797eaf15
			...
			8336467c77
		
	
	| Author | SHA1 | Date | 
|---|---|---|
|  | 8336467c77 | |
|  | f3e83b985c | |
|  | e3e3638b79 | |
|  | c1a6dc693d | |
|  | 75122da389 | |
|  | 7063bba4b3 | |
|  | bdbd7dcfbc | |
|  | a64940def7 | |
|  | fe4a87e8f2 | |
|  | 9ecd2dab83 | |
|  | c9f36eb350 | 
|  | @ -0,0 +1,46 @@ | ||||||
|  | { | ||||||
|  |     "Source x Target Mapper": "Quelle x Ziel Zuordnung", | ||||||
|  |     "select a source image": "Wähle ein Quellbild", | ||||||
|  |     "Preview": "Vorschau", | ||||||
|  |     "select a target image or video": "Wähle ein Zielbild oder Video", | ||||||
|  |     "save image output file": "Bildausgabedatei speichern", | ||||||
|  |     "save video output file": "Videoausgabedatei speichern", | ||||||
|  |     "select a target image": "Wähle ein Zielbild", | ||||||
|  |     "source": "Quelle", | ||||||
|  |     "Select a target": "Wähle ein Ziel", | ||||||
|  |     "Select a face": "Wähle ein Gesicht", | ||||||
|  |     "Keep audio": "Audio beibehalten", | ||||||
|  |     "Face Enhancer": "Gesichtsverbesserung", | ||||||
|  |     "Many faces": "Mehrere Gesichter", | ||||||
|  |     "Show FPS": "FPS anzeigen", | ||||||
|  |     "Keep fps": "FPS beibehalten", | ||||||
|  |     "Keep frames": "Frames beibehalten", | ||||||
|  |     "Fix Blueish Cam": "Bläuliche Kamera korrigieren", | ||||||
|  |     "Mouth Mask": "Mundmaske", | ||||||
|  |     "Show Mouth Mask Box": "Mundmaskenrahmen anzeigen", | ||||||
|  |     "Start": "Starten", | ||||||
|  |     "Live": "Live", | ||||||
|  |     "Destroy": "Beenden", | ||||||
|  |     "Map faces": "Gesichter zuordnen", | ||||||
|  |     "Processing...": "Verarbeitung läuft...", | ||||||
|  |     "Processing succeed!": "Verarbeitung erfolgreich!", | ||||||
|  |     "Processing ignored!": "Verarbeitung ignoriert!", | ||||||
|  |     "Failed to start camera": "Kamera konnte nicht gestartet werden", | ||||||
|  |     "Please complete pop-up or close it.": "Bitte das Pop-up komplettieren oder schließen.", | ||||||
|  |     "Getting unique faces": "Einzigartige Gesichter erfassen", | ||||||
|  |     "Please select a source image first": "Bitte zuerst ein Quellbild auswählen", | ||||||
|  |     "No faces found in target": "Keine Gesichter im Zielbild gefunden", | ||||||
|  |     "Add": "Hinzufügen", | ||||||
|  |     "Clear": "Löschen", | ||||||
|  |     "Submit": "Absenden", | ||||||
|  |     "Select source image": "Quellbild auswählen", | ||||||
|  |     "Select target image": "Zielbild auswählen", | ||||||
|  |     "Please provide mapping!": "Bitte eine Zuordnung angeben!", | ||||||
|  |     "At least 1 source with target is required!": "Mindestens eine Quelle mit einem Ziel ist erforderlich!", | ||||||
|  |     "At least 1 source with target is required!": "Mindestens eine Quelle mit einem Ziel ist erforderlich!", | ||||||
|  |     "Face could not be detected in last upload!": "Im letzten Upload konnte kein Gesicht erkannt werden!", | ||||||
|  |     "Select Camera:": "Kamera auswählen:", | ||||||
|  |     "All mappings cleared!": "Alle Zuordnungen gelöscht!", | ||||||
|  |     "Mappings successfully submitted!": "Zuordnungen erfolgreich übermittelt!", | ||||||
|  |     "Source x Target Mapper is already open.": "Quell-zu-Ziel-Zuordnung ist bereits geöffnet." | ||||||
|  | } | ||||||
|  | @ -1,11 +1,11 @@ | ||||||
| { | { | ||||||
|     "Source x Target Mapper": "Source x Target Mapper", |     "Source x Target Mapper": "Source x Target Mapper", | ||||||
|     "select an source image": "选择一个源图像", |     "select a source image": "选择一个源图像", | ||||||
|     "Preview": "预览", |     "Preview": "预览", | ||||||
|     "select an target image or video": "选择一个目标图像或视频", |     "select a target image or video": "选择一个目标图像或视频", | ||||||
|     "save image output file": "保存图像输出文件", |     "save image output file": "保存图像输出文件", | ||||||
|     "save video output file": "保存视频输出文件", |     "save video output file": "保存视频输出文件", | ||||||
|     "select an target image": "选择一个目标图像", |     "select a target image": "选择一个目标图像", | ||||||
|     "source": "源", |     "source": "源", | ||||||
|     "Select a target": "选择一个目标", |     "Select a target": "选择一个目标", | ||||||
|     "Select a face": "选择一张脸", |     "Select a face": "选择一张脸", | ||||||
|  | @ -36,11 +36,11 @@ | ||||||
|     "Select source image": "请选取源图像", |     "Select source image": "请选取源图像", | ||||||
|     "Select target image": "请选取目标图像", |     "Select target image": "请选取目标图像", | ||||||
|     "Please provide mapping!": "请提供映射", |     "Please provide mapping!": "请提供映射", | ||||||
|     "Atleast 1 source with target is required!": "至少需要一个来源图像与目标图像相关!", |     "At least 1 source with target is required!": "至少需要一个来源图像与目标图像相关!", | ||||||
|     "At least 1 source with target is required!": "至少需要一个来源图像与目标图像相关!", |     "At least 1 source with target is required!": "至少需要一个来源图像与目标图像相关!", | ||||||
|     "Face could not be detected in last upload!": "最近上传的图像中没有检测到人脸!", |     "Face could not be detected in last upload!": "最近上传的图像中没有检测到人脸!", | ||||||
|     "Select Camera:": "选择摄像头", |     "Select Camera:": "选择摄像头", | ||||||
|     "All mappings cleared!": "所有映射均已清除!", |     "All mappings cleared!": "所有映射均已清除!", | ||||||
|     "Mappings successfully submitted!": "成功提交映射!", |     "Mappings successfully submitted!": "成功提交映射!", | ||||||
|     "Source x Target Mapper is already open.": "源 x 目标映射器已打开。" |     "Source x Target Mapper is already open.": "源 x 目标映射器已打开。" | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,18 @@ | ||||||
|  | import os  | ||||||
|  | import cv2 | ||||||
|  | import numpy as np | ||||||
|  | 
 | ||||||
|  | # Utility function to support unicode characters in file paths for reading | ||||||
|  | def imread_unicode(path, flags=cv2.IMREAD_COLOR): | ||||||
|  |     return cv2.imdecode(np.fromfile(path, dtype=np.uint8), flags) | ||||||
|  | 
 | ||||||
|  | # Utility function to support unicode characters in file paths for writing | ||||||
|  | def imwrite_unicode(path, img, params=None): | ||||||
|  |     root, ext = os.path.splitext(path) | ||||||
|  |     if not ext: | ||||||
|  |         ext = ".png" | ||||||
|  |     result, encoded_img = cv2.imencode(ext, img, params if params else []) | ||||||
|  |     result, encoded_img = cv2.imencode(f".{ext}", img, params if params is not None else []) | ||||||
|  |         encoded_img.tofile(path) | ||||||
|  |         return True | ||||||
|  |     return False | ||||||
|  | @ -429,7 +429,7 @@ def create_source_target_popup( | ||||||
|             POPUP.destroy() |             POPUP.destroy() | ||||||
|             select_output_path(start) |             select_output_path(start) | ||||||
|         else: |         else: | ||||||
|             update_pop_status("Atleast 1 source with target is required!") |             update_pop_status("At least 1 source with target is required!") | ||||||
| 
 | 
 | ||||||
|     scrollable_frame = ctk.CTkScrollableFrame( |     scrollable_frame = ctk.CTkScrollableFrame( | ||||||
|         POPUP, width=POPUP_SCROLL_WIDTH, height=POPUP_SCROLL_HEIGHT |         POPUP, width=POPUP_SCROLL_WIDTH, height=POPUP_SCROLL_HEIGHT | ||||||
|  | @ -489,7 +489,7 @@ def update_popup_source( | ||||||
|     global source_label_dict |     global source_label_dict | ||||||
| 
 | 
 | ||||||
|     source_path = ctk.filedialog.askopenfilename( |     source_path = ctk.filedialog.askopenfilename( | ||||||
|         title=_("select an source image"), |         title=_("select a source image"), | ||||||
|         initialdir=RECENT_DIRECTORY_SOURCE, |         initialdir=RECENT_DIRECTORY_SOURCE, | ||||||
|         filetypes=[img_ft], |         filetypes=[img_ft], | ||||||
|     ) |     ) | ||||||
|  | @ -584,7 +584,7 @@ def select_source_path() -> None: | ||||||
| 
 | 
 | ||||||
|     PREVIEW.withdraw() |     PREVIEW.withdraw() | ||||||
|     source_path = ctk.filedialog.askopenfilename( |     source_path = ctk.filedialog.askopenfilename( | ||||||
|         title=_("select an source image"), |         title=_("select a source image"), | ||||||
|         initialdir=RECENT_DIRECTORY_SOURCE, |         initialdir=RECENT_DIRECTORY_SOURCE, | ||||||
|         filetypes=[img_ft], |         filetypes=[img_ft], | ||||||
|     ) |     ) | ||||||
|  | @ -627,7 +627,7 @@ def select_target_path() -> None: | ||||||
| 
 | 
 | ||||||
|     PREVIEW.withdraw() |     PREVIEW.withdraw() | ||||||
|     target_path = ctk.filedialog.askopenfilename( |     target_path = ctk.filedialog.askopenfilename( | ||||||
|         title=_("select an target image or video"), |         title=_("select a target image or video"), | ||||||
|         initialdir=RECENT_DIRECTORY_TARGET, |         initialdir=RECENT_DIRECTORY_TARGET, | ||||||
|         filetypes=[img_ft, vid_ft], |         filetypes=[img_ft, vid_ft], | ||||||
|     ) |     ) | ||||||
|  | @ -799,73 +799,39 @@ def webcam_preview(root: ctk.CTk, camera_index: int): | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def get_available_cameras(): | def get_available_cameras(): | ||||||
|     """Returns a list of available camera names and indices.""" |     """ | ||||||
|  |     Safe camera detection for macOS and Unix-like systems that avoids threading and AVX crashes. | ||||||
|  |     Returns a tuple of (camera_indices, camera_names). | ||||||
|  |     """ | ||||||
|  |     import cv2 | ||||||
|  |     import platform | ||||||
|  | 
 | ||||||
|     if platform.system() == "Windows": |     if platform.system() == "Windows": | ||||||
|         try: |         try: | ||||||
|  |             from pygrabber.dshow_graph import FilterGraph | ||||||
|             graph = FilterGraph() |             graph = FilterGraph() | ||||||
|             devices = graph.get_input_devices() |             devices = graph.get_input_devices() | ||||||
| 
 |  | ||||||
|             # Create list of indices and names |  | ||||||
|             camera_indices = list(range(len(devices))) |             camera_indices = list(range(len(devices))) | ||||||
|             camera_names = devices |             camera_names = devices | ||||||
| 
 |  | ||||||
|             # If no cameras found through DirectShow, try OpenCV fallback |  | ||||||
|             if not camera_names: |  | ||||||
|                 # Try to open camera with index -1 and 0 |  | ||||||
|                 test_indices = [-1, 0] |  | ||||||
|                 working_cameras = [] |  | ||||||
| 
 |  | ||||||
|                 for idx in test_indices: |  | ||||||
|                     cap = cv2.VideoCapture(idx) |  | ||||||
|                     if cap.isOpened(): |  | ||||||
|                         working_cameras.append(f"Camera {idx}") |  | ||||||
|                         cap.release() |  | ||||||
| 
 |  | ||||||
|                 if working_cameras: |  | ||||||
|                     return test_indices[: len(working_cameras)], working_cameras |  | ||||||
| 
 |  | ||||||
|             # If still no cameras found, return empty lists |  | ||||||
|             if not camera_names: |             if not camera_names: | ||||||
|                 return [], ["No cameras found"] |                 return [], ["No cameras found"] | ||||||
| 
 |  | ||||||
|             return camera_indices, camera_names |             return camera_indices, camera_names | ||||||
| 
 |  | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             print(f"Error detecting cameras: {str(e)}") |             print(f"[Camera Detection Error - Windows]: {e}") | ||||||
|             return [], ["No cameras found"] |  | ||||||
|     else: |  | ||||||
|         # Unix-like systems (Linux/Mac) camera detection |  | ||||||
|         camera_indices = [] |  | ||||||
|         camera_names = [] |  | ||||||
| 
 |  | ||||||
|         if platform.system() == "Darwin":  # macOS specific handling |  | ||||||
|             # Try to open the default FaceTime camera first |  | ||||||
|             cap = cv2.VideoCapture(0) |  | ||||||
|             if cap.isOpened(): |  | ||||||
|                 camera_indices.append(0) |  | ||||||
|                 camera_names.append("FaceTime Camera") |  | ||||||
|                 cap.release() |  | ||||||
| 
 |  | ||||||
|             # On macOS, additional cameras typically use indices 1 and 2 |  | ||||||
|             for i in [1, 2]: |  | ||||||
|                 cap = cv2.VideoCapture(i) |  | ||||||
|                 if cap.isOpened(): |  | ||||||
|                     camera_indices.append(i) |  | ||||||
|                     camera_names.append(f"Camera {i}") |  | ||||||
|                     cap.release() |  | ||||||
|         else: |  | ||||||
|             # Linux camera detection - test first 10 indices |  | ||||||
|             for i in range(10): |  | ||||||
|                 cap = cv2.VideoCapture(i) |  | ||||||
|                 if cap.isOpened(): |  | ||||||
|                     camera_indices.append(i) |  | ||||||
|                     camera_names.append(f"Camera {i}") |  | ||||||
|                     cap.release() |  | ||||||
| 
 |  | ||||||
|         if not camera_names: |  | ||||||
|             return [], ["No cameras found"] |             return [], ["No cameras found"] | ||||||
| 
 | 
 | ||||||
|         return camera_indices, camera_names |     # macOS or Linux | ||||||
|  |     try: | ||||||
|  |         print("[Info] Safely checking for available cameras...") | ||||||
|  |         cap = cv2.VideoCapture(0) | ||||||
|  |         if cap is None or not cap.isOpened(): | ||||||
|  |             print("[Warning] Default camera (index 0) not available.") | ||||||
|  |             return [], ["No cameras found"] | ||||||
|  |         cap.release() | ||||||
|  |         return [0], ["Default Camera (Index 0)"] | ||||||
|  |     except Exception as e: | ||||||
|  |         print(f"[Camera Detection Error - Unix]: {e}") | ||||||
|  |         return [], ["No cameras found"] | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def create_webcam_preview(camera_index: int): | def create_webcam_preview(camera_index: int): | ||||||
|  | @ -1108,7 +1074,7 @@ def update_webcam_source( | ||||||
|     global source_label_dict_live |     global source_label_dict_live | ||||||
| 
 | 
 | ||||||
|     source_path = ctk.filedialog.askopenfilename( |     source_path = ctk.filedialog.askopenfilename( | ||||||
|         title=_("select an source image"), |         title=_("select a source image"), | ||||||
|         initialdir=RECENT_DIRECTORY_SOURCE, |         initialdir=RECENT_DIRECTORY_SOURCE, | ||||||
|         filetypes=[img_ft], |         filetypes=[img_ft], | ||||||
|     ) |     ) | ||||||
|  | @ -1160,7 +1126,7 @@ def update_webcam_target( | ||||||
|     global target_label_dict_live |     global target_label_dict_live | ||||||
| 
 | 
 | ||||||
|     target_path = ctk.filedialog.askopenfilename( |     target_path = ctk.filedialog.askopenfilename( | ||||||
|         title=_("select an target image"), |         title=_("select a target image"), | ||||||
|         initialdir=RECENT_DIRECTORY_SOURCE, |         initialdir=RECENT_DIRECTORY_SOURCE, | ||||||
|         filetypes=[img_ft], |         filetypes=[img_ft], | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue