- Fix Hal chat: add Chat/Agent mode toggle so users can switch between direct LLM streaming (Chat) and tool-using Agent mode - Fix Agent system: graceful degradation when model can't follow structured THOUGHT/ACTION/PARAMS format (falls back to direct answer after 2 parse failures instead of looping 20 times) - Fix frozen build: remove llama_cpp from PyInstaller excludes list so LLM works in compiled exe - Add system tray icon: autarch.ico (from icon.svg) used for exe icons, installer shortcuts, and runtime tray icon - Update tray.py to load .ico file with fallback to programmatic generation - Add inline critical CSS for FOUC prevention - Bump version to 1.5.1 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
148 lines
4.8 KiB
Python
148 lines
4.8 KiB
Python
"""AUTARCH System Tray Icon
|
|
|
|
Provides a taskbar/system tray icon with Start, Stop, Restart, Open Dashboard,
|
|
and Exit controls for the web dashboard.
|
|
|
|
Requires: pystray, Pillow
|
|
"""
|
|
|
|
import sys
|
|
import threading
|
|
import webbrowser
|
|
from pathlib import Path
|
|
|
|
try:
|
|
import pystray
|
|
from PIL import Image, ImageDraw, ImageFont
|
|
TRAY_AVAILABLE = True
|
|
except ImportError:
|
|
TRAY_AVAILABLE = False
|
|
|
|
|
|
def _get_icon_path():
|
|
"""Find the .ico file — works in both source and frozen (PyInstaller) builds."""
|
|
if getattr(sys, 'frozen', False):
|
|
base = Path(sys._MEIPASS)
|
|
else:
|
|
base = Path(__file__).parent.parent
|
|
ico = base / 'autarch.ico'
|
|
if ico.exists():
|
|
return ico
|
|
return None
|
|
|
|
|
|
def create_icon_image(size=64):
|
|
"""Load tray icon from .ico file, falling back to programmatic generation."""
|
|
ico_path = _get_icon_path()
|
|
if ico_path:
|
|
try:
|
|
img = Image.open(str(ico_path))
|
|
img = img.resize((size, size), Image.LANCZOS)
|
|
return img.convert('RGBA')
|
|
except Exception:
|
|
pass
|
|
|
|
# Fallback: generate programmatically
|
|
img = Image.new('RGBA', (size, size), (0, 0, 0, 0))
|
|
draw = ImageDraw.Draw(img)
|
|
draw.ellipse([1, 1, size - 2, size - 2], fill=(15, 15, 25, 255),
|
|
outline=(0, 180, 255, 255), width=2)
|
|
try:
|
|
font = ImageFont.truetype("arial.ttf", int(size * 0.5))
|
|
except OSError:
|
|
font = ImageFont.load_default()
|
|
bbox = draw.textbbox((0, 0), "A", font=font)
|
|
tw, th = bbox[2] - bbox[0], bbox[3] - bbox[1]
|
|
x = (size - tw) // 2
|
|
y = (size - th) // 2 - bbox[1]
|
|
draw.text((x, y), "A", fill=(0, 200, 255, 255), font=font)
|
|
return img
|
|
|
|
|
|
class TrayManager:
|
|
"""Manages the system tray icon and Flask server lifecycle."""
|
|
|
|
def __init__(self, app, host, port, ssl_context=None):
|
|
self.app = app
|
|
self.host = host
|
|
self.port = port
|
|
self.ssl_context = ssl_context
|
|
self._server = None
|
|
self._thread = None
|
|
self.running = False
|
|
self._icon = None
|
|
self._proto = 'https' if ssl_context else 'http'
|
|
|
|
def start_server(self):
|
|
"""Start the Flask web server in a background thread."""
|
|
if self.running:
|
|
return
|
|
|
|
from werkzeug.serving import make_server
|
|
self._server = make_server(self.host, self.port, self.app, threaded=True)
|
|
|
|
if self.ssl_context:
|
|
import ssl
|
|
ctx = ssl.SSLContext(ssl.PROTOCOL_TLS_SERVER)
|
|
ctx.load_cert_chain(self.ssl_context[0], self.ssl_context[1])
|
|
self._server.socket = ctx.wrap_socket(self._server.socket, server_side=True)
|
|
|
|
self.running = True
|
|
self._thread = threading.Thread(target=self._server.serve_forever, daemon=True)
|
|
self._thread.start()
|
|
|
|
def stop_server(self):
|
|
"""Stop the Flask web server."""
|
|
if not self.running or not self._server:
|
|
return
|
|
self._server.shutdown()
|
|
self._server = None
|
|
self._thread = None
|
|
self.running = False
|
|
|
|
def restart_server(self):
|
|
"""Stop and restart the Flask web server."""
|
|
self.stop_server()
|
|
self.start_server()
|
|
|
|
def open_browser(self):
|
|
"""Open the dashboard in the default web browser."""
|
|
if self.running:
|
|
host = 'localhost' if self.host in ('0.0.0.0', '::') else self.host
|
|
webbrowser.open(f"{self._proto}://{host}:{self.port}")
|
|
|
|
def quit(self):
|
|
"""Stop server and exit the tray icon."""
|
|
self.stop_server()
|
|
if self._icon:
|
|
self._icon.stop()
|
|
|
|
def run(self):
|
|
"""Start server and show tray icon. Blocks until Exit is clicked."""
|
|
if not TRAY_AVAILABLE:
|
|
raise RuntimeError("pystray or Pillow not installed")
|
|
|
|
self.start_server()
|
|
|
|
image = create_icon_image()
|
|
menu = pystray.Menu(
|
|
pystray.MenuItem(
|
|
lambda item: f"AUTARCH — {'Running' if self.running else 'Stopped'}",
|
|
None, enabled=False),
|
|
pystray.Menu.SEPARATOR,
|
|
pystray.MenuItem("Start", lambda: self.start_server(),
|
|
enabled=lambda item: not self.running),
|
|
pystray.MenuItem("Stop", lambda: self.stop_server(),
|
|
enabled=lambda item: self.running),
|
|
pystray.MenuItem("Restart", lambda: self.restart_server(),
|
|
enabled=lambda item: self.running),
|
|
pystray.Menu.SEPARATOR,
|
|
pystray.MenuItem("Open Dashboard", lambda: self.open_browser(),
|
|
enabled=lambda item: self.running, default=True),
|
|
pystray.Menu.SEPARATOR,
|
|
pystray.MenuItem("Exit", lambda: self.quit()),
|
|
)
|
|
|
|
self._icon = pystray.Icon("autarch", image, "AUTARCH", menu=menu)
|
|
self._icon.run() # Blocks until quit()
|