feat: desktop pet v2 with 8 skins and state notifications
This commit is contained in:
818
frontends/desktop_pet_v2.pyw
Normal file
818
frontends/desktop_pet_v2.pyw
Normal file
@@ -0,0 +1,818 @@
|
||||
"""Desktop Pet with Skin System — Cross-platform with True Transparency"""
|
||||
import os
|
||||
import sys
|
||||
import json
|
||||
import threading
|
||||
from http.server import HTTPServer, BaseHTTPRequestHandler
|
||||
from urllib.parse import urlparse, parse_qs
|
||||
from PIL import Image, ImageDraw, ImageFont, ImageOps
|
||||
import io
|
||||
|
||||
PORT = 51983
|
||||
SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__))
|
||||
PROJECT_DIR = os.path.dirname(SCRIPT_DIR)
|
||||
SKINS_DIR = os.path.join(SCRIPT_DIR, 'skins')
|
||||
|
||||
class SkinLoader:
|
||||
"""Load and parse skin configuration"""
|
||||
@staticmethod
|
||||
def load_skin(skin_path):
|
||||
"""Load skin.json and return skin config"""
|
||||
config_file = os.path.join(skin_path, 'skin.json')
|
||||
if not os.path.exists(config_file):
|
||||
raise FileNotFoundError(f"skin.json not found in {skin_path}")
|
||||
|
||||
with open(config_file, 'r', encoding='utf-8') as f:
|
||||
config = json.load(f)
|
||||
|
||||
if 'animations' not in config:
|
||||
raise ValueError("skin.json must contain 'animations' field")
|
||||
|
||||
config['path'] = skin_path
|
||||
return config
|
||||
|
||||
@staticmethod
|
||||
def list_skins():
|
||||
"""List all available skins"""
|
||||
if not os.path.exists(SKINS_DIR):
|
||||
return []
|
||||
|
||||
skins = []
|
||||
for item in os.listdir(SKINS_DIR):
|
||||
skin_path = os.path.join(SKINS_DIR, item)
|
||||
if os.path.isdir(skin_path):
|
||||
config_file = os.path.join(skin_path, 'skin.json')
|
||||
if os.path.exists(config_file):
|
||||
skins.append(item)
|
||||
return skins
|
||||
|
||||
class AnimationLoader:
|
||||
"""Load animation frames from sprite sheet"""
|
||||
@staticmethod
|
||||
def load_sprite_frames(skin_path, anim_config):
|
||||
"""Load frames from sprite sheet"""
|
||||
file_path = os.path.join(skin_path, anim_config['file'])
|
||||
sprite_config = anim_config['sprite']
|
||||
|
||||
img = Image.open(file_path)
|
||||
frames = []
|
||||
|
||||
frame_width = sprite_config['frameWidth']
|
||||
frame_height = sprite_config['frameHeight']
|
||||
frame_count = sprite_config['frameCount']
|
||||
columns = sprite_config['columns']
|
||||
start_frame = sprite_config.get('startFrame', 0)
|
||||
|
||||
for i in range(frame_count):
|
||||
frame_idx = start_frame + i
|
||||
row = frame_idx // columns
|
||||
col = frame_idx % columns
|
||||
|
||||
x = col * frame_width
|
||||
y = row * frame_height
|
||||
|
||||
frame = img.crop((x, y, x + frame_width, y + frame_height))
|
||||
frames.append(frame)
|
||||
|
||||
return frames
|
||||
|
||||
|
||||
def _find_bubble_asset():
|
||||
"""Find user-provided bubble asset in project root."""
|
||||
candidates = [
|
||||
os.path.join(PROJECT_DIR, '聊天气泡.png'),
|
||||
os.path.join(PROJECT_DIR, 'bubble.png'),
|
||||
]
|
||||
for path in candidates:
|
||||
if os.path.exists(path):
|
||||
return path
|
||||
return None
|
||||
|
||||
|
||||
def _load_default_font(size):
|
||||
"""Load a usable font for bubble text."""
|
||||
font_candidates = [
|
||||
'/System/Library/Fonts/Supplemental/Arial Unicode.ttf',
|
||||
'/System/Library/Fonts/PingFang.ttc',
|
||||
'/System/Library/Fonts/STHeiti Light.ttc',
|
||||
'C:/Windows/Fonts/msyh.ttc',
|
||||
'C:/Windows/Fonts/simhei.ttf',
|
||||
'C:/Windows/Fonts/arial.ttf',
|
||||
]
|
||||
for font_path in font_candidates:
|
||||
if os.path.exists(font_path):
|
||||
try:
|
||||
return ImageFont.truetype(font_path, size=size)
|
||||
except Exception:
|
||||
pass
|
||||
return ImageFont.load_default()
|
||||
|
||||
|
||||
def _wrap_text_for_width(draw, text, font, max_width):
|
||||
"""Wrap text to fit inside max_width."""
|
||||
text = (text or '').strip()
|
||||
if not text:
|
||||
return ['']
|
||||
|
||||
paragraphs = text.replace('\r\n', '\n').replace('\r', '\n').split('\n')
|
||||
lines = []
|
||||
|
||||
for paragraph in paragraphs:
|
||||
if not paragraph:
|
||||
lines.append('')
|
||||
continue
|
||||
|
||||
current = ''
|
||||
for ch in paragraph:
|
||||
candidate = current + ch
|
||||
bbox = draw.textbbox((0, 0), candidate, font=font)
|
||||
width = bbox[2] - bbox[0]
|
||||
if current and width > max_width:
|
||||
lines.append(current)
|
||||
current = ch
|
||||
else:
|
||||
current = candidate
|
||||
if current:
|
||||
lines.append(current)
|
||||
|
||||
return lines or ['']
|
||||
|
||||
|
||||
def build_bubble_image(message, max_width=220):
|
||||
"""Build a PIL image for the toast bubble using the user asset when available."""
|
||||
message = (message or '').strip()
|
||||
bubble_path = _find_bubble_asset()
|
||||
|
||||
if bubble_path and os.path.exists(bubble_path):
|
||||
bubble = Image.open(bubble_path).convert('RGBA')
|
||||
else:
|
||||
bubble = Image.new('RGBA', (256, 128), (255, 255, 255, 0))
|
||||
draw = ImageDraw.Draw(bubble)
|
||||
draw.rounded_rectangle((8, 8, 247, 87), radius=12, fill=(255, 255, 255, 255), outline=(0, 0, 0, 255), width=3)
|
||||
draw.polygon([(48, 87), (72, 87), (56, 112)], fill=(255, 255, 255, 255), outline=(0, 0, 0, 255))
|
||||
|
||||
bubble = ImageOps.contain(bubble, (max_width, max(64, int(max_width * bubble.height / bubble.width))), Image.NEAREST)
|
||||
|
||||
font_size = max(12, bubble.height // 7)
|
||||
font = _load_default_font(font_size)
|
||||
draw = ImageDraw.Draw(bubble)
|
||||
|
||||
pad_left = max(10, bubble.width // 18)
|
||||
pad_right = max(10, bubble.width // 18) + max(6, bubble.width // 14)
|
||||
pad_top = max(8, bubble.height // 10)
|
||||
pad_bottom = max(18, bubble.height // 4)
|
||||
text_area_width = max(40, bubble.width - pad_left - pad_right)
|
||||
|
||||
lines = _wrap_text_for_width(draw, message, font, text_area_width)
|
||||
ascent, descent = font.getmetrics() if hasattr(font, 'getmetrics') else (font_size, font_size // 4)
|
||||
line_height = max(font_size, ascent + descent)
|
||||
max_lines = max(1, (bubble.height - pad_top - pad_bottom) // line_height)
|
||||
if len(lines) > max_lines:
|
||||
lines = lines[:max_lines]
|
||||
if lines:
|
||||
last = lines[-1]
|
||||
while last and draw.textbbox((0, 0), last + '…', font=font)[2] > text_area_width:
|
||||
last = last[:-1]
|
||||
lines[-1] = (last + '…') if last else '…'
|
||||
|
||||
total_text_height = len(lines) * line_height
|
||||
y = pad_top + max(0, (bubble.height - pad_top - pad_bottom - total_text_height) // 2)
|
||||
|
||||
for line in lines:
|
||||
bbox = draw.textbbox((0, 0), line, font=font)
|
||||
text_width = bbox[2] - bbox[0]
|
||||
x = pad_left + (text_area_width - text_width) / 2
|
||||
draw.text((x, y), line, font=font, fill=(32, 32, 32, 255))
|
||||
y += line_height
|
||||
|
||||
alpha = bubble.getchannel('A')
|
||||
bbox = alpha.getbbox()
|
||||
if bbox:
|
||||
bubble = bubble.crop(bbox)
|
||||
|
||||
width, height = bubble.size
|
||||
alpha = bubble.getchannel('A')
|
||||
bottom_y = height - 1
|
||||
tail_x = width // 2
|
||||
for y in range(height - 1, -1, -1):
|
||||
xs = [x for x in range(width) if alpha.getpixel((x, y)) > 0]
|
||||
if xs:
|
||||
bottom_y = y
|
||||
tail_x = xs[len(xs) // 2]
|
||||
break
|
||||
|
||||
return {
|
||||
'image': bubble,
|
||||
'size': bubble.size,
|
||||
'tail_tip': (tail_x, bottom_y),
|
||||
}
|
||||
|
||||
# ============================================================================
|
||||
# macOS Implementation - Pure Cocoa with True Transparency
|
||||
# ============================================================================
|
||||
if sys.platform == 'darwin':
|
||||
from Cocoa import (
|
||||
NSApplication, NSWindow, NSImageView, NSImage, NSData, NSTimer,
|
||||
NSMenu, NSMenuItem, NSApp, NSFloatingWindowLevel, NSColor,
|
||||
NSBackingStoreBuffered, NSWindowStyleMaskBorderless,
|
||||
NSApplicationActivationPolicyAccessory
|
||||
)
|
||||
from Foundation import NSMakeRect, NSMakePoint, NSMakeSize
|
||||
from PyObjCTools import AppHelper
|
||||
import objc
|
||||
|
||||
class MacPet:
|
||||
def __init__(self, skin_name=None):
|
||||
self.app = NSApplication.sharedApplication()
|
||||
self.app.setActivationPolicy_(NSApplicationActivationPolicyAccessory)
|
||||
|
||||
# Load skin
|
||||
self.load_skin(skin_name)
|
||||
|
||||
# Get screen size
|
||||
from AppKit import NSScreen, NSWindowCollectionBehaviorCanJoinAllSpaces, NSWindowCollectionBehaviorStationary
|
||||
screen = NSScreen.mainScreen()
|
||||
screen_frame = screen.frame()
|
||||
screen_width = screen_frame.size.width
|
||||
screen_height = screen_frame.size.height
|
||||
|
||||
# Position at right side
|
||||
x_pos = screen_width - 200
|
||||
y_pos = 300
|
||||
|
||||
# Create transparent window
|
||||
self.window = NSWindow.alloc().initWithContentRect_styleMask_backing_defer_(
|
||||
NSMakeRect(x_pos, y_pos, self.display_width, self.display_height),
|
||||
NSWindowStyleMaskBorderless,
|
||||
NSBackingStoreBuffered,
|
||||
False
|
||||
)
|
||||
|
||||
self.window.setOpaque_(False)
|
||||
self.window.setBackgroundColor_(NSColor.clearColor())
|
||||
self.window.setLevel_(NSFloatingWindowLevel)
|
||||
self.window.setMovableByWindowBackground_(True)
|
||||
self.window.setAcceptsMouseMovedEvents_(True)
|
||||
|
||||
# Make window sticky across spaces (stays in fixed screen position)
|
||||
self.window.setCollectionBehavior_(
|
||||
NSWindowCollectionBehaviorCanJoinAllSpaces |
|
||||
NSWindowCollectionBehaviorStationary
|
||||
)
|
||||
|
||||
# Create custom view for handling mouse events
|
||||
from AppKit import NSView
|
||||
from objc import super as objc_super
|
||||
|
||||
class DraggableImageView(NSView):
|
||||
"""Custom view that handles dragging and double-click"""
|
||||
def initWithFrame_(self, frame):
|
||||
self = objc_super(DraggableImageView, self).initWithFrame_(frame)
|
||||
if self is None:
|
||||
return None
|
||||
self.image_view = NSImageView.alloc().initWithFrame_(self.bounds())
|
||||
self.image_view.setImageScaling_(1) # NSImageScaleProportionallyUpOrDown
|
||||
self.addSubview_(self.image_view)
|
||||
|
||||
# Create overlay view for toast (always on top)
|
||||
# Make it non-opaque so it doesn't block the image
|
||||
self.overlay_view = NSView.alloc().initWithFrame_(self.bounds())
|
||||
self.overlay_view.setWantsLayer_(True)
|
||||
self.addSubview_(self.overlay_view)
|
||||
|
||||
self.drag_start = None
|
||||
return self
|
||||
|
||||
def mouseDown_(self, event):
|
||||
"""Handle mouse down for dragging"""
|
||||
if event.clickCount() == 2:
|
||||
# Double-click to quit
|
||||
from AppKit import NSApp
|
||||
NSApp.terminate_(None)
|
||||
else:
|
||||
# Start dragging
|
||||
self.drag_start = event.locationInWindow()
|
||||
|
||||
def mouseDragged_(self, event):
|
||||
"""Handle mouse drag"""
|
||||
if self.drag_start:
|
||||
current_location = event.locationInWindow()
|
||||
window_frame = self.window().frame()
|
||||
|
||||
dx = current_location.x - self.drag_start.x
|
||||
dy = current_location.y - self.drag_start.y
|
||||
|
||||
new_origin = NSMakePoint(
|
||||
window_frame.origin.x + dx,
|
||||
window_frame.origin.y + dy
|
||||
)
|
||||
|
||||
self.window().setFrameOrigin_(new_origin)
|
||||
|
||||
def acceptsFirstMouse_(self, event):
|
||||
"""Accept first mouse click"""
|
||||
return True
|
||||
|
||||
# Create draggable view
|
||||
self.content_view = DraggableImageView.alloc().initWithFrame_(
|
||||
NSMakeRect(0, 0, self.display_width, self.display_height)
|
||||
)
|
||||
self.image_view = self.content_view.image_view
|
||||
self.overlay_view = self.content_view.overlay_view
|
||||
self.window.setContentView_(self.content_view)
|
||||
|
||||
# Animation state
|
||||
self.current_state = 'idle'
|
||||
self.frame_idx = 0
|
||||
|
||||
# Toast state
|
||||
self.toast_label = None
|
||||
self.toast_timer = None
|
||||
self.toast_image = None
|
||||
self.toast_window = None
|
||||
|
||||
# Start animation timer
|
||||
self.timer = NSTimer.scheduledTimerWithTimeInterval_target_selector_userInfo_repeats_(
|
||||
1.0 / self.animations[self.current_state]['fps'],
|
||||
self,
|
||||
'animate:',
|
||||
None,
|
||||
True
|
||||
)
|
||||
|
||||
# Show window
|
||||
self.window.makeKeyAndOrderFront_(None)
|
||||
|
||||
# Start HTTP server
|
||||
self._start_server()
|
||||
|
||||
print(f"✓ macOS Pet started at ({x_pos}, {y_pos})")
|
||||
print(f" Animations: {', '.join(self.animations.keys())}")
|
||||
|
||||
def load_skin(self, skin_name=None):
|
||||
"""Load skin configuration and animations"""
|
||||
available_skins = SkinLoader.list_skins()
|
||||
if not available_skins:
|
||||
raise FileNotFoundError(f"No skins found in {SKINS_DIR}")
|
||||
|
||||
if skin_name is None or skin_name not in available_skins:
|
||||
skin_name = available_skins[0]
|
||||
|
||||
skin_path = os.path.join(SKINS_DIR, skin_name)
|
||||
self.skin_config = SkinLoader.load_skin(skin_path)
|
||||
|
||||
# Get display size
|
||||
display_size = self.skin_config.get('size', {})
|
||||
self.display_width = display_size.get('width', 128)
|
||||
self.display_height = display_size.get('height', 128)
|
||||
|
||||
# Load animations
|
||||
self.animations = {}
|
||||
for anim_name, anim_config in self.skin_config['animations'].items():
|
||||
pil_frames = AnimationLoader.load_sprite_frames(skin_path, anim_config)
|
||||
|
||||
# Scale frames
|
||||
scaled_frames = []
|
||||
for frame in pil_frames:
|
||||
if frame.mode != 'RGBA':
|
||||
frame = frame.convert('RGBA')
|
||||
scaled = frame.resize((self.display_width, self.display_height), Image.NEAREST)
|
||||
scaled_frames.append(scaled)
|
||||
|
||||
# Convert to NSImage with proper alpha handling
|
||||
ns_images = []
|
||||
for pil_img in scaled_frames:
|
||||
# Ensure RGBA mode for transparency
|
||||
if pil_img.mode != 'RGBA':
|
||||
pil_img = pil_img.convert('RGBA')
|
||||
|
||||
# Convert PIL to PNG bytes (PNG preserves alpha channel)
|
||||
png_buffer = io.BytesIO()
|
||||
pil_img.save(png_buffer, format='PNG')
|
||||
png_data = png_buffer.getvalue()
|
||||
|
||||
# Create NSImage from PNG data
|
||||
ns_data = NSData.dataWithBytes_length_(png_data, len(png_data))
|
||||
ns_image = NSImage.alloc().initWithData_(ns_data)
|
||||
ns_images.append(ns_image)
|
||||
|
||||
self.animations[anim_name] = {
|
||||
'frames': ns_images,
|
||||
'fps': anim_config.get('sprite', {}).get('fps', 6)
|
||||
}
|
||||
|
||||
def animate_(self, timer):
|
||||
"""Animation callback"""
|
||||
anim = self.animations[self.current_state]
|
||||
frames = anim['frames']
|
||||
|
||||
if frames:
|
||||
self.image_view.setImage_(frames[self.frame_idx])
|
||||
self.frame_idx = (self.frame_idx + 1) % len(frames)
|
||||
|
||||
def set_state(self, state):
|
||||
"""Change animation state (must be called on main thread)"""
|
||||
if state in self.animations and state != self.current_state:
|
||||
self.current_state = state
|
||||
self.frame_idx = 0
|
||||
|
||||
# Update timer interval
|
||||
self.timer.invalidate()
|
||||
self.timer = NSTimer.scheduledTimerWithTimeInterval_target_selector_userInfo_repeats_(
|
||||
1.0 / self.animations[self.current_state]['fps'],
|
||||
self,
|
||||
'animate:',
|
||||
None,
|
||||
True
|
||||
)
|
||||
print(f"→ State: {state}")
|
||||
|
||||
def set_state_safe(self, state):
|
||||
"""Thread-safe wrapper for set_state"""
|
||||
AppHelper.callAfter(lambda: self.set_state(state))
|
||||
|
||||
def show_toast(self, message):
|
||||
"""Show toast message above pet"""
|
||||
from AppKit import NSImageView
|
||||
|
||||
with open('/tmp/pet_toast_debug.log', 'a') as f:
|
||||
f.write(f"[DEBUG] show_toast called with: {message}\n")
|
||||
f.flush()
|
||||
|
||||
if self.toast_window:
|
||||
self.toast_window.orderOut_(None)
|
||||
self.toast_window = None
|
||||
self.toast_label = None
|
||||
if self.toast_timer:
|
||||
self.toast_timer.invalidate()
|
||||
self.toast_timer = None
|
||||
|
||||
bubble_info = build_bubble_image(message, max_width=max(180, min(260, self.display_width * 2)))
|
||||
bubble_pil = bubble_info['image']
|
||||
bubble_width, bubble_height = bubble_info['size']
|
||||
tail_x, tail_y = bubble_info['tail_tip']
|
||||
|
||||
png_buffer = io.BytesIO()
|
||||
bubble_pil.save(png_buffer, format='PNG')
|
||||
png_data = png_buffer.getvalue()
|
||||
ns_data = NSData.dataWithBytes_length_(png_data, len(png_data))
|
||||
self.toast_image = NSImage.alloc().initWithData_(ns_data)
|
||||
|
||||
pet_frame = self.window.frame()
|
||||
anchor_x = pet_frame.origin.x + self.display_width * 0.75
|
||||
anchor_y = pet_frame.origin.y + self.display_height * 1.65
|
||||
toast_x = anchor_x - tail_x
|
||||
toast_y = anchor_y - tail_y
|
||||
|
||||
self.toast_window = NSWindow.alloc().initWithContentRect_styleMask_backing_defer_(
|
||||
NSMakeRect(toast_x, toast_y, bubble_width, bubble_height),
|
||||
NSWindowStyleMaskBorderless,
|
||||
NSBackingStoreBuffered,
|
||||
False
|
||||
)
|
||||
self.toast_window.setOpaque_(False)
|
||||
self.toast_window.setBackgroundColor_(NSColor.clearColor())
|
||||
self.toast_window.setLevel_(NSFloatingWindowLevel)
|
||||
self.toast_window.setIgnoresMouseEvents_(True)
|
||||
self.toast_window.setHasShadow_(False)
|
||||
|
||||
self.toast_label = NSImageView.alloc().initWithFrame_(
|
||||
NSMakeRect(0, 0, bubble_width, bubble_height)
|
||||
)
|
||||
self.toast_label.setImage_(self.toast_image)
|
||||
self.toast_label.setImageScaling_(0)
|
||||
self.toast_window.setContentView_(self.toast_label)
|
||||
self.toast_window.orderFrontRegardless()
|
||||
|
||||
with open('/tmp/pet_toast_debug.log', 'a') as f:
|
||||
f.write(f"[DEBUG] Toast window shown at ({toast_x}, {toast_y}) size={bubble_info['size']} tail={bubble_info['tail_tip']}\n")
|
||||
f.flush()
|
||||
|
||||
self.toast_timer = NSTimer.scheduledTimerWithTimeInterval_target_selector_userInfo_repeats_(
|
||||
3.0,
|
||||
self,
|
||||
'hideToast:',
|
||||
None,
|
||||
False
|
||||
)
|
||||
print(f"Toast: {message}")
|
||||
|
||||
def show_toast_safe(self, message):
|
||||
"""Thread-safe wrapper for show_toast"""
|
||||
from Foundation import NSRunLoop
|
||||
from PyObjCTools import AppHelper
|
||||
|
||||
# Write to log file for debugging
|
||||
with open('/tmp/pet_toast_debug.log', 'a') as f:
|
||||
f.write(f"[DEBUG] show_toast_safe called with: {message}\n")
|
||||
f.flush()
|
||||
|
||||
# Schedule on main thread
|
||||
def show_on_main():
|
||||
with open('/tmp/pet_toast_debug.log', 'a') as f:
|
||||
f.write(f"[DEBUG] show_on_main executing\n")
|
||||
f.flush()
|
||||
self.show_toast(message)
|
||||
|
||||
AppHelper.callAfter(show_on_main)
|
||||
|
||||
def hideToast_(self, timer):
|
||||
"""Hide toast message"""
|
||||
if self.toast_window:
|
||||
self.toast_window.orderOut_(None)
|
||||
self.toast_window = None
|
||||
self.toast_label = None
|
||||
self.toast_image = None
|
||||
self.toast_timer = None
|
||||
|
||||
def _start_server(self):
|
||||
"""Start HTTP control server"""
|
||||
pet = self
|
||||
|
||||
class Handler(BaseHTTPRequestHandler):
|
||||
def do_GET(self):
|
||||
parsed = urlparse(self.path)
|
||||
params = parse_qs(parsed.query)
|
||||
|
||||
if 'state' in params:
|
||||
state = params['state'][0]
|
||||
pet.set_state_safe(state)
|
||||
self.send_response(200)
|
||||
self.end_headers()
|
||||
self.wfile.write(b'ok')
|
||||
elif 'msg' in params:
|
||||
msg = params['msg'][0]
|
||||
pet.show_toast_safe(msg)
|
||||
self.send_response(200)
|
||||
self.end_headers()
|
||||
self.wfile.write(b'ok')
|
||||
else:
|
||||
self.send_response(400)
|
||||
self.end_headers()
|
||||
self.wfile.write(b'?state=idle/walk/run/sprint or ?msg=hello')
|
||||
|
||||
def do_POST(self):
|
||||
body = self.rfile.read(int(self.headers.get('Content-Length', 0))).decode()
|
||||
if body:
|
||||
pet.show_toast_safe(body)
|
||||
self.send_response(200)
|
||||
self.end_headers()
|
||||
self.wfile.write(b'ok')
|
||||
else:
|
||||
self.send_response(400)
|
||||
self.end_headers()
|
||||
self.wfile.write(b'empty body')
|
||||
|
||||
def log_message(self, *a):
|
||||
pass
|
||||
|
||||
try:
|
||||
HTTPServer.allow_reuse_address = True
|
||||
srv = HTTPServer(('127.0.0.1', PORT), Handler)
|
||||
t = threading.Thread(target=srv.serve_forever, daemon=True)
|
||||
t.start()
|
||||
print(f'✓ Server: http://127.0.0.1:{PORT}/?state=walk')
|
||||
except OSError as e:
|
||||
if e.errno == 48:
|
||||
print(f'⚠ Port {PORT} already in use')
|
||||
else:
|
||||
raise
|
||||
|
||||
def run(self):
|
||||
"""Run the application"""
|
||||
AppHelper.runEventLoop()
|
||||
|
||||
# ============================================================================
|
||||
# Windows Implementation - tkinter with transparentcolor
|
||||
# ============================================================================
|
||||
else:
|
||||
import tkinter as tk
|
||||
from PIL import ImageTk
|
||||
|
||||
class WinPet:
|
||||
def __init__(self, skin_name=None):
|
||||
self.root = tk.Tk()
|
||||
self.root.wm_attributes('-topmost', True)
|
||||
|
||||
# Load skin
|
||||
self.load_skin(skin_name)
|
||||
|
||||
# Setup window
|
||||
screen_width = self.root.winfo_screenwidth()
|
||||
screen_height = self.root.winfo_screenheight()
|
||||
|
||||
x_pos = screen_width - 200
|
||||
y_pos = screen_height - 300
|
||||
|
||||
self.root.geometry(f'{self.display_width}x{self.display_height}+{x_pos}+{y_pos}')
|
||||
self.root.overrideredirect(True)
|
||||
self.root.wm_attributes('-topmost', True)
|
||||
|
||||
# Transparent background
|
||||
self.root.wm_attributes('-transparentcolor', '#F0F0F0')
|
||||
self.root.config(bg='#F0F0F0')
|
||||
|
||||
# Create label
|
||||
self.label = tk.Label(self.root, bg='#F0F0F0', bd=0)
|
||||
self.label.pack()
|
||||
|
||||
# Bind events
|
||||
self.label.bind('<Button-1>', lambda e: setattr(self, '_d', (e.x, e.y)))
|
||||
self.label.bind('<B1-Motion>', self._drag)
|
||||
self.label.bind('<Double-1>', lambda e: (self.root.destroy(), os._exit(0)))
|
||||
|
||||
# Animation state
|
||||
self.current_state = 'idle'
|
||||
self.frame_idx = 0
|
||||
|
||||
# Toast state
|
||||
self.toast_window = None
|
||||
self.toast_photo = None
|
||||
|
||||
# Start animation
|
||||
self._animate()
|
||||
self._start_server()
|
||||
|
||||
print(f"✓ Windows Pet started at ({x_pos}, {y_pos})")
|
||||
print(f" Animations: {', '.join(self.animations.keys())}")
|
||||
|
||||
self.root.mainloop()
|
||||
|
||||
def load_skin(self, skin_name=None):
|
||||
"""Load skin configuration and animations"""
|
||||
available_skins = SkinLoader.list_skins()
|
||||
if not available_skins:
|
||||
raise FileNotFoundError(f"No skins found in {SKINS_DIR}")
|
||||
|
||||
if skin_name is None or skin_name not in available_skins:
|
||||
skin_name = available_skins[0]
|
||||
|
||||
skin_path = os.path.join(SKINS_DIR, skin_name)
|
||||
self.skin_config = SkinLoader.load_skin(skin_path)
|
||||
|
||||
# Get display size
|
||||
display_size = self.skin_config.get('size', {})
|
||||
self.display_width = display_size.get('width', 128)
|
||||
self.display_height = display_size.get('height', 128)
|
||||
|
||||
# Load animations
|
||||
self.animations = {}
|
||||
for anim_name, anim_config in self.skin_config['animations'].items():
|
||||
pil_frames = AnimationLoader.load_sprite_frames(skin_path, anim_config)
|
||||
|
||||
# Scale and convert frames
|
||||
tk_frames = []
|
||||
for frame in pil_frames:
|
||||
if frame.mode != 'RGBA':
|
||||
frame = frame.convert('RGBA')
|
||||
scaled = frame.resize((self.display_width, self.display_height), Image.NEAREST)
|
||||
tk_frames.append(ImageTk.PhotoImage(scaled))
|
||||
|
||||
self.animations[anim_name] = {
|
||||
'frames': tk_frames,
|
||||
'fps': anim_config.get('sprite', {}).get('fps', 6)
|
||||
}
|
||||
|
||||
def set_state(self, state):
|
||||
"""Change animation state"""
|
||||
if state in self.animations and state != self.current_state:
|
||||
self.current_state = state
|
||||
self.frame_idx = 0
|
||||
print(f"→ State: {state}")
|
||||
|
||||
def _drag(self, e):
|
||||
x = self.root.winfo_x() + e.x - self._d[0]
|
||||
y = self.root.winfo_y() + e.y - self._d[1]
|
||||
self.root.geometry(f'+{x}+{y}')
|
||||
|
||||
def _animate(self):
|
||||
"""Animate current state"""
|
||||
if self.current_state not in self.animations:
|
||||
self.root.after(100, self._animate)
|
||||
return
|
||||
|
||||
anim = self.animations[self.current_state]
|
||||
frames = anim['frames']
|
||||
|
||||
if frames:
|
||||
self.label.config(image=frames[self.frame_idx])
|
||||
self.frame_idx = (self.frame_idx + 1) % len(frames)
|
||||
|
||||
delay = int(1000 / anim['fps'])
|
||||
self.root.after(delay, self._animate)
|
||||
|
||||
def show_toast(self, message):
|
||||
"""Show toast message above pet"""
|
||||
if self.toast_window:
|
||||
try:
|
||||
self.toast_window.destroy()
|
||||
except:
|
||||
pass
|
||||
self.toast_window = None
|
||||
|
||||
bubble_info = build_bubble_image(message, max_width=max(180, min(260, self.display_width * 2)))
|
||||
bubble_pil = bubble_info['image']
|
||||
bubble_width, bubble_height = bubble_info['size']
|
||||
tail_x, tail_y = bubble_info['tail_tip']
|
||||
|
||||
self.toast_photo = ImageTk.PhotoImage(bubble_pil)
|
||||
|
||||
self.toast_window = tk.Toplevel(self.root)
|
||||
self.toast_window.overrideredirect(True)
|
||||
self.toast_window.wm_attributes('-topmost', True)
|
||||
self.toast_window.wm_attributes('-transparentcolor', '#00ff01')
|
||||
self.toast_window.config(bg='#00ff01')
|
||||
|
||||
toast_label = tk.Label(
|
||||
self.toast_window,
|
||||
image=self.toast_photo,
|
||||
bg='#00ff01',
|
||||
bd=0,
|
||||
highlightthickness=0
|
||||
)
|
||||
toast_label.pack()
|
||||
|
||||
pet_x = self.root.winfo_x()
|
||||
pet_y = self.root.winfo_y()
|
||||
anchor_x = pet_x + int(self.display_width * 0.75)
|
||||
anchor_y = pet_y
|
||||
toast_x = anchor_x - tail_x
|
||||
toast_y = anchor_y - bubble_height
|
||||
|
||||
self.toast_window.geometry(f'{bubble_width}x{bubble_height}+{toast_x}+{toast_y}')
|
||||
|
||||
self.root.after(3000, self._hide_toast)
|
||||
print(f"Toast: {message}")
|
||||
|
||||
def _hide_toast(self):
|
||||
"""Hide toast message"""
|
||||
if self.toast_window:
|
||||
try:
|
||||
self.toast_window.destroy()
|
||||
self.toast_window = None
|
||||
except:
|
||||
pass
|
||||
|
||||
def _start_server(self):
|
||||
"""Start HTTP control server"""
|
||||
pet = self
|
||||
|
||||
class Handler(BaseHTTPRequestHandler):
|
||||
def do_GET(self):
|
||||
parsed = urlparse(self.path)
|
||||
params = parse_qs(parsed.query)
|
||||
|
||||
if 'state' in params:
|
||||
state = params['state'][0]
|
||||
pet.root.after(0, pet.set_state, state)
|
||||
self.send_response(200)
|
||||
self.end_headers()
|
||||
self.wfile.write(b'ok')
|
||||
elif 'msg' in params:
|
||||
msg = params['msg'][0]
|
||||
pet.root.after(0, pet.show_toast, msg)
|
||||
self.send_response(200)
|
||||
self.end_headers()
|
||||
self.wfile.write(b'ok')
|
||||
else:
|
||||
self.send_response(400)
|
||||
self.end_headers()
|
||||
self.wfile.write(b'?state=idle/walk/run/sprint or ?msg=hello')
|
||||
|
||||
def do_POST(self):
|
||||
body = self.rfile.read(int(self.headers.get('Content-Length', 0))).decode()
|
||||
if body:
|
||||
pet.root.after(0, pet.show_toast, body)
|
||||
self.send_response(200)
|
||||
self.end_headers()
|
||||
self.wfile.write(b'ok')
|
||||
else:
|
||||
self.send_response(400)
|
||||
self.end_headers()
|
||||
self.wfile.write(b'empty body')
|
||||
|
||||
def log_message(self, *a):
|
||||
pass
|
||||
|
||||
try:
|
||||
HTTPServer.allow_reuse_address = True
|
||||
srv = HTTPServer(('127.0.0.1', PORT), Handler)
|
||||
t = threading.Thread(target=srv.serve_forever, daemon=True)
|
||||
t.start()
|
||||
print(f'✓ Server: http://127.0.0.1:{PORT}/?state=walk')
|
||||
except OSError as e:
|
||||
if e.errno == 48:
|
||||
print(f'⚠ Port {PORT} already in use')
|
||||
else:
|
||||
raise
|
||||
|
||||
def run(self):
|
||||
"""Run the application (already in mainloop)"""
|
||||
pass
|
||||
|
||||
if __name__ == '__main__':
|
||||
if sys.platform == 'darwin':
|
||||
pet = MacPet()
|
||||
pet.run()
|
||||
else:
|
||||
pet = WinPet()
|
||||
Reference in New Issue
Block a user