v1.9.18: revert GUI to v1.9.14 state — fix broken window display
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -15,12 +15,10 @@ class AboutDialog(ctk.CTkToplevel):
|
|||||||
self.geometry("500x480")
|
self.geometry("500x480")
|
||||||
self.resizable(False, False)
|
self.resizable(False, False)
|
||||||
self.transient(master)
|
self.transient(master)
|
||||||
|
self.grab_set()
|
||||||
self.focus_force()
|
self.focus_force()
|
||||||
self.protocol("WM_DELETE_WINDOW", self._on_close)
|
self.protocol("WM_DELETE_WINDOW", self._on_close)
|
||||||
|
|
||||||
self._master_ref = master
|
|
||||||
self._map_bind_id = master.bind("<Map>", self._on_parent_map, add="+")
|
|
||||||
|
|
||||||
# ── Header ──
|
# ── Header ──
|
||||||
ctk.CTkLabel(
|
ctk.CTkLabel(
|
||||||
self, text=t("about_title"),
|
self, text=t("about_title"),
|
||||||
@@ -80,20 +78,9 @@ class AboutDialog(ctk.CTkToplevel):
|
|||||||
self, text=t("close"), width=120, command=self._on_close
|
self, text=t("close"), width=120, command=self._on_close
|
||||||
).pack(pady=(10, 20))
|
).pack(pady=(10, 20))
|
||||||
|
|
||||||
def _on_parent_map(self, event=None):
|
|
||||||
"""Restore dialog when parent is un-minimized."""
|
|
||||||
try:
|
|
||||||
if not self.winfo_exists():
|
|
||||||
return
|
|
||||||
self.deiconify()
|
|
||||||
self.lift()
|
|
||||||
self.focus_force()
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _on_close(self):
|
def _on_close(self):
|
||||||
try:
|
try:
|
||||||
self._master_ref.unbind("<Map>", self._map_bind_id)
|
self.grab_release()
|
||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
self.destroy()
|
self.destroy()
|
||||||
|
|||||||
32
gui/app.py
32
gui/app.py
@@ -3,7 +3,6 @@ Main application window — sidebar + tabview layout.
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
import tkinter
|
import tkinter
|
||||||
import sys
|
|
||||||
import customtkinter as ctk
|
import customtkinter as ctk
|
||||||
from tkinter import messagebox
|
from tkinter import messagebox
|
||||||
|
|
||||||
@@ -119,37 +118,6 @@ class App(ctk.CTk):
|
|||||||
# Cleanup on close
|
# Cleanup on close
|
||||||
self.protocol("WM_DELETE_WINDOW", self._on_close)
|
self.protocol("WM_DELETE_WINDOW", self._on_close)
|
||||||
|
|
||||||
# Fix: restore window after Win+D (Show Desktop)
|
|
||||||
self.bind("<Map>", self._on_map, add="+")
|
|
||||||
if sys.platform == "win32":
|
|
||||||
self._setup_win32_restore()
|
|
||||||
|
|
||||||
def _on_map(self, event=None):
|
|
||||||
"""Ensure window is fully visible when restored from taskbar."""
|
|
||||||
try:
|
|
||||||
self.deiconify()
|
|
||||||
self.lift()
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _setup_win32_restore(self):
|
|
||||||
"""Win32 fallback: periodic check for stuck minimized state."""
|
|
||||||
import ctypes
|
|
||||||
self._user32 = ctypes.windll.user32
|
|
||||||
self._hwnd = int(self.wm_frame(), 16)
|
|
||||||
self._check_minimized()
|
|
||||||
|
|
||||||
def _check_minimized(self):
|
|
||||||
"""If window is iconic but should be visible, force restore."""
|
|
||||||
try:
|
|
||||||
if self._user32.IsIconic(self._hwnd):
|
|
||||||
fg = self._user32.GetForegroundWindow()
|
|
||||||
if fg == self._hwnd:
|
|
||||||
self._user32.ShowWindow(self._hwnd, 9) # SW_RESTORE
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
self.after(500, self._check_minimized)
|
|
||||||
|
|
||||||
def _build_layout(self):
|
def _build_layout(self):
|
||||||
# PanedWindow — resizable sidebar | main area
|
# PanedWindow — resizable sidebar | main area
|
||||||
self._paned = tkinter.PanedWindow(
|
self._paned = tkinter.PanedWindow(
|
||||||
|
|||||||
@@ -32,11 +32,7 @@ class GroupDialog(ctk.CTkToplevel):
|
|||||||
self.geometry("340x200")
|
self.geometry("340x200")
|
||||||
self.resizable(False, False)
|
self.resizable(False, False)
|
||||||
self.transient(master)
|
self.transient(master)
|
||||||
self.focus_force()
|
self.grab_set()
|
||||||
self.protocol("WM_DELETE_WINDOW", self._on_close)
|
|
||||||
|
|
||||||
self._master_ref = master
|
|
||||||
self._map_bind_id = master.bind("<Map>", self._on_parent_map, add="+")
|
|
||||||
|
|
||||||
# ── Name ──
|
# ── Name ──
|
||||||
ctk.CTkLabel(self, text=t("group_name"), anchor="w").pack(
|
ctk.CTkLabel(self, text=t("group_name"), anchor="w").pack(
|
||||||
@@ -75,7 +71,7 @@ class GroupDialog(ctk.CTkToplevel):
|
|||||||
btn_frame.pack(fill="x", padx=20, pady=(15, 10))
|
btn_frame.pack(fill="x", padx=20, pady=(15, 10))
|
||||||
|
|
||||||
ctk.CTkButton(btn_frame, text=t("cancel"), width=80,
|
ctk.CTkButton(btn_frame, text=t("cancel"), width=80,
|
||||||
fg_color="gray", command=self._on_close).pack(side="left")
|
fg_color="gray", command=self.destroy).pack(side="left")
|
||||||
ctk.CTkButton(btn_frame, text=t("save"), width=80,
|
ctk.CTkButton(btn_frame, text=t("save"), width=80,
|
||||||
command=self._save).pack(side="right")
|
command=self._save).pack(side="right")
|
||||||
|
|
||||||
@@ -94,23 +90,6 @@ class GroupDialog(ctk.CTkToplevel):
|
|||||||
else:
|
else:
|
||||||
btn.configure(border_color=fg)
|
btn.configure(border_color=fg)
|
||||||
|
|
||||||
def _on_parent_map(self, event=None):
|
|
||||||
try:
|
|
||||||
if not self.winfo_exists():
|
|
||||||
return
|
|
||||||
self.deiconify()
|
|
||||||
self.lift()
|
|
||||||
self.focus_force()
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _on_close(self):
|
|
||||||
try:
|
|
||||||
self._master_ref.unbind("<Map>", self._map_bind_id)
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
self.destroy()
|
|
||||||
|
|
||||||
def _save(self):
|
def _save(self):
|
||||||
name = self._name_var.get().strip()
|
name = self._name_var.get().strip()
|
||||||
if not name:
|
if not name:
|
||||||
@@ -128,4 +107,4 @@ class GroupDialog(ctk.CTkToplevel):
|
|||||||
group = self.store.add_group(name, self._selected_color)
|
group = self.store.add_group(name, self._selected_color)
|
||||||
self.result = group
|
self.result = group
|
||||||
|
|
||||||
self._on_close()
|
self.destroy()
|
||||||
|
|||||||
@@ -54,13 +54,13 @@ class ServerDialog(ctk.CTkToplevel):
|
|||||||
self.geometry("450x720")
|
self.geometry("450x720")
|
||||||
self.resizable(False, False)
|
self.resizable(False, False)
|
||||||
|
|
||||||
|
# transient BEFORE grab_set — prevents focus lock on minimize
|
||||||
self.transient(master)
|
self.transient(master)
|
||||||
|
self.grab_set()
|
||||||
self.focus_force()
|
self.focus_force()
|
||||||
self.protocol("WM_DELETE_WINDOW", self._on_close)
|
|
||||||
|
|
||||||
# Restore dialog when parent is un-minimized
|
# Release grab on close (prevents stuck app)
|
||||||
self._master_ref = master
|
self.protocol("WM_DELETE_WINDOW", self._on_close)
|
||||||
self._map_bind_id = master.bind("<Map>", self._on_parent_map, add="+")
|
|
||||||
|
|
||||||
self._field_frames: dict[str, ctk.CTkFrame] = {}
|
self._field_frames: dict[str, ctk.CTkFrame] = {}
|
||||||
self._build_ui(server)
|
self._build_ui(server)
|
||||||
@@ -485,20 +485,10 @@ class ServerDialog(ctk.CTkToplevel):
|
|||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
self._show_error(str(e))
|
self._show_error(str(e))
|
||||||
|
|
||||||
def _on_parent_map(self, event=None):
|
|
||||||
"""Restore dialog when parent window is un-minimized."""
|
|
||||||
try:
|
|
||||||
if not self.winfo_exists():
|
|
||||||
return
|
|
||||||
self.deiconify()
|
|
||||||
self.lift()
|
|
||||||
self.focus_force()
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _on_close(self):
|
def _on_close(self):
|
||||||
|
"""Release grab and destroy — prevents stuck app on minimize."""
|
||||||
try:
|
try:
|
||||||
self._master_ref.unbind("<Map>", self._map_bind_id)
|
self.grab_release()
|
||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
self.destroy()
|
self.destroy()
|
||||||
|
|||||||
@@ -83,11 +83,7 @@ class UpdateDialog(ctk.CTkToplevel):
|
|||||||
self.geometry("500x420")
|
self.geometry("500x420")
|
||||||
self.resizable(False, False)
|
self.resizable(False, False)
|
||||||
self.transient(parent)
|
self.transient(parent)
|
||||||
self.focus_force()
|
self.grab_set()
|
||||||
self.protocol("WM_DELETE_WINDOW", self._on_close)
|
|
||||||
|
|
||||||
self._master_ref = parent
|
|
||||||
self._map_bind_id = parent.bind("<Map>", self._on_parent_map, add="+")
|
|
||||||
|
|
||||||
self._info = info
|
self._info = info
|
||||||
self._downloaded_path = downloaded_path
|
self._downloaded_path = downloaded_path
|
||||||
@@ -103,23 +99,6 @@ class UpdateDialog(ctk.CTkToplevel):
|
|||||||
py = parent.winfo_y() + (parent.winfo_height() - 420) // 2
|
py = parent.winfo_y() + (parent.winfo_height() - 420) // 2
|
||||||
self.geometry(f"+{px}+{py}")
|
self.geometry(f"+{px}+{py}")
|
||||||
|
|
||||||
def _on_parent_map(self, event=None):
|
|
||||||
try:
|
|
||||||
if not self.winfo_exists():
|
|
||||||
return
|
|
||||||
self.deiconify()
|
|
||||||
self.lift()
|
|
||||||
self.focus_force()
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _on_close(self):
|
|
||||||
try:
|
|
||||||
self._master_ref.unbind("<Map>", self._map_bind_id)
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
self.destroy()
|
|
||||||
|
|
||||||
def _build_ui(self):
|
def _build_ui(self):
|
||||||
from version import __version__
|
from version import __version__
|
||||||
|
|
||||||
@@ -215,7 +194,7 @@ class UpdateDialog(ctk.CTkToplevel):
|
|||||||
width=80, height=34, corner_radius=8,
|
width=80, height=34, corner_radius=8,
|
||||||
fg_color="#4b5563", hover_color="#374151",
|
fg_color="#4b5563", hover_color="#374151",
|
||||||
font=ctk.CTkFont(size=13),
|
font=ctk.CTkFont(size=13),
|
||||||
command=self._on_close,
|
command=self.destroy,
|
||||||
).pack(side="right", padx=(8, 0))
|
).pack(side="right", padx=(8, 0))
|
||||||
|
|
||||||
ctk.CTkButton(
|
ctk.CTkButton(
|
||||||
@@ -289,4 +268,4 @@ class UpdateDialog(ctk.CTkToplevel):
|
|||||||
def _on_skip_click(self):
|
def _on_skip_click(self):
|
||||||
if self._on_skip:
|
if self._on_skip:
|
||||||
self._on_skip(self._info["version"])
|
self._on_skip(self._info["version"])
|
||||||
self._on_close()
|
self.destroy()
|
||||||
|
|||||||
Binary file not shown.
@@ -1,6 +1,6 @@
|
|||||||
"""Version info for ServerManager."""
|
"""Version info for ServerManager."""
|
||||||
|
|
||||||
__version__ = "1.9.16"
|
__version__ = "1.9.18"
|
||||||
__app_name__ = "ServerManager"
|
__app_name__ = "ServerManager"
|
||||||
__author__ = "aibot777"
|
__author__ = "aibot777"
|
||||||
__description__ = "Desktop GUI for managing remote servers"
|
__description__ = "Desktop GUI for managing remote servers"
|
||||||
|
|||||||
Reference in New Issue
Block a user