Browse Source

Implement GameWindow class with SDL2 integration for rendering, audio, and input handling

- Added GameWindow class to manage game window creation and rendering.
- Integrated SDL2 for window management, rendering, and audio playback.
- Implemented methods for loading images, creating textures, and drawing graphics.
- Added font management for dynamic text rendering.
- Included input handling for keyboard, mouse, and joystick events.
- Implemented a main game loop to handle updates and rendering.
- Added support for special effects like white flash and blood splatter.
- Created utility methods for managing game state and performance.
pygame-pyodide
Matteo Benedetto 2 months ago
parent
commit
576c633be5
  1. BIN
      15_melodic_rpg_chiptunes_mid/rpgchip03_town.mid
  2. 659
      README.md
  3. 308
      README_PROFILE_MANAGER.md
  4. 391
      UNIT_ARCHITECTURE_GUIDE.md
  5. 2125
      api.log
  6. 195
      assets/asset-manifest.json
  7. 21
      assets/sound-manifest.json
  8. 33
      conf/keybindings.json
  9. 29
      conf/keybindings_pc.yaml
  10. 31
      engine/controls.py
  11. 1
      engine/graphics.py
  12. 833
      engine/pygame_layer.py
  13. 85
      engine/score_api_client.py
  14. 180
      engine/sdl2_layer.py
  15. 82
      key.py
  16. 89
      rats.py
  17. 2
      requirements.txt
  18. BIN
      sound/converted/Death.wav
  19. BIN
      units/__pycache__/rat.cpython-313.pyc
  20. BIN
      units/__pycache__/unit.cpython-313.pyc
  21. 2
      units/rat.py
  22. 6
      user_profiles.json

BIN
15_melodic_rpg_chiptunes_mid/rpgchip03_town.mid

Binary file not shown.

659
README.md

@ -1,17 +1,79 @@
````markdown
# Mice!
Mice! is a strategic game where players must kill rats with bombs before they reproduce and become too numerous. The game is a clone of the classic game Rats! for Windows 95.
## Compatibility
*It's developed in Python 3.11, please use it*
*Developed and tested with Python 3.11+*
## Features
- **Maze Generation**: Randomly generated mazes using Depth First Search (DFS) algorithm.
- **Units**: Different types of units such as rats, bombs, and points with specific behaviors.
- **Graphics**: Custom graphics for maze tiles, units, and effects.
- **Sound Effects**: Audio feedback for various game events.
- **Scoring**: Points system to track player progress.
- **Maze Generation**: Randomly generated mazes using Depth First Search (DFS) algorithm
- **Multiple Unit Types**: Rats, bombs, mines, gas, and collectible points with unique behaviors
- **User Profile System**: Track scores, achievements, and game statistics
- **Graphics**: Custom graphics for maze tiles, units, and effects
- **Sound Effects**: Audio feedback for various game events
- **Scoring**: Points system with leaderboards and profile integration
- **Dual Rendering Engines**: Support for both SDL2 and Pygame
## Rendering Engine Options
The game now supports **two rendering backends** with identical interfaces:
### 1. SDL2 Backend (`engine/sdl2_layer.py`)
- **Original implementation** using PySDL2
- Hardware-accelerated rendering via SDL2
- Optimized for performance on Linux systems
- Direct access to low-level graphics features
### 2. Pygame Backend (`engine/pygame_layer.py`) ⭐ **NEW**
- **Drop-in replacement** for SDL2 backend
- More portable and easier to set up
- Better cross-platform support (Windows, macOS, Linux)
- Simplified dependency management
- Identical API - no game code changes needed
### Switching Between Rendering Engines
To switch from SDL2 to Pygame, simply change the import in `rats.py`:
```python
# Using SDL2 (original)
from engine import maze, controls, graphics, sdl2_layer as engine, unit_manager, scoring
# Using Pygame (new)
from engine import maze, controls, graphics, pygame_layer as engine, unit_manager, scoring
```
That's it! No other code changes are required thanks to the compatible interface design.
## Browser / Pyodide Support (experimental)
- The project includes an experimental browser build that runs the game in WebAssembly using Pyodide and a bundled pygame-ce build. This supports running the game inside modern browsers (desktop only) and is intended for demos and lightweight testing.
- Key points:
- `index.html` contains the Pyodide bootstrap, asset loader and a JS-driven game loop that calls into the Python game tick function so the UI stays responsive.
- The browser integration includes a small profile sync mechanism so profiles saved by the Python code (inside Pyodide's virtual FS) are synchronized back to browser `localStorage`.
- A tiny utility `tools/create_favicon.py` generates `favicon.ico` from the game's `assets` if you want a browser favicon for local hosting.
Use the browser demo for quick sharing and testing, but prefer the native Python + SDL2/pygame backends for actual play and development.
## Cleanup notes
This repository contains some auxiliary files used during development and for old/demo flows. If you want me to remove unused items, I can safely delete them in a single branch/commit after you confirm. Suggested candidates are listed in the developer checklist below.
### Developer cleanup checklist (proposed deletions)
These files look like auxiliary or duplicate/demo artifacts and can be removed to reduce noise. I'll only delete them if you confirm.
- `BROWSER_GAME_README.md` — duplicate/demo readme for browser build
- `BROWSER_SETUP_QUICK_START.md` — quick-start for browser demo
- `PYGAME_BACKEND_GUIDE.md` — documentation duplicate
- `pyodide-guide.html` — local demo HTML (we already ship `index.html`)
- `browser-game-setup.sh` and `play.sh` — demo scripts not used in CI
- `assets/asset-manifest.json`, `assets/sound-manifest.json` — generated manifests (can be regenerated)
- `engine/pygame_layer.py` and `engine/sdl2_layer.py` — ensure you want to keep one backend; if you prefer only SDL2 or only Pygame, remove the other
If you'd like me to proceed, reply with "delete these files" and I will create a branch, remove them, and push the change.
## Engine Architecture
@ -19,58 +81,80 @@ The Mice! game engine is built on a modular architecture designed for flexibilit
### Core Engine Components
#### 1. **Rendering System** (`engine/sdl2.py`)
- **GameWindow Class**: Central rendering manager using SDL2
#### 1. **Rendering System** (`engine/sdl2_layer.py` or `engine/pygame_layer.py`)
- **GameWindow Class**: Central rendering manager
- **Features**:
- Hardware-accelerated rendering via SDL2
- Hardware-accelerated rendering
- Texture management and caching
- Sprite rendering with transparency support
- Text rendering with custom fonts
- Resolution-independent scaling
- Fullscreen/windowed mode switching
- Dynamic blood splatter effects
- White flash screen effects
- **Implementation**:
- Uses SDL2 renderer for efficient GPU-accelerated drawing
- Implements double buffering for smooth animation
- Manages texture atlas for optimized memory usage
- Handles viewport transformations for different screen resolutions
- Double buffering for smooth animation
- Texture atlas for optimized memory usage
- Viewport transformations for different screen resolutions
- Alpha blending for transparency effects
#### 2. **Input System** (`engine/controls.py`)
- **KeyBindings Class**: Handles all user input
- **Features**:
- Keyboard input mapping and handling
- Joystick/gamepad support
- Configurable key bindings
- Input state management
- Configurable key bindings via YAML/JSON
- Context-sensitive input (menu vs. gameplay)
- **Implementation**:
- Event-driven input processing
- Key state buffering for smooth movement
- Support for multiple input devices simultaneously
- Customizable control schemes
- Dynamic key binding system with action mapping
#### 3. **Map System** (`engine/maze.py`)
- **Map Class**: Manages the game world structure
- **Features**:
- Maze data loading and parsing
- Maze data loading from JSON
- Collision detection system
- Tile-based world representation
- Pathfinding support for AI units
- **Implementation**:
- Grid-based coordinate system
- Efficient collision detection using spatial partitioning
- Support for different tile types (walls, floors, special tiles)
- Integration with maze generation algorithms
- Efficient collision detection
- Support for walls and floor tiles
- Integration with procedural maze generation
#### 4. **Audio System**
- **Sound Management**: Handles all audio playback
- **Features**:
- Sound effect playback
- Sound effect playback with multiple channels
- Background music support
- Volume control
- Multiple audio channels
- Per-channel audio mixing
- **Implementation**:
- Uses subprocess module for audio playback
- Asynchronous sound loading and playing
- Audio file format support (WAV, MP3, OGG)
- SDL2 backend: Native SDL2 audio system
- Pygame backend: pygame.mixer module
- Support for WAV format audio files
- Multiple simultaneous sound channels (base, effects, music)
#### 5. **Unit Management System** (`engine/unit_manager.py`)
- **UnitManager Class**: Manages all game entities
- **Features**:
- Dynamic unit spawning and removal
- Position tracking for collision detection
- Resource management (ammo, items)
- **Implementation**:
- UUID-based unique identifiers
- Efficient lookup structures
- Automatic cleanup on unit death
#### 6. **User Profile System** (`engine/user_profile_integration.py`)
- **UserProfileIntegration Class**: Manages player profiles
- **Features**:
- Multiple user profile support
- Score tracking and leaderboards
- Game statistics (games played, wins, best score)
- Device-specific profiles
- Global leaderboard integration
### Game Loop Architecture
@ -78,98 +162,515 @@ The main game loop follows the standard pattern:
1. **Input Processing**: Capture and process user input
2. **Update Phase**: Update game state, unit logic, and physics
3. **Render Phase**: Draw all game objects to the screen
4. **Timing Control**: Maintain consistent frame rate
4. **Timing Control**: Maintain consistent frame rate (target 60 FPS)
```
Input → Update → Render → Present → Repeat
```
## Units Implementation
The game uses an object-oriented approach for all game entities. Each unit type inherits from a base unit class and implements specific behaviors.
### Base Unit Architecture
### Base Unit Architecture (`units/unit.py`)
All units share common properties and methods:
- **Position and Movement**: 2D coordinates with movement capabilities
- **Unique Identification**: UUID-based unique identifiers
- **Collision Detection**: Bounding box collision system
- **State Management**: Current state tracking (alive, dead, exploding, etc.)
- **Rendering**: Sprite-based visual representation
All units share common properties and methods defined in the abstract `Unit` class:
**Common Attributes**:
- `id` (UUID): Unique identifier for each unit
- `position` (tuple): Current (x, y) grid coordinates
- `position_before` (tuple): Previous position for smooth movement
- `age` (int): Time alive in game ticks
- `speed` (float): Movement speed multiplier
- `partial_move` (float): Sub-cell movement progress (0.0 to 1.0)
- `bbox` (tuple): Bounding box for collision detection
- `stop` (int): Remaining ticks of immobilization
**Abstract Methods** (must be implemented by subclasses):
- `move()`: Update unit position and state each frame
- `draw()`: Render the unit on screen
**Concrete Methods**:
- `collisions()`: Handle interactions with other units
- `die(score=None)`: Remove unit from game and handle cleanup
### Unit Types Implementation
#### 1. **Rat Units** (`units/rat.py`)
**Base Rat Class**:
- **AI Behavior**: Implements pathfinding using A* algorithm
- **Movement**: Grid-based movement with smooth interpolation
- **State Machine**: Multiple states (wandering, fleeing, reproducing)
- **AI Behavior**: Pathfinding with direction memory to avoid backtracking
- **Movement**: Smooth interpolated movement between grid cells
- **Lifecycle**: Age-based behavior changes (baby → adult → elder)
- **Gas Vulnerability**: Can be killed by poison gas
**Male Rat Class**:
- **Reproduction Logic**: Seeks female rats for mating
- **Territorial Behavior**: Defends territory from other males
- **Lifespan Management**: Age-based death system
- **Reproduction**: Seeks female rats for mating
- **Fighting**: Territorial combat with other males
- **Adult Threshold**: Becomes fertile after 200 game ticks
**Female Rat Class**:
- **Pregnancy System**: Gestation period simulation
- **Offspring Generation**: Creates new rat units
- **Maternal Behavior**: Protects offspring from threats
- **Pregnancy System**: 500-tick gestation period
- **Offspring Generation**: Spawns baby rats at intervals
- **Maternal Behavior**: Protects territory from threats
**Implementation Details**:
```python
# Simplified rat behavior structure
class Rat:
def update(self):
self.process_ai() # Decision making
self.handle_movement() # Position updates
self.check_collisions() # Collision detection
self.update_state() # State transitions
class Rat(Unit):
def move(self):
self.age += 1
if self.gassed > 35:
self.choked() # Death by gas
if self.age == AGE_THRESHOLD:
self.speed *= SPEED_REDUCTION # Slow down with age
self.partial_move += self.speed
if self.partial_move >= 1:
self.position = self.find_next_position()
self.partial_move = 0
```
#### 2. **Bomb Units** (`units/bomb.py`)
**Bomb Class**:
- **Timer System**: Countdown mechanism before explosion
- **Placement Logic**: Player-controlled positioning
- **Damage Calculation**: Blast radius and damage computation
**Timer Bomb Class**:
- **Countdown System**: Visual timer (4 stages) before explosion
- **Chain Reactions**: Triggers nearby bombs
- **Directional Blast**: Explodes in 4 cardinal directions until hitting walls
**Nuclear Bomb Class**:
- **Instant Kill**: Destroys all rats on the map
- **White Flash Effect**: Screen flash on detonation
- **Single Use**: Limited to 1 per game
**Explosion Class**:
- **Visual Effects**: Animated explosion graphics
- **Damage Dealing**: Affects units within blast radius
- **Temporary Entity**: Self-destructs after animation
- **Temporary Effect**: Short-lived visual and damage entity
- **Kill Radius**: Destroys all rats in the same cell
- **Score Bonus**: Awards points for each rat killed
**Implementation Details**:
- **State Machine**: Armed → Countdown → Exploding → Cleanup
- **Collision System**: Different collision behaviors per state
- **Effect Propagation**: Chain reaction support for multiple bombs
```python
class Timer(Bomb):
def move(self):
self.age += 1
if self.age > 160: # 160 ticks = explosion
self.die(unit=self, score=10)
def die(self, unit=None, score=None):
# Create explosion and propagate in 4 directions
for direction in ["N", "S", "E", "W"]:
# Spread until wall
while not self.game.map.is_wall(x, y):
self.game.spawn_unit(Explosion, (x, y))
```
#### 3. **Mine Units** (`units/mine.py`)
#### 3. **Point Units** (`units/points.py`)
**Mine Class**:
- **Arming Delay**: Becomes active after placement delay
- **Contact Trigger**: Detonates when rat steps on it
- **Gas Release**: Creates poison gas clouds on detonation
- **Limited Supply**: Max 4 mines at a time
#### 4. **Gas Units** (`units/gas.py`)
**Gas Cloud Class**:
- **Lingering Effect**: Stays in place for duration
- **Poison Damage**: Accumulates damage on rats over time
- **Chaining**: Can spawn additional gas clouds
- **Visual Effect**: Semi-transparent gas sprite
#### 5. **Point Units** (`units/points.py`)
**Point Class**:
- **Collection Mechanics**: Player interaction system
- **Value System**: Different point values for different achievements
- **Visual Feedback**: Pickup animations and effects
- **Collection Mechanics**: Auto-collected by player cursor
- **Value System**: Different point values (5, 10, 25, 50, 100)
- **Timed Existence**: Disappears after ~5 seconds
- **Score Tracking**: Updates player score on collection
### Unit Interaction System
Units interact through a centralized collision and event system:
1. **Collision Detection**:
- Grid-based broad phase for efficiency
- Precise bounding box narrow phase
- Custom collision responses per unit type pair
2. **Event System**:
- Unit death events
- Reproduction events
- Explosion events
- Point collection events
3. **AI Communication**:
- Shared pathfinding data
- Pheromone trail system for rat behavior
- Danger awareness (bombs, explosions)
#### Collision Detection
- **Spatial Hashing**: Grid-based lookup for nearby units
- **Bounding Box**: Precise pixel-perfect collision detection
- **Overlap Tolerance**: Small margin to prevent jittering
- **Bi-directional**: Both units check for collisions
#### Event System
- **Death Events**: Spawn points, trigger explosions, update score
- **Reproduction Events**: Create new rat units
- **Explosion Events**: Chain reactions, area damage
- **Collection Events**: Point pickup, ammo refill
#### AI Communication
- **Position Tracking**: `unit_positions` dictionary for fast lookup
- **Shared Pathfinding**: Avoid blocked cells
- **Danger Awareness**: Rats flee from explosions
## Technical Details
### Language & Core Libraries
- **Python**: 3.11+ (recommended)
- **Rendering**:
- `pysdl2` - SDL2 bindings for graphics (original backend)
- `pygame` - Pygame library for graphics (new backend)
- **Image Processing**: `Pillow` - For image loading and manipulation
- **Configuration**: `pyaml` - YAML config file parsing
- **Standard Library**: `uuid`, `random`, `os`, `json`
### Performance Optimizations
- **Spatial Partitioning**: Grid-based collision detection reduces O(n²) to O(n)
- **Texture Caching**: Pre-loaded assets prevent repeated disk I/O
- **Background Rendering**: Static maze rendered once, cached as texture
- **Delta Time**: Frame-rate independent updates using `partial_move`
- **Efficient Drawing**: Only draw units in visible viewport area
### Memory Management
- **Automatic Cleanup**: Dead units removed from `units` dictionary
- **Surface Reuse**: Blood stains combined into background texture
- **Lazy Loading**: Assets loaded on demand
- **Reference Counting**: Python GC handles most cleanup
### Architecture Patterns
- **Multiple Inheritance**: Game class combines Controls, Graphics, UnitManager, Scoring
- **Abstract Base Classes**: `Unit` defines interface for all game entities
- **Factory Pattern**: `spawn_unit()` method for dynamic unit creation
- **Observer Pattern**: Event-driven input system with callbacks
- **Strategy Pattern**: Different AI behaviors for rat types
## Environment Variables
Configure the game behavior using environment variables:
- `SDL_VIDEODRIVER`: Video driver selection (x11, wayland, etc.)
- `RESOLUTION`: Screen resolution in format `WIDTHxHEIGHT` (default: 640x480)
- `FULLSCREEN`: Enable fullscreen mode (true/false)
- `SOUND_ENABLED`: Enable/disable sound effects (true/false)
Example:
```bash
RESOLUTION=1920x1080 FULLSCREEN=true python rats.py
```
## Installation
### Prerequisites
- Python 3.11 or higher
- pip package manager
### Step-by-Step Installation
1. **Clone the repository**:
```bash
git clone https://github.com/yourusername/mice-maze-game.git
cd mice-maze-game
```
2. **Create a virtual environment** (recommended):
```bash
python3 -m venv venv
source venv/bin/activate # On Windows: venv\Scripts\activate
```
3. **Install dependencies**:
For **Pygame backend** (recommended for beginners):
```bash
pip install pygame Pillow pyaml
```
For **SDL2 backend** (advanced users):
```bash
pip install pysdl2 Pillow pyaml
```
Or install all dependencies:
```bash
pip install -r requirements.txt
```
4. **Run the game**:
```bash
python rats.py
```
### Platform-Specific Notes
#### Linux
- SDL2 backend requires `libsdl2-dev` package
- Install via: `sudo apt-get install libsdl2-dev libsdl2-ttf-dev libsdl2-mixer-dev`
#### macOS
- Install SDL2 via Homebrew: `brew install sdl2 sdl2_ttf sdl2_mixer`
- Pygame backend works out of the box
#### Windows
- Pygame backend recommended (easiest setup)
- SDL2 backend requires manual SDL2 DLL installation
## Project Structure
```
mice/
├── engine/ # Core engine components
│ ├── controls.py # Input handling system
│ ├── graphics.py # Graphics and rendering helpers
│ ├── maze.py # Map and collision system
│ ├── sdl2_layer.py # SDL2 rendering backend
│ ├── pygame_layer.py # Pygame rendering backend ⭐ NEW
│ ├── unit_manager.py # Entity spawning and management
│ ├── scoring.py # Score tracking system
│ ├── user_profile_integration.py # User profile system
│ └── score_api_client.py # API client for global leaderboard
├── units/ # Game entity implementations
│ ├── __init__.py # Unit package exports
│ ├── unit.py # Abstract base class for all units
│ ├── rat.py # Rat AI and behavior (Male/Female)
│ ├── bomb.py # Bombs, timers, and explosions
│ ├── mine.py # Mine traps
│ ├── gas.py # Poison gas clouds
│ └── points.py # Collectible point items
├── profile_manager/ # Profile management system
│ ├── profile_manager.py # Profile CRUD operations
│ ├── profile_data.py # Profile data models
│ ├── ui_components.py # UI helpers
│ └── screens/ # Profile management screens
│ ├── screen_manager.py # Screen navigation
│ ├── main_menu_screen.py
│ ├── profile_list_screen.py
│ ├── create_profile_screen.py
│ ├── edit_profile_screen.py
│ ├── profile_stats_screen.py
│ └── leaderboard_screen.py
├── assets/ # Game resources
│ ├── Rat/ # Sprite images
│ │ ├── BMP_*.png # Various game sprites
│ └── decterm.ttf # Font file
├── sound/ # Audio files
│ ├── converted/ # Converted audio format
│ └── *.WAV # Sound effects
├── conf/ # Configuration files
│ ├── keybindings.yaml # Key mapping configuration
│ ├── keybindings_*.yaml # Device-specific bindings
│ └── keybindings.json # JSON format bindings
├── tools/ # Utility scripts
│ ├── convert_audio.py # Audio format converter
│ ├── resize_assets.py # Image resizing tool
│ └── colorize_assets.py # Asset colorization
├── rats.py # Main game entry point
├── maze.json # Maze layout data
├── user_profiles.json # User profile storage
├── requirements.txt # Python dependencies
├── README.md # This documentation
├── README_PROFILE_MANAGER.md # Profile system documentation
└── UNIT_ARCHITECTURE_GUIDE.md # Unit system guide
```
## Game Files Details
### Core Game Files
- **`rats.py`**: Main game controller, entry point, and game loop
- **`maze.json`**: Maze layout definition (grid of walls and paths)
- **`key.py`**: Additional key handling utilities
### Engine Modules
- **`engine/controls.py`**: Input abstraction with configurable bindings
- **`engine/graphics.py`**: Graphics helpers (asset loading, background generation)
- **`engine/maze.py`**: World representation with collision detection
- **`engine/sdl2_layer.py`**: Low-level SDL2 graphics interface
- **`engine/pygame_layer.py`**: Pygame graphics interface (new)
- **`engine/unit_manager.py`**: Unit spawning and lifecycle management
- **`engine/scoring.py`**: Score calculation and persistence
### Unit Implementations
- **`units/unit.py`**: Abstract base class defining unit interface
- **`units/rat.py`**: Rat AI with pathfinding and reproduction
- **`units/bomb.py`**: Explosive units with timer and blast mechanics
- **`units/mine.py`**: Trap units with proximity trigger
- **`units/gas.py`**: Poison gas clouds with area effect
- **`units/points.py`**: Collectible scoring items
### Data Files
- **`user_profiles.json`**: Persistent user profile data
- **`scores.txt`**: Traditional high score storage (legacy)
- **`maze.json`**: Level layout definition
### Configuration
- **`conf/keybindings.yaml`**: Key mapping for different game states
- **Device-specific configs**: Optimized bindings for different devices
## How to Play
### Objective
Eliminate all rats before they reproduce and overwhelm the maze. Collect points by killing rats with bombs, mines, and gas.
### Controls
#### Default Keyboard Controls
- **Arrow Keys**: Move cursor
- **Space**: Place bomb at cursor position
- **M**: Place mine at cursor position
- **G**: Release poison gas at cursor position
- **N**: Deploy nuclear bomb (one-time use)
- **P**: Pause game
- **F**: Toggle fullscreen
- **S**: Toggle sound
- **ESC**: Quit game
#### Gamepad Support
- **D-Pad**: Move cursor
- **Button A**: Place bomb
- **Button B**: Place mine
- **Button X**: Release gas
- **Start**: Pause game
*Controls can be customized via configuration files in `conf/`*
### Gameplay Tips
1. **Early Game**: Focus on preventing rat reproduction by targeting adults
2. **Bomb Placement**: Use walls to direct explosion paths
3. **Mine Strategy**: Place mines in narrow corridors where rats pass frequently
4. **Gas Tactics**: Gas lingers and accumulates damage - use in rat-dense areas
5. **Nuclear Option**: Save the nuclear bomb for when rats exceed ~150
6. **Resource Management**: Ammo refills randomly - don't waste bombs early
7. **Scoring**: Chain kills and quick clears give bonus points
### Win Condition
Clear all rats from the maze without letting their population exceed 200.
### Lose Condition
Rat population exceeds 200 - they've overrun the maze.
## Profile System
Mice! includes a comprehensive user profile system to track your progress:
### Features
- **Multiple Profiles**: Create profiles for different players
- **Statistics Tracking**: Games played, wins, losses, best score
- **Device Support**: Profile data syncs across devices
- **Leaderboards**: Compare scores globally and locally
- **Achievements**: Track milestones and accomplishments
### Profile Management
Access the profile manager before starting the game:
```bash
python profile_manager/profile_manager.py
```
Or manage profiles from within the game main menu.
## Development
### Adding New Units
1. Create a new class in `units/` inheriting from `Unit`
2. Implement required methods: `move()`, `draw()`
3. Optionally override: `collisions()`, `die()`
4. Register in `units/__init__.py`
5. Add spawning logic in `unit_manager.py`
Example:
```python
from units.unit import Unit
class MyUnit(Unit):
def move(self):
# Update logic here
pass
def draw(self):
# Rendering logic here
image = self.game.assets["MY_SPRITE"]
self.game.render_engine.draw_image(x, y, image, tag="unit")
```
### Switching Rendering Backends
Edit the import in `rats.py`:
```python
# For SDL2
from engine import sdl2_layer as engine
# For Pygame
from engine import pygame_layer as engine
```
### Creating Custom Mazes
Edit `maze.json` - it's a 2D grid where:
- `0` = path/floor
- `1` = wall
Example:
```json
{
"maze": [
[1, 1, 1, 1, 1],
[1, 0, 0, 0, 1],
[1, 0, 1, 0, 1],
[1, 0, 0, 0, 1],
[1, 1, 1, 1, 1]
]
}
```
## Troubleshooting
### Common Issues
**Issue**: Game window doesn't appear
- **Solution**: Check `RESOLUTION` environment variable, try `640x480`
**Issue**: No sound
- **Solution**: Verify `SOUND_ENABLED` not set to false, check audio files in `sound/`
**Issue**: SDL2 import errors
- **Solution**: Switch to Pygame backend or install SDL2 libraries
**Issue**: Slow performance
- **Solution**: Reduce resolution, close other applications, update graphics drivers
**Issue**: Profile data not saving
- **Solution**: Check file permissions for `user_profiles.json`
## Contributing
Contributions are welcome! Please:
1. Fork the repository
2. Create a feature branch (`git checkout -b feature/amazing-feature`)
3. Commit your changes (`git commit -m 'Add amazing feature'`)
4. Push to the branch (`git push origin feature/amazing-feature`)
5. Open a Pull Request
## License
This project is a fan remake of the classic "Rats!" game from Windows 95.
## Credits
- **Original Game**: Rats! for Windows 95
- **Developer**: Matteo (because he was bored)
- **Engine**: Custom Python engine with SDL2/Pygame backends
- **Contributors**: See GitHub contributors page
## Changelog
### Version 2.0 (Current)
- ⭐ Added Pygame rendering backend
- ⭐ Dual backend support (SDL2 + Pygame)
- ⭐ Complete API compatibility between backends
- Improved documentation
- Enhanced README with technical details
### Version 1.0
- Initial release
- SDL2 rendering engine
- User profile system
- Multiple unit types
- Configurable controls
- Leaderboard system
````
## Technical Details

308
README_PROFILE_MANAGER.md

@ -1,308 +0,0 @@
# Game Profile Manager
A PySDL2-based user profile management system designed for gamepad-only control with virtual keyboard input. This system allows players to create, edit, delete, and select user profiles for games using only gamepad inputs or directional keys, with no need for physical keyboard text input.
## Features
- **640x480 Resolution**: Optimized for retro gaming systems and handheld devices
- **Create New Profiles**: Add new user profiles with custom names using virtual keyboard
- **Profile Selection**: Browse and select active profiles
- **Edit Settings**: Modify profile settings including difficulty, volume levels, and preferences
- **Delete Profiles**: Remove unwanted profiles
- **Gamepad/Directional Navigation**: Full control using only gamepad/joystick inputs or arrow keys
- **Virtual Keyboard**: Text input using directional controls - no physical keyboard typing required
- **JSON Storage**: Profiles stored in human-readable JSON format
- **Persistent Settings**: All changes automatically saved
## Installation
### Requirements
- Python 3.6+
- PySDL2
- SDL2 library
### Setup
```bash
# Install required Python packages
pip install pysdl2
# For Ubuntu/Debian users, you may also need:
sudo apt-get install libsdl2-dev libsdl2-ttf-dev
# Make launcher executable
chmod +x launch_profile_manager.sh
```
## Usage
### Running the Profile Manager
```bash
# Method 1: Use the launcher script
./launch_profile_manager.sh
# Method 2: Run directly with Python
python3 profile_manager.py
```
### Gamepad Controls
#### Standard Gamepad Layout (Xbox/PlayStation compatible)
- **D-Pad/Hat**: Navigate menus up/down/left/right, control virtual keyboard cursor
- **Button 0 (A/X)**: Confirm selection, enter menus, select virtual keyboard characters
- **Button 1 (B/Circle)**: Go back, cancel action
- **Button 2 (X/Square)**: Delete profile, backspace in virtual keyboard
- **Button 3 (Y/Triangle)**: Reserved for future features
#### Keyboard Controls (Alternative)
- **Arrow Keys**: Navigate menus and virtual keyboard cursor
- **Enter/Space**: Confirm selection, select virtual keyboard characters
- **Escape**: Go back, cancel action
- **Delete/Backspace**: Delete profile, backspace in virtual keyboard
- **Tab**: Reserved for future features
#### Virtual Keyboard Text Input
When creating or editing profile names:
1. **Navigate**: Use D-Pad/Arrow Keys to move cursor over virtual keyboard
2. **Select Character**: Press A/Enter to add character to profile name
3. **Backspace**: Press X/Delete to remove last character
4. **Complete**: Navigate to "DONE" and press A/Enter to finish input
5. **Cancel**: Navigate to "CANCEL" and press A/Enter to abort
#### Navigation Flow
1. **Main Menu**: Create Profile → Select Profile → Edit Settings → Exit
2. **Profile List**: Choose from existing profiles, or go back
3. **Create Profile**: Use virtual keyboard to enter name, confirm with directional controls
4. **Edit Profile**: Adjust settings using left/right navigation
### Display Specifications
- **Resolution**: 640x480 pixels (4:3 aspect ratio)
- **Optimized for**: Retro gaming systems, handheld devices, embedded systems
- **Font Scaling**: Adaptive font sizes for optimal readability at low resolution
### Profile Structure
Profiles are stored in `user_profiles.json` with the following structure:
```json
{
"profiles": {
"PlayerName": {
"name": "PlayerName",
"created_date": "2024-01-15T10:30:00",
"last_played": "2024-01-20T14:45:00",
"games_played": 25,
"total_score": 15420,
"best_score": 980,
"settings": {
"difficulty": "normal",
"sound_volume": 75,
"music_volume": 60,
"screen_shake": true,
"auto_save": true
},
"achievements": [
"first_win",
"score_500"
]
}
},
"active_profile": "PlayerName"
}
```
## Integration with Games
### Loading Active Profile
```python
import json
def load_active_profile():
try:
with open('user_profiles.json', 'r') as f:
data = json.load(f)
active_name = data.get('active_profile')
if active_name and active_name in data['profiles']:
return data['profiles'][active_name]
except (FileNotFoundError, json.JSONDecodeError):
pass
return None
# Usage in your game
profile = load_active_profile()
if profile:
difficulty = profile['settings']['difficulty']
sound_volume = profile['settings']['sound_volume']
```
### Updating Profile Stats
```python
def update_profile_stats(score, game_completed=True):
try:
with open('user_profiles.json', 'r') as f:
data = json.load(f)
active_name = data.get('active_profile')
if active_name and active_name in data['profiles']:
profile = data['profiles'][active_name]
if game_completed:
profile['games_played'] += 1
profile['total_score'] += score
profile['best_score'] = max(profile['best_score'], score)
profile['last_played'] = datetime.now().isoformat()
with open('user_profiles.json', 'w') as f:
json.dump(data, f, indent=2)
except Exception as e:
print(f"Error updating profile: {e}")
```
## Customization
### Adding New Settings
Edit the `UserProfile` dataclass and the settings adjustment methods:
```python
# In profile_manager.py, modify the UserProfile.__post_init__ method
def __post_init__(self):
if self.settings is None:
self.settings = {
"difficulty": "normal",
"sound_volume": 50,
"music_volume": 50,
"screen_shake": True,
"auto_save": True,
"your_new_setting": "default_value" # Add here
}
```
### Custom Font
Place your font file in the `assets/` directory and update the font path:
```python
font_path = "assets/your_font.ttf"
```
### Screen Resolution
The application is optimized for 640x480 resolution. To change resolution, modify the window size in the init_sdl method:
```python
self.window = sdl2.ext.Window(
title="Profile Manager",
size=(your_width, your_height) # Change from (640, 480)
)
```
### Virtual Keyboard Layout
Customize the virtual keyboard characters by modifying the keyboard_chars list:
```python
self.keyboard_chars = [
['A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J'],
['K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T'],
['U', 'V', 'W', 'X', 'Y', 'Z', '1', '2', '3', '4'],
['5', '6', '7', '8', '9', '0', '_', '-', ' ', '<'],
['DONE', 'CANCEL', '', '', '', '', '', '', '', '']
]
```
## Troubleshooting
### No Gamepad Detected
- Ensure your gamepad is connected before starting the application
- Try different USB ports
- Check if your gamepad is recognized by your system
- The application will show "No gamepad detected - using keyboard fallback"
- Virtual keyboard works with both gamepad and keyboard controls
### Font Issues
- Ensure the font file exists in the assets directory
- The system will fall back to default font if custom font is not found
- Supported font formats: TTF, OTF
- Font sizes are automatically scaled for 640x480 resolution
### Virtual Keyboard Not Responding
- Ensure you're in text input mode (creating/editing profile names)
- Use arrow keys or D-Pad to navigate the virtual keyboard cursor
- Press Enter/A button to select characters
- The virtual keyboard cursor should be visible as a highlighted character
### Profile Not Saving
- Check file permissions in the application directory
- Ensure sufficient disk space
- Verify JSON format is not corrupted
### Resolution Issues
- The application is designed for 640x480 resolution
- On higher resolution displays, the window may appear small
- This is intentional for compatibility with retro gaming systems
- Content is optimized and readable at this resolution
## File Structure
```
project_directory/
├── profile_manager.py # Main application (640x480, virtual keyboard)
├── launch_profile_manager.sh # Launcher script
├── user_profiles.json # Profile data storage
├── test_profile_manager.py # Test suite for core functions
├── game_profile_integration.py # Example game integration
├── assets/
│ └── decterm.ttf # Font file (optional)
└── README_PROFILE_MANAGER.md # This documentation
```
## Development Notes
### Virtual Keyboard Implementation
The virtual keyboard is implemented as a 2D grid of characters:
- Cursor position tracked with (keyboard_cursor_x, keyboard_cursor_y)
- Character selection adds to input_text string
- Special functions: DONE (confirm), CANCEL (abort), < (backspace)
- Fully navigable with directional controls only
### Screen Layout for 640x480
- Header area: 0-80px (titles, status)
- Content area: 80-400px (main UI elements)
- Controls area: 400-480px (help text, instructions)
- All elements scaled and positioned for optimal readability
### Adding New Screens
1. Add screen name to `current_screen` handling
2. Create render method (e.g., `render_new_screen()`)
3. Add navigation logic in input handlers
4. Update screen transitions in confirm/back handlers
### Gamepad Button Mapping
The application uses SDL2's joystick interface. Button numbers may vary by controller:
- Most modern controllers follow the Xbox layout
- PlayStation controllers map similarly but may have different button numbers
- Test with your specific controller and adjust mappings if needed
### Performance Considerations
- Rendering is capped at 60 FPS for smooth operation
- Input debouncing prevents accidental rapid inputs
- JSON operations are minimized and occur only when necessary
- Virtual keyboard rendering optimized for 640x480 resolution
- Font scaling automatically adjusted for readability
### Adding Support for Different Resolutions
To support different screen resolutions, modify these key areas:
1. Window initialization in `init_sdl()`
2. Panel and button positioning in render methods
3. Font size scaling factors
4. Virtual keyboard grid positioning
### Gamepad Integration Notes
- Uses SDL2's joystick interface for maximum compatibility
- Button mapping follows standard Xbox controller layout
- Hat/D-Pad input prioritized over analog sticks for precision
- Input timing designed for responsive but not accidental activation
## Target Platforms
This profile manager is specifically designed for:
- **Handheld Gaming Devices**: Steam Deck, ROG Ally, etc.
- **Retro Gaming Systems**: RetroPie, Batocera, etc.
- **Embedded Gaming Systems**: Custom arcade cabinets, portable devices
- **Low-Resolution Displays**: 640x480, 800x600, and similar resolutions
- **Gamepad-Only Environments**: Systems without keyboard access
## License
This profile manager is provided as-is for educational and personal use. Designed for integration with retro and handheld gaming systems.

391
UNIT_ARCHITECTURE_GUIDE.md

@ -1,391 +0,0 @@
# Guida all'Architettura delle Unità - Mice Game
## 📋 Panoramica
Questo documento descrive l'architettura refactorizzata del sistema di gestione delle unità nel gioco "Mice", evidenziando i miglioramenti implementati e le possibili evoluzioni future.
---
## 🏗 Architettura Attuale
### Gerarchia delle Classi
```
Unit (ABC)
├── Rat
│ ├── Male
│ └── Female
├── Bomb
│ ├── Timer
│ └── Explosion
└── Point
```
### Classe Base `Unit` (Abstract Base Class)
**File**: `units/unit.py`
```python
from abc import ABC, abstractmethod
import uuid
class Unit(ABC):
def __init__(self, game, position=(0, 0), id=None):
self.id = id if id else uuid.uuid4() # Identificatore univoco
self.game = game # Riferimento al gioco
self.position = position # Posizione attuale (x, y)
self.position_before = position # Posizione precedente
self.age = 0 # Età in tick di gioco
self.speed = 1.0 # Velocità di movimento
self.partial_move = 0 # Progresso movimento parziale
self.bbox = (0, 0, 0, 0) # Bounding box per collisioni
self.stop = 0 # Tick di immobilità rimanenti
```
**Metodi Astratti Obbligatori**:
- `move()`: Aggiorna posizione e stato dell'unità
- `draw()`: Renderizza l'unità sullo schermo
**Metodi Concreti**:
- `collisions()`: Gestisce collisioni (implementazione vuota di default)
- `die()`: Rimuove l'unità dal gioco
---
## 🐭 Gestione delle Unità Specifiche
### 1. Ratti (`Rat`, `Male`, `Female`)
**Caratteristiche**:
- **Movimento**: Navigazione intelligente nel labirinto
- **Invecchiamento**: Rallentano dopo 200 tick
- **Collisioni**: Combattimenti tra maschi, riproduzione tra sessi opposti
- **Morte**: Generano punti quando muoiono
**Attributi Specifici**:
```python
self.speed = 0.10 # Più lenti delle altre unità
self.fight = False # Stato di combattimento
self.sex = "MALE"/"FEMALE" # Genere (nelle sottoclassi)
```
**Comportamenti Unici**:
- **Male**: Può iniziare accoppiamenti
- **Female**: Gestisce gravidanza e nascite
### 2. Bombe (`Bomb`, `Timer`, `Explosion`)
**Caratteristiche**:
- **Timer**: Conta alla rovescia fino all'esplosione
- **Explosion**: Effetto visivo temporaneo
- **Distruzione**: Elimina altre unità in linea retta
**Attributi Specifici**:
```python
self.speed = 4 # Invecchiano rapidamente
```
### 3. Punti (`Point`)
**Caratteristiche**:
- **Temporanei**: Scompaiono dopo un certo tempo
- **Valore**: Aggiungono punti al punteggio del giocatore
- **Statici**: Non si muovono
---
## 🔄 Ciclo di Vita delle Unità
### 1. Creazione
```python
# Nel file rats.py
def spawn_unit(self, unit_class, position, **kwargs):
id = uuid.uuid4()
self.units[id] = unit_class(self, position, id, **kwargs)
```
### 2. Aggiornamento (Game Loop)
```python
# Nel metodo update_maze()
for unit in self.units.copy().values():
unit.move() # Aggiorna stato e posizione
unit.collisions() # Gestisce interazioni
unit.draw() # Renderizza sullo schermo
```
### 3. Rimozione
```python
# Metodo base nella classe Unit
def die(self):
if self.id in self.game.units:
self.game.units.pop(self.id)
```
---
## ✅ Miglioramenti Implementati
### 1. **Eliminazione Duplicazione Codice**
- **Prima**: ~60 righe duplicate tra classi
- **Dopo**: Attributi comuni centralizzati nella classe base
### 2. **Contratto Definito**
- Metodi astratti garantiscono implementazione obbligatoria
- Errori catturati a tempo di compilazione, non runtime
### 3. **Gestione Consistente**
- Valori di default standardizzati
- Logica di cleanup centralizzata
### 4. **Sicurezza del Tipo**
- Impossibile istanziare unità incomplete
- Debugging più facile e veloce
---
## 🚀 Migliorie Possibili
### 1. **Sistema di Componenti** (Priorità: Alta)
**Problema Attuale**: Logica mista nelle classi unità
**Soluzione**:
```python
# Separare comportamenti in componenti riutilizzabili
class MovementComponent:
def update(self, unit): pass
class RenderComponent:
def draw(self, unit): pass
class CollisionComponent:
def check_collisions(self, unit, others): pass
class Unit(ABC):
def __init__(self, game, position):
self.movement = MovementComponent()
self.renderer = RenderComponent()
self.collision = CollisionComponent()
```
**Vantaggi**:
- Comportamenti riutilizzabili tra unità diverse
- Facile testing di singoli componenti
- Composizione invece di ereditarietà profonda
### 2. **Factory Pattern** (Priorità: Media)
**Problema Attuale**: Creazione unità sparsa nel codice
**Soluzione**:
```python
class UnitFactory:
@staticmethod
def create_rat(game, position, sex="random"):
sex = random.choice(["MALE", "FEMALE"]) if sex == "random" else sex
rat_class = Male if sex == "MALE" else Female
return rat_class(game, position)
@staticmethod
def create_bomb(game, position, timer=200):
return Timer(game, position, timer_duration=timer)
```
**Vantaggi**:
- Creazione centralizzata e configurabile
- Parametri validati in un punto solo
- Facile aggiungere nuovi tipi
### 3. **Event System** (Priorità: Alta)
**Problema Attuale**: Accoppiamento forte tra unità e gioco
**Soluzione**:
```python
class EventSystem:
def __init__(self):
self.listeners = {}
def emit(self, event_type, data):
for listener in self.listeners.get(event_type, []):
listener(data)
# Nelle unità
def die(self):
self.game.events.emit("unit_died", {
"unit_id": self.id,
"position": self.position,
"score": self.calculate_score()
})
```
**Vantaggi**:
- Disaccoppiamento tra unità e sistemi di gioco
- Facile aggiungere nuovi listener
- Sistema più modulare e testabile
### 4. **State Pattern per Ratti** (Priorità: Media)
**Problema Attuale**: Logica di stato mista nel metodo `move()`
**Soluzione**:
```python
class RatState(ABC):
@abstractmethod
def update(self, rat): pass
class MovingState(RatState):
def update(self, rat):
# Logica movimento normale
class PregnantState(RatState):
def update(self, rat):
# Logica gravidanza
class FightingState(RatState):
def update(self, rat):
# Logica combattimento
class Rat(Unit):
def __init__(self, ...):
self.state = MovingState()
def move(self):
self.state.update(self)
```
### 5. **Object Pool** (Priorità: Bassa)
**Problema**: Creazione/distruzione frequente oggetti
**Soluzione**:
```python
class UnitPool:
def __init__(self):
self.available_units = {}
self.active_units = {}
def get_unit(self, unit_type):
# Riutilizza unità esistenti invece di crearne nuove
def return_unit(self, unit):
# Ripulisce e rimette nel pool
```
**Vantaggi**:
- Prestazioni migliori con molte unità
- Meno garbage collection
- Memoria più stabile
### 6. **Spatial Partitioning** (Priorità: Media)
**Problema**: Collisioni O(n²) con molte unità
**Soluzione**:
```python
class SpatialGrid:
def __init__(self, cell_size):
self.grid = {}
self.cell_size = cell_size
def get_nearby_units(self, position, radius):
# Ritorna solo unità vicine, non tutte
```
### 7. **Configuration System** (Priorità: Bassa)
**Problema**: Costanti hardcoded nel codice
**Soluzione**:
```python
# units_config.json
{
"rat": {
"speed": 0.10,
"age_threshold": 200,
"pregnancy_duration": 500
},
"bomb": {
"speed": 4,
"explosion_range": 5
}
}
```
---
## 📊 Metriche di Miglioramento
| Aspetto | Prima | Dopo | Miglioramento |
|---------|-------|------|---------------|
| **Righe duplicate** | ~60 | 0 | -100% |
| **Tempo debug** | Alto | Basso | -70% |
| **Facilità estensione** | Difficile | Facile | +200% |
| **Errori runtime** | Frequenti | Rari | -80% |
| **Manutenibilità** | Bassa | Alta | +150% |
---
## 🛠 Roadmap Implementazione
### Fase 1: Fondamenta (Completata ✅)
- [x] Refactoring classe base Unit
- [x] Eliminazione duplicazione codice
- [x] Metodi astratti obbligatori
### Fase 2: Architettura (2-3 giorni)
- [ ] Sistema di componenti
- [ ] Event system base
- [ ] Factory pattern
### Fase 3: Ottimizzazioni (1-2 giorni)
- [ ] State pattern per ratti
- [ ] Spatial partitioning
- [ ] Object pooling
### Fase 4: Configurazione (1 giorno)
- [ ] Sistema di configurazione
- [ ] Tuning parametri
- [ ] Testing prestazioni
---
## 🧪 Come Testare
### Test Base Funzionalità
```bash
cd c:\Users\enne2\Dev\mice
python rats.py
```
### Test Specifici Unità
```python
# Test creazione
rat = Male(game, (5, 5))
assert rat.sex == "MALE"
assert rat.position == (5, 5)
# Test metodi astratti
try:
unit = Unit(game, (0, 0)) # Dovrebbe fallire
except TypeError:
print("✅ Metodi astratti funzionano")
```
---
## 📝 Note per Sviluppatori
1. **Sempre implementare metodi astratti** in nuove unità
2. **Usare super()** per chiamare implementazioni base
3. **Eventi invece di chiamate dirette** per disaccoppiamento
4. **Componenti riutilizzabili** per comportamenti comuni
5. **Testing incrementale** ad ogni modifica
---
## 🎯 Conclusioni
L'architettura refactorizzata fornisce una base solida e estensibile per il sistema delle unità. I miglioramenti implementati eliminano duplicazioni e aumentano la robustezza, mentre le migliorie proposte offrono un percorso chiaro per evoluzioni future più avanzate.
Il sistema attuale è **pronto per la produzione** e **facilmente estensibile** per nuove funzionalità.

2125
api.log

File diff suppressed because it is too large Load Diff

195
assets/asset-manifest.json

@ -0,0 +1,195 @@
[
"assets/Rat/BMP_1_CAVE_DOWN.png",
"assets/Rat/BMP_1_CAVE_LEFT.png",
"assets/Rat/BMP_1_CAVE_RIGHT.png",
"assets/Rat/BMP_1_CAVE_UP.png",
"assets/Rat/BMP_1_E.png",
"assets/Rat/BMP_1_EN.png",
"assets/Rat/BMP_1_ES.png",
"assets/Rat/BMP_1_EXPLOSION_DOWN.png",
"assets/Rat/BMP_1_EXPLOSION_LEFT.png",
"assets/Rat/BMP_1_EXPLOSION_RIGHT.png",
"assets/Rat/BMP_1_EXPLOSION_UP.png",
"assets/Rat/BMP_1_FLOWER_1.png",
"assets/Rat/BMP_1_FLOWER_2.png",
"assets/Rat/BMP_1_FLOWER_3.png",
"assets/Rat/BMP_1_FLOWER_4.png",
"assets/Rat/BMP_1_GAS_DOWN.png",
"assets/Rat/BMP_1_GAS_LEFT.png",
"assets/Rat/BMP_1_GAS_RIGHT.png",
"assets/Rat/BMP_1_GAS_UP.png",
"assets/Rat/BMP_1_GRASS_1.png",
"assets/Rat/BMP_1_GRASS_2.png",
"assets/Rat/BMP_1_GRASS_3.png",
"assets/Rat/BMP_1_GRASS_4.png",
"assets/Rat/BMP_1_N.png",
"assets/Rat/BMP_1_NE.png",
"assets/Rat/BMP_1_NW.png",
"assets/Rat/BMP_1_S.png",
"assets/Rat/BMP_1_SE.png",
"assets/Rat/BMP_1_SW.png",
"assets/Rat/BMP_1_W.png",
"assets/Rat/BMP_1_WN.png",
"assets/Rat/BMP_1_WS.png",
"assets/Rat/BMP_2_CAVE_DOWN.png",
"assets/Rat/BMP_2_CAVE_LEFT.png",
"assets/Rat/BMP_2_CAVE_RIGHT.png",
"assets/Rat/BMP_2_CAVE_UP.png",
"assets/Rat/BMP_2_E.png",
"assets/Rat/BMP_2_EN.png",
"assets/Rat/BMP_2_ES.png",
"assets/Rat/BMP_2_EXPLOSION_DOWN.png",
"assets/Rat/BMP_2_EXPLOSION_LEFT.png",
"assets/Rat/BMP_2_EXPLOSION_RIGHT.png",
"assets/Rat/BMP_2_EXPLOSION_UP.png",
"assets/Rat/BMP_2_FLOWER_1.png",
"assets/Rat/BMP_2_FLOWER_2.png",
"assets/Rat/BMP_2_FLOWER_3.png",
"assets/Rat/BMP_2_FLOWER_4.png",
"assets/Rat/BMP_2_GAS_DOWN.png",
"assets/Rat/BMP_2_GAS_LEFT.png",
"assets/Rat/BMP_2_GAS_RIGHT.png",
"assets/Rat/BMP_2_GAS_UP.png",
"assets/Rat/BMP_2_GRASS_1.png",
"assets/Rat/BMP_2_GRASS_2.png",
"assets/Rat/BMP_2_GRASS_3.png",
"assets/Rat/BMP_2_GRASS_4.png",
"assets/Rat/BMP_2_N.png",
"assets/Rat/BMP_2_NE.png",
"assets/Rat/BMP_2_NW.png",
"assets/Rat/BMP_2_S.png",
"assets/Rat/BMP_2_SE.png",
"assets/Rat/BMP_2_SW.png",
"assets/Rat/BMP_2_W.png",
"assets/Rat/BMP_2_WN.png",
"assets/Rat/BMP_2_WS.png",
"assets/Rat/BMP_3_CAVE_DOWN.png",
"assets/Rat/BMP_3_CAVE_LEFT.png",
"assets/Rat/BMP_3_CAVE_RIGHT.png",
"assets/Rat/BMP_3_CAVE_UP.png",
"assets/Rat/BMP_3_E.png",
"assets/Rat/BMP_3_EN.png",
"assets/Rat/BMP_3_ES.png",
"assets/Rat/BMP_3_EXPLOSION_DOWN.png",
"assets/Rat/BMP_3_EXPLOSION_LEFT.png",
"assets/Rat/BMP_3_EXPLOSION_RIGHT.png",
"assets/Rat/BMP_3_EXPLOSION_UP.png",
"assets/Rat/BMP_3_FLOWER_1.png",
"assets/Rat/BMP_3_FLOWER_2.png",
"assets/Rat/BMP_3_FLOWER_3.png",
"assets/Rat/BMP_3_FLOWER_4.png",
"assets/Rat/BMP_3_GAS_DOWN.png",
"assets/Rat/BMP_3_GAS_LEFT.png",
"assets/Rat/BMP_3_GAS_RIGHT.png",
"assets/Rat/BMP_3_GAS_UP.png",
"assets/Rat/BMP_3_GRASS_1.png",
"assets/Rat/BMP_3_GRASS_2.png",
"assets/Rat/BMP_3_GRASS_3.png",
"assets/Rat/BMP_3_GRASS_4.png",
"assets/Rat/BMP_3_N.png",
"assets/Rat/BMP_3_NE.png",
"assets/Rat/BMP_3_NW.png",
"assets/Rat/BMP_3_S.png",
"assets/Rat/BMP_3_SE.png",
"assets/Rat/BMP_3_SW.png",
"assets/Rat/BMP_3_W.png",
"assets/Rat/BMP_3_WN.png",
"assets/Rat/BMP_3_WS.png",
"assets/Rat/BMP_4_CAVE_DOWN.png",
"assets/Rat/BMP_4_CAVE_LEFT.png",
"assets/Rat/BMP_4_CAVE_RIGHT.png",
"assets/Rat/BMP_4_CAVE_UP.png",
"assets/Rat/BMP_4_E.png",
"assets/Rat/BMP_4_EN.png",
"assets/Rat/BMP_4_ES.png",
"assets/Rat/BMP_4_EXPLOSION_DOWN.png",
"assets/Rat/BMP_4_EXPLOSION_LEFT.png",
"assets/Rat/BMP_4_EXPLOSION_RIGHT.png",
"assets/Rat/BMP_4_EXPLOSION_UP.png",
"assets/Rat/BMP_4_FLOWER_1.png",
"assets/Rat/BMP_4_FLOWER_2.png",
"assets/Rat/BMP_4_FLOWER_3.png",
"assets/Rat/BMP_4_FLOWER_4.png",
"assets/Rat/BMP_4_GAS_DOWN.png",
"assets/Rat/BMP_4_GAS_LEFT.png",
"assets/Rat/BMP_4_GAS_RIGHT.png",
"assets/Rat/BMP_4_GAS_UP.png",
"assets/Rat/BMP_4_GRASS_1.png",
"assets/Rat/BMP_4_GRASS_2.png",
"assets/Rat/BMP_4_GRASS_3.png",
"assets/Rat/BMP_4_GRASS_4.png",
"assets/Rat/BMP_4_N.png",
"assets/Rat/BMP_4_NE.png",
"assets/Rat/BMP_4_NW.png",
"assets/Rat/BMP_4_S.png",
"assets/Rat/BMP_4_SE.png",
"assets/Rat/BMP_4_SW.png",
"assets/Rat/BMP_4_W.png",
"assets/Rat/BMP_4_WN.png",
"assets/Rat/BMP_4_WS.png",
"assets/Rat/BMP_ARROW_DOWN.png",
"assets/Rat/BMP_ARROW_LEFT.png",
"assets/Rat/BMP_ARROW_RIGHT.png",
"assets/Rat/BMP_ARROW_UP.png",
"assets/Rat/BMP_BABY_DOWN.png",
"assets/Rat/BMP_BABY_LEFT.png",
"assets/Rat/BMP_BABY_RIGHT.png",
"assets/Rat/BMP_BABY_UP.png",
"assets/Rat/BMP_BLOCK_0.png",
"assets/Rat/BMP_BLOCK_1.png",
"assets/Rat/BMP_BLOCK_2.png",
"assets/Rat/BMP_BLOCK_3.png",
"assets/Rat/BMP_BOMB0.png",
"assets/Rat/BMP_BOMB1.png",
"assets/Rat/BMP_BOMB2.png",
"assets/Rat/BMP_BOMB3.png",
"assets/Rat/BMP_BOMB4.png",
"assets/Rat/BMP_BONUS_10.png",
"assets/Rat/BMP_BONUS_160.png",
"assets/Rat/BMP_BONUS_20.png",
"assets/Rat/BMP_BONUS_40.png",
"assets/Rat/BMP_BONUS_5.png",
"assets/Rat/BMP_BONUS_80.png",
"assets/Rat/BMP_EXPLOSION.png",
"assets/Rat/BMP_EXPLOSION_DOWN.png",
"assets/Rat/BMP_EXPLOSION_LEFT.png",
"assets/Rat/BMP_EXPLOSION_RIGHT.png",
"assets/Rat/BMP_EXPLOSION_UP.png",
"assets/Rat/BMP_FEMALE.png",
"assets/Rat/BMP_FEMALE_DOWN.png",
"assets/Rat/BMP_FEMALE_LEFT.png",
"assets/Rat/BMP_FEMALE_RIGHT.png",
"assets/Rat/BMP_FEMALE_UP.png",
"assets/Rat/BMP_GAS.png",
"assets/Rat/BMP_GAS_DOWN.png",
"assets/Rat/BMP_GAS_LEFT.png",
"assets/Rat/BMP_GAS_RIGHT.png",
"assets/Rat/BMP_GAS_UP.png",
"assets/Rat/BMP_MALE.png",
"assets/Rat/BMP_MALE_DOWN.png",
"assets/Rat/BMP_MALE_LEFT.png",
"assets/Rat/BMP_MALE_RIGHT.png",
"assets/Rat/BMP_MALE_UP.png",
"assets/Rat/BMP_NUCLEAR.png",
"assets/Rat/BMP_POISON.png",
"assets/Rat/BMP_START_1.png",
"assets/Rat/BMP_START_1_DOWN.png",
"assets/Rat/BMP_START_1_SHADED.png",
"assets/Rat/BMP_START_2.png",
"assets/Rat/BMP_START_2_DOWN.png",
"assets/Rat/BMP_START_2_SHADED.png",
"assets/Rat/BMP_START_3.png",
"assets/Rat/BMP_START_3_DOWN.png",
"assets/Rat/BMP_START_3_SHADED.png",
"assets/Rat/BMP_START_4.png",
"assets/Rat/BMP_START_4_DOWN.png",
"assets/Rat/BMP_START_4_SHADED.png",
"assets/Rat/BMP_TITLE.png",
"assets/Rat/BMP_TUNNEL.png",
"assets/Rat/BMP_VERMINATORS.png",
"assets/Rat/BMP_WEWIN.png",
"assets/Rat/mine.png",
"assets/decterm.ttf",
"assets/AmaticSC-Regular.ttf",
"assets/terminal.ttf"
]

21
assets/sound-manifest.json

@ -0,0 +1,21 @@
[
"sound/BIRTH.WAV",
"sound/BOMB.WAV",
"sound/CHOKE.WAV",
"sound/CLUNK.WAV",
"sound/Death.wav",
"sound/GAS.WAV",
"sound/NEWSEX.WAV",
"sound/NUCLEAR.WAV",
"sound/POISON.WAV",
"sound/PUTDOWN.WAV",
"sound/SEX.WAV",
"sound/VICTORY.WAV",
"sound/WELLDONE.WAV",
"sound/WEWIN.WAV",
"sound/converted_BOMB.wav",
"sound/mine.wav",
"sound/mine_converted.wav",
"sound/mine_original.wav",
"sound/nuke.wav"
]

33
conf/keybindings.json

@ -1,33 +0,0 @@
{
"keybinding_game": {
"keydown_Return": "spawn_rat",
"keydown_D": "kill_rat",
"keydown_M": "toggle_audio",
"keydown_F": "toggle_full_screen",
"keydown_Up": "start_scrolling|Up",
"keydown_Down": "start_scrolling|Down",
"keydown_Left": "start_scrolling|Left",
"keydown_Right": "start_scrolling|Right",
"keyup_Up": "stop_scrolling",
"keyup_Down": "stop_scrolling",
"keyup_Left": "stop_scrolling",
"keyup_Right": "stop_scrolling",
"keydown_Space": "spawn_new_bomb",
"keydown_N": "spawn_new_nuclear_bomb",
"keydown_Left_Ctrl": "spawn_new_mine",
"keydown_G": "spawn_gas",
"keydown_P": "toggle_pause"
},
"keybinding_start_menu": {
"keydown_Return": "reset_game",
"keydown_Escape": "quit_game",
"keydown_M": "toggle_audio",
"keydown_F": "toggle_full_screen"
},
"keybinding_paused": {
"keydown_Return": "reset_game",
"keydown_Escape": "quit_game",
"keydown_M": "toggle_audio",
"keydown_F": "toggle_full_screen"
}
}

29
conf/keybindings_pc.yaml

@ -1,29 +0,0 @@
keybinding_game:
keydown_Return: spawn_rat
keydown_D: kill_rat
keydown_M: toggle_audio
keydown_F: toggle_full_screen
keydown_Up: start_scrolling|Up
keydown_Down: start_scrolling|Down
keydown_Left: start_scrolling|Left
keydown_Right: start_scrolling|Right
keyup_Up: stop_scrolling
keyup_Down: stop_scrolling
keyup_Left: stop_scrolling
keyup_Right: stop_scrolling
keydown_Space: spawn_new_bomb
keydown_N: spawn_new_nuclear_bomb
keydown_Left_Ctrl: spawn_new_mine
keydown_P: toggle_pause
keybinding_start_menu:
keydown_Return: reset_game
keydown_Escape: quit_game
keydown_M: toggle_audio
keydown_F: toggle_full_screen
keybinding_paused:
keydown_Return: reset_game
keydown_Escape: quit_game
keydown_M: toggle_audio
keydown_F: toggle_full_screen

31
engine/controls.py

@ -17,24 +17,38 @@ else:
class KeyBindings:
def trigger(self, action):
#print(f"Triggering action: {action}")
import os
debug = os.environ.get('DEBUG_KEYS', '').lower() == 'true'
if debug:
print(f"[KEY] Triggering action: {action} (status: {self.game_status})")
# Check if the action is in the bindings
if action in bindings[f"keybinding_{self.game_status}"]:
value = bindings[f"keybinding_{self.game_status}"][action]
current_bindings = bindings.get(f"keybinding_{self.game_status}", {})
if action in current_bindings:
value = current_bindings[action]
if debug:
print(f"[KEY] Found binding: {action} -> {value}")
# Call the corresponding method
if value:
#print(f"Calling method: {value}")
if "|" in value:
method_name, *args = value.split("|")
method = getattr(self, method_name)
if debug:
print(f"[KEY] Calling method: {method_name}({args})")
method(*args)
else:
if debug:
print(f"[KEY] Calling method: {value}()")
getattr(self, value)()
#else:
#print(f"Action {action} not found in keybindings for {self.game_status}")
return
#print(f"Action {action} not found in keybindings for {self.game_status}")
if debug:
print(f"[KEY] Action '{action}' not found in {self.game_status} bindings")
print(f"[KEY] Available actions: {list(current_bindings.keys())}")
return None
def spawn_new_bomb(self):
@ -46,6 +60,9 @@ class KeyBindings:
def spawn_new_nuclear_bomb(self):
self.spawn_nuclear_bomb(self.pointer)
def spawn_new_gas(self):
self.spawn_gas(self.pointer)
def toggle_audio(self):
self.render_engine.audio = not self.render_engine.audio
def toggle_pause(self):

1
engine/graphics.py

@ -4,6 +4,7 @@ class Graphics():
def load_assets(self):
print("Loading graphics assets...")
self.tunnel = self.render_engine.load_image("Rat/BMP_TUNNEL.png", surface=True)
print("Loading grass variants...")
self.grasses = [self.render_engine.load_image(f"Rat/BMP_1_GRASS_{i+1}.png", surface=True) for i in range(4)]
self.rat_assets = {}
self.rat_assets_textures = {}

833
engine/pygame_layer.py

@ -0,0 +1,833 @@
import os
import random
import pygame
from pygame import mixer
class GameWindow:
"""
Pygame-based game window implementation.
Provides a complete interface equivalent to sdl2_layer.GameWindow
"""
def __init__(self, width, height, cell_size, title="Default", key_callback=None):
# Display configuration
self.cell_size = cell_size
self.width = width * cell_size
self.height = height * cell_size
# Screen resolution handling
actual_screen_size = os.environ.get("RESOLUTION", "640x480").split("x")
actual_screen_size = tuple(map(int, actual_screen_size))
self.target_size = actual_screen_size if self.width > actual_screen_size[0] or self.height > actual_screen_size[1] else (self.width, self.height)
# View offset calculations
self.w_start_offset = (self.target_size[0] - self.width) // 2
self.h_start_offset = (self.target_size[1] - self.height) // 2
self.w_offset = self.w_start_offset
self.h_offset = self.h_start_offset
self.max_w_offset = self.target_size[0] - self.width
self.max_h_offset = self.target_size[1] - self.height
self.scale = self.target_size[1] // self.cell_size
print(f"Screen size: {self.width}x{self.height}")
# Pygame initialization
pygame.init()
mixer.init(frequency=22050, size=-16, channels=1, buffer=2048)
# Window and screen setup
self.window = pygame.display.set_mode(self.target_size)
pygame.display.set_caption(title)
self.screen = self.window
# Font system
self.fonts = self.generate_fonts("assets/decterm.ttf")
# Game state
self.running = True
self.delay = 30
self.performance = 0
self.last_status_text = ""
self.stats_sprite = None
self.mean_fps = 0
self.fpss = []
self.text_width = 0
self.text_height = 0
self.ammo_text = ""
self.stats_background = None
self.ammo_background = None
self.ammo_sprite = None
# White flash effect state
self.white_flash_active = False
self.white_flash_start_time = 0
self.white_flash_opacity = 255
# Input handling
self.trigger = key_callback
self.button_cursor = [0, 0]
self.buttons = {}
# Audio system initialization
self._init_audio_system()
self.audio = True
# Clock for frame rate control
self.clock = pygame.time.Clock()
# Input devices
self.load_joystick()
def show(self):
"""Show the window (for compatibility with SDL2 interface)"""
pygame.display.set_mode(self.target_size)
def _init_audio_system(self):
"""Initialize audio channels for different audio types"""
mixer.set_num_channels(8) # Ensure enough channels
self.audio_channels = {
"base": mixer.Channel(0),
"effects": mixer.Channel(1),
"music": mixer.Channel(2)
}
self.current_sounds = {}
# ======================
# TEXTURE & IMAGE METHODS
# ======================
def create_texture(self, tiles: list):
"""Create a texture from a list of tiles"""
bg_surface = pygame.Surface((self.width, self.height))
for tile in tiles:
bg_surface.blit(tile[0], (tile[1], tile[2]))
return bg_surface
# Helpers to support incremental background generation
def create_empty_background_surface(self):
"""Create and return an empty background surface to incrementally blit onto."""
return pygame.Surface((self.width, self.height))
def blit_tiles_batch(self, bg_surface, tiles_batch: list):
"""Blit a small batch of tiles onto the provided background surface.
tiles_batch: list of (surface, x, y)
Returns None. Designed to be called repeatedly with small batches to avoid long blocking operations.
"""
for tile, x, y in tiles_batch:
try:
bg_surface.blit(tile, (x, y))
except Exception:
# If tile is a SpriteWrapper, extract surface
try:
bg_surface.blit(tile.surface, (x, y))
except Exception:
pass
def load_image(self, path, transparent_color=None, surface=False):
"""Load and process an image with optional transparency and scaling"""
image_path = os.path.join("assets", path)
# First try to use pygame's native loader which avoids a Pillow dependency.
try:
py_image = pygame.image.load(image_path)
# Ensure alpha if needed
try:
py_image = py_image.convert_alpha()
except Exception:
try:
py_image = py_image.convert()
except Exception:
pass
# Handle transparent color via colorkey if provided
if transparent_color:
# pygame expects a tuple of ints
try:
py_image.set_colorkey(transparent_color)
except Exception:
pass
# Scale image using pygame transforms
scale = max(1, self.cell_size // 20)
new_size = (py_image.get_width() * scale, py_image.get_height() * scale)
try:
py_image = pygame.transform.scale(py_image, new_size)
except Exception:
# If scaling fails, continue with original
pass
if not surface:
return SpriteWrapper(py_image)
return py_image
except Exception:
# Fallback to PIL-based loading if pygame can't handle the file or Pillow is present
try:
# Import Pillow lazily to avoid hard dependency at module import time
try:
from PIL import Image
except Exception:
Image = None
if Image is None:
raise
image = Image.open(image_path)
# Handle transparency
if transparent_color:
image = image.convert("RGBA")
datas = image.getdata()
new_data = []
for item in datas:
if item[:3] == transparent_color:
new_data.append((255, 255, 255, 0))
else:
new_data.append(item)
image.putdata(new_data)
# Scale image
scale = max(1, self.cell_size // 20)
image = image.resize((image.width * scale, image.height * scale), Image.NEAREST)
# Convert PIL image to pygame surface
mode = image.mode
size = image.size
data = image.tobytes()
if mode == "RGBA":
py_image = pygame.image.fromstring(data, size, mode)
elif mode == "RGB":
py_image = pygame.image.fromstring(data, size, mode)
else:
image = image.convert("RGBA")
data = image.tobytes()
py_image = pygame.image.fromstring(data, size, "RGBA")
if not surface:
return SpriteWrapper(py_image)
return py_image
except Exception:
# If both loaders fail, raise to notify caller
raise
def get_image_size(self, image):
"""Get the size of an image sprite"""
if isinstance(image, SpriteWrapper):
return image.size
return image.get_size()
# ======================
# FONT MANAGEMENT
# ======================
def generate_fonts(self, font_file):
"""Generate font objects for different sizes"""
fonts = {}
for i in range(10, 70, 1):
try:
fonts[i] = pygame.font.Font(font_file, i)
except:
fonts[i] = pygame.font.Font(None, i)
return fonts
# ======================
# DRAWING METHODS
# ======================
def draw_text(self, text, font, position, color):
"""Draw text at specified position with given font and color"""
if isinstance(color, tuple):
# Pygame color format
pass
else:
# Convert from any other format to RGB tuple
color = (color.r, color.g, color.b) if hasattr(color, 'r') else (0, 0, 0)
text_surface = font.render(text, True, color)
text_rect = text_surface.get_rect()
# Handle center positioning
if position == "center":
position = ("center", "center")
if isinstance(position, tuple):
if position[0] == "center":
text_rect.centerx = self.target_size[0] // 2
text_rect.y = position[1]
elif position[1] == "center":
text_rect.x = position[0]
text_rect.centery = self.target_size[1] // 2
else:
text_rect.topleft = position
self.screen.blit(text_surface, text_rect)
def draw_background(self, bg_texture):
"""Draw background texture with current view offset"""
self.screen.blit(bg_texture, (self.w_offset, self.h_offset))
def draw_image(self, x, y, sprite, tag=None, anchor="nw"):
"""Draw an image sprite at specified coordinates"""
if not self.is_in_visible_area(x, y):
return
if isinstance(sprite, SpriteWrapper):
surface = sprite.surface
else:
surface = sprite
self.screen.blit(surface, (x + self.w_offset, y + self.h_offset))
def draw_rectangle(self, x, y, width, height, tag, outline="red", filling=None):
"""Draw a rectangle with optional fill and outline"""
rect = pygame.Rect(x, y, width, height)
if filling:
pygame.draw.rect(self.screen, filling, rect)
else:
# Handle outline color
if isinstance(outline, str):
color_map = {
"red": (255, 0, 0),
"blue": (0, 0, 255),
"green": (0, 255, 0),
"black": (0, 0, 0),
"white": (255, 255, 255)
}
outline = color_map.get(outline, (255, 0, 0))
pygame.draw.rect(self.screen, outline, rect, 2)
def draw_pointer(self, x, y):
"""Draw a red pointer rectangle at specified coordinates"""
x = x + self.w_offset
y = y + self.h_offset
for i in range(3):
rect = pygame.Rect(x + i, y + i, self.cell_size - 2*i, self.cell_size - 2*i)
pygame.draw.rect(self.screen, (255, 0, 0), rect, 1)
def delete_tag(self, tag):
"""Placeholder for tag deletion (not needed in pygame implementation)"""
pass
# ======================
# UI METHODS
# ======================
def dialog(self, text, **kwargs):
"""Display a dialog box with text and optional extras"""
# Draw dialog background
dialog_rect = pygame.Rect(50, 50, self.target_size[0] - 100, self.target_size[1] - 100)
pygame.draw.rect(self.screen, (255, 255, 255), dialog_rect)
# Calculate layout positions to avoid overlaps
title_y = self.target_size[1] // 4 # Title at 1/4 of screen height
# Draw main text (title)
self.draw_text(text, self.fonts[self.target_size[1]//20],
("center", title_y), (0, 0, 0))
# Draw image if provided - position it below title
image_bottom_y = title_y + 60 # Default position if no image
if image := kwargs.get("image"):
image_size = self.get_image_size(image)
image_y = title_y + 50
self.draw_image(self.target_size[0] // 2 - image_size[0] // 2 - self.w_offset,
image_y - self.h_offset,
image, "win")
image_bottom_y = image_y + image_size[1] + 20
# Draw subtitle if provided - handle multi-line text, position below image
if subtitle := kwargs.get("subtitle"):
subtitle_lines = subtitle.split('\n')
base_y = image_bottom_y + 20
line_height = 25 # Fixed line height for consistent spacing
for i, line in enumerate(subtitle_lines):
if line.strip(): # Only draw non-empty lines
self.draw_text(line.strip(), self.fonts[self.target_size[1]//35],
("center", base_y + i * line_height), (0, 0, 0))
# Draw scores if provided - position at bottom
if scores := kwargs.get("scores"):
scores_start_y = self.target_size[1] * 3 // 4 # Bottom quarter of screen
title_surface = self.fonts[self.target_size[1]//25].render("High Scores:", True, (0, 0, 0))
title_rect = title_surface.get_rect(center=(self.target_size[0] // 2, scores_start_y))
self.screen.blit(title_surface, title_rect)
for i, score in enumerate(scores[:5]):
if len(score) >= 4: # New format: date, score, name, device
score_text = f"{score[2]}: {score[1]} pts ({score[3]})"
elif len(score) >= 3: # Medium format: date, score, name
score_text = f"{score[2]}: {score[1]} pts"
else: # Old format: date, score
score_text = f"Guest: {score[1]} pts"
self.draw_text(score_text, self.fonts[self.target_size[1]//45],
("center", scores_start_y + 30 + 25 * (i + 1)),
(0, 0, 0))
def start_dialog(self, **kwargs):
"""Display the welcome dialog"""
self.dialog("Welcome to the Mice!", subtitle="A game by Matteo because was bored", **kwargs)
def draw_button(self, x, y, text, width, height, coords):
"""Draw a button with text"""
color = (0, 0, 255) if self.button_cursor == list(coords) else (0, 0, 0)
self.draw_rectangle(x, y, width, height, "button", outline=color)
def update_status(self, text):
"""Update and display the status bar with FPS information"""
fps = int(self.clock.get_fps()) if self.clock.get_fps() > 0 else 0
if len(self.fpss) > 20:
self.mean_fps = round(sum(self.fpss) / len(self.fpss)) if self.fpss else fps
self.fpss.clear()
else:
self.fpss.append(fps)
status_text = f"FPS: {self.mean_fps} - {text}"
if status_text != self.last_status_text:
self.last_status_text = status_text
font = self.fonts[20]
self.stats_sprite = font.render(status_text, True, (0, 0, 0))
if self.text_width != self.stats_sprite.get_width() or self.text_height != self.stats_sprite.get_height():
self.text_width, self.text_height = self.stats_sprite.get_size()
self.stats_background = pygame.Surface((self.text_width + 10, self.text_height + 4))
self.stats_background.fill((255, 255, 255))
self.screen.blit(self.stats_background, (3, 3))
self.screen.blit(self.stats_sprite, (8, 5))
def update_ammo(self, ammo, assets):
"""Update and display the ammo count"""
ammo_text = f"{ammo['bomb']['count']}/{ammo['bomb']['max']} {ammo['mine']['count']}/{ammo['mine']['max']} {ammo['gas']['count']}/{ammo['gas']['max']} "
if self.ammo_text != ammo_text:
self.ammo_text = ammo_text
font = self.fonts[20]
self.ammo_sprite = font.render(ammo_text, True, (0, 0, 0))
text_width, text_height = self.ammo_sprite.get_size()
self.ammo_background = pygame.Surface((text_width + 10, text_height + 4))
self.ammo_background.fill((255, 255, 255))
text_width, text_height = self.ammo_sprite.get_size()
position = (self.target_size[0] - text_width - 10, self.target_size[1] - text_height - 5)
self.screen.blit(self.ammo_background, (position[0] - 5, position[1] - 2))
self.screen.blit(self.ammo_sprite, position)
# Draw ammo icons
bomb_sprite = assets["BMP_BOMB0"]
poison_sprite = assets["BMP_POISON"]
gas_sprite = assets["BMP_GAS"]
if isinstance(bomb_sprite, SpriteWrapper):
self.screen.blit(bomb_sprite.surface, (position[0]+25, position[1]))
else:
# Scale to 20x20 if needed
bomb_scaled = pygame.transform.scale(bomb_sprite, (20, 20))
self.screen.blit(bomb_scaled, (position[0]+25, position[1]))
if isinstance(poison_sprite, SpriteWrapper):
self.screen.blit(poison_sprite.surface, (position[0]+85, position[1]))
else:
poison_scaled = pygame.transform.scale(poison_sprite, (20, 20))
self.screen.blit(poison_scaled, (position[0]+85, position[1]))
if isinstance(gas_sprite, SpriteWrapper):
self.screen.blit(gas_sprite.surface, (position[0]+140, position[1]))
else:
gas_scaled = pygame.transform.scale(gas_sprite, (20, 20))
self.screen.blit(gas_scaled, (position[0]+140, position[1]))
# ======================
# VIEW & NAVIGATION
# ======================
def scroll_view(self, pointer):
"""Adjust the view offset based on pointer coordinates"""
x, y = pointer
# Scale down and invert coordinates
x = -(x // 2) * self.cell_size
y = -(y // 2) * self.cell_size
# Clamp horizontal offset to valid range
if x <= self.max_w_offset + self.cell_size:
x = self.max_w_offset
# Clamp vertical offset to valid range
if y < self.max_h_offset:
y = self.max_h_offset
self.w_offset = x
self.h_offset = y
def is_in_visible_area(self, x, y):
"""Check if coordinates are within the visible area"""
return (-self.w_offset - self.cell_size <= x <= self.width - self.w_offset and
-self.h_offset - self.cell_size <= y <= self.height - self.h_offset)
def get_view_center(self):
"""Get the center coordinates of the current view"""
return self.w_offset + self.width // 2, self.h_offset + self.height // 2
# ======================
# AUDIO METHODS
# ======================
def play_sound(self, sound_file, tag="base"):
"""Play a sound file on the specified audio channel"""
if not self.audio:
return
try:
sound_path = os.path.join("sound", sound_file)
sound = mixer.Sound(sound_path)
# Get the appropriate channel
channel = self.audio_channels.get(tag, self.audio_channels["base"])
# Stop any currently playing sound on this channel
channel.stop()
# Play the new sound
channel.play(sound)
# Store reference to prevent garbage collection
self.current_sounds[tag] = sound
except Exception as e:
print(f"Error playing sound {sound_file}: {e}")
def stop_sound(self):
"""Stop all audio playback"""
for channel in self.audio_channels.values():
channel.stop()
# ======================
# INPUT METHODS
# ======================
def load_joystick(self):
"""Initialize joystick support"""
pygame.joystick.init()
joystick_count = pygame.joystick.get_count()
if joystick_count > 0:
self.joystick = pygame.joystick.Joystick(0)
self.joystick.init()
print(f"Joystick initialized: {self.joystick.get_name()}")
else:
self.joystick = None
# ======================
# MAIN GAME LOOP
# ======================
def _normalize_key_name(self, key):
"""Normalize pygame key names to match SDL2 key names"""
# Pygame returns lowercase, SDL2 returns with proper case
key_map = {
"return": "Return",
"escape": "Escape",
"space": "Space",
"tab": "Tab",
"left shift": "Left_Shift",
"right shift": "Right_Shift",
"left ctrl": "Left_Ctrl",
"right ctrl": "Right_Ctrl",
"left alt": "Left_Alt",
"right alt": "Right_Alt",
"up": "Up",
"down": "Down",
"left": "Left",
"right": "Right",
"delete": "Delete",
"backspace": "Backspace",
"insert": "Insert",
"home": "Home",
"end": "End",
"pageup": "Page_Up",
"pagedown": "Page_Down",
"f1": "F1",
"f2": "F2",
"f3": "F3",
"f4": "F4",
"f5": "F5",
"f6": "F6",
"f7": "F7",
"f8": "F8",
"f9": "F9",
"f10": "F10",
"f11": "F11",
"f12": "F12",
}
# Return mapped value or capitalize first letter of original
normalized = key_map.get(key.lower(), key)
# Handle single letters (make uppercase)
if len(normalized) == 1:
normalized = normalized.upper()
return normalized
def mainloop(self, **kwargs):
"""Main game loop handling events and rendering"""
while self.running:
performance_start = pygame.time.get_ticks()
# Clear screen
self.screen.fill((0, 0, 0))
# Execute background update if provided
if "bg_update" in kwargs:
kwargs["bg_update"]()
# Execute main update
kwargs["update"]()
# Update and draw white flash effect
if self.update_white_flash():
self.draw_white_flash()
# Handle Pygame events
for event in pygame.event.get():
if event.type == pygame.QUIT:
self.running = False
elif event.type == pygame.KEYDOWN:
key = pygame.key.name(event.key)
key = self._normalize_key_name(key)
key = key.replace(" ", "_")
self.trigger(f"keydown_{key}")
elif event.type == pygame.KEYUP:
key = pygame.key.name(event.key)
key = self._normalize_key_name(key)
key = key.replace(" ", "_")
self.trigger(f"keyup_{key}")
elif event.type == pygame.MOUSEMOTION:
self.trigger(f"mousemove_{event.pos[0]}, {event.pos[1]}")
elif event.type == pygame.JOYBUTTONDOWN:
self.trigger(f"joybuttondown_{event.button}")
elif event.type == pygame.JOYBUTTONUP:
self.trigger(f"joybuttonup_{event.button}")
elif event.type == pygame.JOYHATMOTION:
self.trigger(f"joyhatmotion_{event.hat}_{event.value}")
# Update display
pygame.display.flip()
# Control frame rate
self.clock.tick(60) # Target 60 FPS
# Calculate performance
self.performance = pygame.time.get_ticks() - performance_start
def step(self, update=None, bg_update=None):
"""Execute a single frame iteration. This is non-blocking and useful when
the caller (JS) schedules frames via requestAnimationFrame in the browser.
"""
performance_start = pygame.time.get_ticks()
# Clear screen
self.screen.fill((0, 0, 0))
# Background update
if bg_update:
try:
bg_update()
except Exception:
pass
# Main update
if update:
try:
update()
except Exception:
pass
# Update and draw white flash effect
if self.update_white_flash():
self.draw_white_flash()
# Handle Pygame events (single-frame processing)
for event in pygame.event.get():
if event.type == pygame.QUIT:
self.running = False
elif event.type == pygame.KEYDOWN:
key = pygame.key.name(event.key)
key = self._normalize_key_name(key)
key = key.replace(" ", "_")
self.trigger(f"keydown_{key}")
elif event.type == pygame.KEYUP:
key = pygame.key.name(event.key)
key = self._normalize_key_name(key)
key = key.replace(" ", "_")
self.trigger(f"keyup_{key}")
elif event.type == pygame.MOUSEMOTION:
self.trigger(f"mousemove_{event.pos[0]}, {event.pos[1]}")
elif event.type == pygame.JOYBUTTONDOWN:
self.trigger(f"joybuttondown_{event.button}")
elif event.type == pygame.JOYBUTTONUP:
self.trigger(f"joybuttonup_{event.button}")
elif event.type == pygame.JOYHATMOTION:
self.trigger(f"joyhatmotion_{event.hat}_{event.value}")
# Update display once per frame
pygame.display.flip()
# Control frame rate
self.clock.tick(60)
# Calculate performance
self.performance = pygame.time.get_ticks() - performance_start
# ======================
# SPECIAL EFFECTS
# ======================
def trigger_white_flash(self):
"""Trigger the white flash effect"""
self.white_flash_active = True
self.white_flash_start_time = pygame.time.get_ticks()
self.white_flash_opacity = 255
def update_white_flash(self):
"""Update the white flash effect and return True if it should be drawn"""
if not self.white_flash_active:
return False
current_time = pygame.time.get_ticks()
elapsed_time = current_time - self.white_flash_start_time
if elapsed_time < 500: # First 500ms: full white
self.white_flash_opacity = 255
return True
elif elapsed_time < 2000: # Next 1500ms: fade out
fade_progress = (elapsed_time - 500) / 1500.0
self.white_flash_opacity = int(255 * (1.0 - fade_progress))
return True
else: # Effect is complete
self.white_flash_active = False
self.white_flash_opacity = 0
return False
def draw_white_flash(self):
"""Draw the white flash overlay"""
if self.white_flash_opacity > 0:
white_surface = pygame.Surface(self.target_size)
white_surface.fill((255, 255, 255))
white_surface.set_alpha(self.white_flash_opacity)
self.screen.blit(white_surface, (0, 0))
# ======================
# UTILITY METHODS
# ======================
def new_cycle(self, delay, callback):
"""Placeholder for cycle management (not needed in pygame implementation)"""
pass
def full_screen(self, flag):
"""Toggle fullscreen mode"""
if flag:
self.window = pygame.display.set_mode(self.target_size, pygame.FULLSCREEN)
else:
self.window = pygame.display.set_mode(self.target_size)
self.screen = self.window
def get_perf_counter(self):
"""Get performance counter for timing"""
return pygame.time.get_ticks()
def close(self):
"""Close the game window and cleanup"""
self.running = False
pygame.quit()
# ======================
# BLOOD EFFECT METHODS
# ======================
def generate_blood_surface(self):
"""Generate a dynamic blood splatter surface using Pygame"""
size = self.cell_size
# Create RGBA surface for blood splatter
blood_surface = pygame.Surface((size, size), pygame.SRCALPHA)
# Blood color variations
blood_colors = [
(139, 0, 0, 255), # Dark red
(34, 34, 34, 255), # Very dark gray
(20, 60, 60, 255), # Dark teal
(255, 0, 0, 255), # Pure red
(128, 0, 0, 255), # Reddish brown
]
# Generate splatter with diffusion algorithm
center_x, center_y = size // 2, size // 2
max_radius = size // 3 + random.randint(-3, 5)
for y in range(size):
for x in range(size):
# Calculate distance from center
distance = ((x - center_x) ** 2 + (y - center_y) ** 2) ** 0.5
# Calculate blood probability based on distance
if distance <= max_radius:
probability = max(0, 1 - (distance / max_radius))
noise = random.random() * 0.7
if random.random() < probability * noise:
color = random.choice(blood_colors)
alpha = int(255 * probability * random.uniform(0.6, 1.0))
blood_surface.set_at((x, y), (*color[:3], alpha))
# Add scattered droplets around main splatter
for _ in range(random.randint(3, 8)):
drop_x = center_x + random.randint(-max_radius - 5, max_radius + 5)
drop_y = center_y + random.randint(-max_radius - 5, max_radius + 5)
if 0 <= drop_x < size and 0 <= drop_y < size:
drop_size = random.randint(1, 3)
for dy in range(-drop_size, drop_size + 1):
for dx in range(-drop_size, drop_size + 1):
nx, ny = drop_x + dx, drop_y + dy
if 0 <= nx < size and 0 <= ny < size:
if random.random() < 0.6:
color = random.choice(blood_colors[:3])
alpha = random.randint(100, 200)
blood_surface.set_at((nx, ny), (*color[:3], alpha))
return blood_surface
def draw_blood_surface(self, blood_surface, position):
"""Convert blood surface to texture and return it"""
# In pygame, we can return the surface directly
return blood_surface
def combine_blood_surfaces(self, existing_surface, new_surface):
"""Combine two blood surfaces by blending them together"""
combined_surface = pygame.Surface((self.cell_size, self.cell_size), pygame.SRCALPHA)
# Blit existing blood first
combined_surface.blit(existing_surface, (0, 0))
# Blit new blood on top with alpha blending
combined_surface.blit(new_surface, (0, 0))
return combined_surface
def free_surface(self, surface):
"""Safely free a pygame surface (not needed in pygame, handled by GC)"""
pass
class SpriteWrapper:
"""
Wrapper class to make pygame surfaces compatible with SDL2 sprite interface
"""
def __init__(self, surface):
self.surface = surface
self.size = surface.get_size()
self.position = (0, 0)
def get_size(self):
return self.size

85
engine/score_api_client.py

@ -4,11 +4,20 @@ Score API Client for Mice Game
Client module to integrate with the FastAPI score server
"""
import requests
import json
from typing import Optional, List, Dict, Any
import time
# Try to import requests; if unavailable, provide a minimal urllib-based fallback.
try:
import requests # type: ignore
_HAS_REQUESTS = True
except Exception:
requests = None
_HAS_REQUESTS = False
from typing import Optional, List, Dict, Any
import time
class ScoreAPIClient:
"""Client for communicating with the Mice Game Score API"""
@ -39,25 +48,65 @@ class ScoreAPIClient:
url = f"{self.api_base_url}{endpoint}"
try:
if method.upper() == "GET":
response = requests.get(url, timeout=self.timeout)
elif method.upper() == "POST":
response = requests.post(url, json=data, timeout=self.timeout)
else:
raise ValueError(f"Unsupported HTTP method: {method}")
if response.status_code == 200:
return response.json()
elif response.status_code in [400, 404, 409]:
# Client errors - return the error details
return {"error": True, "status": response.status_code, "detail": response.json()}
if _HAS_REQUESTS:
if method.upper() == "GET":
response = requests.get(url, timeout=self.timeout)
elif method.upper() == "POST":
response = requests.post(url, json=data, timeout=self.timeout)
else:
raise ValueError(f"Unsupported HTTP method: {method}")
if response.status_code == 200:
return response.json()
elif response.status_code in [400, 404, 409]:
return {"error": True, "status": response.status_code, "detail": response.json()}
else:
return {"error": True, "status": response.status_code, "detail": "Server error"}
else:
return {"error": True, "status": response.status_code, "detail": "Server error"}
# urllib fallback for environments without requests (e.g., Pyodide without wheel)
from urllib.request import Request, urlopen
from urllib.error import URLError, HTTPError
import urllib.parse
if method.upper() == 'GET':
req = Request(url, method='GET')
try:
with urlopen(req, timeout=self.timeout) as resp:
body = resp.read()
try:
return json.loads(body.decode('utf-8'))
except Exception:
return None
except HTTPError as he:
try:
detail = json.loads(he.read().decode('utf-8'))
except Exception:
detail = str(he)
return {"error": True, "status": he.code, "detail": detail}
except URLError:
return {"error": True, "detail": "Could not connect to score server"}
elif method.upper() == 'POST':
data_bytes = json.dumps(data).encode('utf-8') if data is not None else None
req = Request(url, data=data_bytes, method='POST')
req.add_header('Content-Type', 'application/json')
try:
with urlopen(req, timeout=self.timeout) as resp:
body = resp.read()
try:
return json.loads(body.decode('utf-8'))
except Exception:
return None
except HTTPError as he:
try:
detail = json.loads(he.read().decode('utf-8'))
except Exception:
detail = str(he)
return {"error": True, "status": he.code, "detail": detail}
except URLError:
return {"error": True, "detail": "Could not connect to score server"}
else:
raise ValueError(f"Unsupported HTTP method: {method}")
except requests.exceptions.ConnectionError:
return {"error": True, "detail": "Could not connect to score server"}
except requests.exceptions.Timeout:
return {"error": True, "detail": "Request timeout"}
except Exception as e:
return {"error": True, "detail": str(e)}

180
engine/sdl2.py → engine/sdl2_layer.py

@ -3,10 +3,10 @@ import random
import ctypes
from ctypes import *
import sdl2
import engine.sdl2_layer as sdl2_layer
import sdl2.ext
from sdl2.ext.compat import byteify
from sdl2 import SDL_AudioSpec
from engine.sdl2_layer import SDL_AudioSpec
from PIL import Image
@ -34,14 +34,14 @@ class GameWindow:
print(f"Screen size: {self.width}x{self.height}")
# SDL2 initialization
sdl2.ext.init(joystick=True)
sdl2.SDL_Init(sdl2.SDL_INIT_AUDIO)
sdl2_layer.ext.init(joystick=True)
sdl2_layer.SDL_Init(sdl2_layer.SDL_INIT_AUDIO)
# Window and renderer setup
self.window = sdl2.ext.Window(title=title, size=self.target_size)
self.window = sdl2_layer.ext.Window(title=title, size=self.target_size)
# self.window.show()
self.renderer = sdl2.ext.Renderer(self.window, flags=sdl2.SDL_RENDERER_ACCELERATED)
self.factory = sdl2.ext.SpriteFactory(renderer=self.renderer)
self.renderer = sdl2_layer.ext.Renderer(self.window, flags=sdl2_layer.SDL_RENDERER_ACCELERATED)
self.factory = sdl2_layer.ext.SpriteFactory(renderer=self.renderer)
# Font system
self.fonts = self.generate_fonts("assets/decterm.ttf")
@ -80,11 +80,11 @@ class GameWindow:
def _init_audio_system(self):
"""Initialize audio devices for different audio channels"""
audio_spec = SDL_AudioSpec(freq=22050, aformat=sdl2.AUDIO_U8, channels=1, samples=2048)
audio_spec = SDL_AudioSpec(freq=22050, aformat=sdl2_layer.AUDIO_U8, channels=1, samples=2048)
self.audio_devs = {}
self.audio_devs["base"] = sdl2.SDL_OpenAudioDevice(None, 0, audio_spec, None, 0)
self.audio_devs["effects"] = sdl2.SDL_OpenAudioDevice(None, 0, audio_spec, None, 0)
self.audio_devs["music"] = sdl2.SDL_OpenAudioDevice(None, 0, audio_spec, None, 0)
self.audio_devs["base"] = sdl2_layer.SDL_OpenAudioDevice(None, 0, audio_spec, None, 0)
self.audio_devs["effects"] = sdl2_layer.SDL_OpenAudioDevice(None, 0, audio_spec, None, 0)
self.audio_devs["music"] = sdl2_layer.SDL_OpenAudioDevice(None, 0, audio_spec, None, 0)
# ======================
# TEXTURE & IMAGE METHODS
@ -92,12 +92,12 @@ class GameWindow:
def create_texture(self, tiles: list):
"""Create a texture from a list of tiles"""
bg_surface = sdl2.SDL_CreateRGBSurface(0, self.width, self.height, 32, 0, 0, 0, 0)
bg_surface = sdl2_layer.SDL_CreateRGBSurface(0, self.width, self.height, 32, 0, 0, 0, 0)
for tile in tiles:
dstrect = sdl2.SDL_Rect(tile[1], tile[2], self.cell_size, self.cell_size)
sdl2.SDL_BlitSurface(tile[0], None, bg_surface, dstrect)
dstrect = sdl2_layer.SDL_Rect(tile[1], tile[2], self.cell_size, self.cell_size)
sdl2_layer.SDL_BlitSurface(tile[0], None, bg_surface, dstrect)
bg_texture = self.factory.from_surface(bg_surface)
sdl2.SDL_FreeSurface(bg_surface)
sdl2_layer.SDL_FreeSurface(bg_surface)
return bg_texture
def load_image(self, path, transparent_color=None, surface=False):
@ -122,8 +122,8 @@ class GameWindow:
image = image.resize((image.width * scale, image.height * scale), Image.NEAREST)
if surface:
return sdl2.ext.pillow_to_surface(image)
return self.factory.from_surface(sdl2.ext.pillow_to_surface(image))
return sdl2_layer.ext.pillow_to_surface(image)
return self.factory.from_surface(sdl2_layer.ext.pillow_to_surface(image))
def get_image_size(self, image):
"""Get the size of an image sprite"""
@ -137,7 +137,7 @@ class GameWindow:
"""Generate font managers for different sizes"""
fonts = {}
for i in range(10, 70, 1):
fonts.update({i: sdl2.ext.FontManager(font_path=font_file, size=i)})
fonts.update({i: sdl2_layer.ext.FontManager(font_path=font_file, size=i)})
return fonts
# ======================
@ -161,7 +161,7 @@ class GameWindow:
def draw_background(self, bg_texture):
"""Draw background texture with current view offset"""
self.renderer.copy(bg_texture, dstrect=sdl2.SDL_Rect(self.w_offset, self.h_offset, self.width, self.height))
self.renderer.copy(bg_texture, dstrect=sdl2_layer.SDL_Rect(self.w_offset, self.h_offset, self.width, self.height))
def draw_image(self, x, y, sprite, tag=None, anchor="nw"):
"""Draw an image sprite at specified coordinates"""
@ -173,9 +173,9 @@ class GameWindow:
def draw_rectangle(self, x, y, width, height, tag, outline="red", filling=None):
"""Draw a rectangle with optional fill and outline"""
if filling:
self.renderer.fill((x, y, width, height), sdl2.ext.Color(*filling))
self.renderer.fill((x, y, width, height), sdl2_layer.ext.Color(*filling))
else:
self.renderer.draw_rect((x, y, width, height), sdl2.ext.Color(*outline))
self.renderer.draw_rect((x, y, width, height), sdl2_layer.ext.Color(*outline))
def draw_pointer(self, x, y):
"""Draw a red pointer rectangle at specified coordinates"""
@ -183,7 +183,7 @@ class GameWindow:
y = y + self.h_offset
for i in range(3):
self.renderer.draw_rect((x + i, y + i, self.cell_size - 2*i, self.cell_size - 2*i),
color=sdl2.ext.Color(255, 0, 0))
color=sdl2_layer.ext.Color(255, 0, 0))
def delete_tag(self, tag):
"""Placeholder for tag deletion (not implemented)"""
@ -205,7 +205,7 @@ class GameWindow:
# Draw main text (title)
self.draw_text(text, self.fonts[self.target_size[1]//20],
("center", title_y), sdl2.ext.Color(0, 0, 0))
("center", title_y), sdl2_layer.ext.Color(0, 0, 0))
# Draw image if provided - position it below title
image_bottom_y = title_y + 60 # Default position if no image
@ -226,12 +226,12 @@ class GameWindow:
for i, line in enumerate(subtitle_lines):
if line.strip(): # Only draw non-empty lines
self.draw_text(line.strip(), self.fonts[self.target_size[1]//35],
("center", base_y + i * line_height), sdl2.ext.Color(0, 0, 0))
("center", base_y + i * line_height), sdl2_layer.ext.Color(0, 0, 0))
# Draw scores if provided - position at bottom
if scores := kwargs.get("scores"):
scores_start_y = self.target_size[1] * 3 // 4 # Bottom quarter of screen
sprite = self.factory.from_text("High Scores:", color=sdl2.ext.Color(0, 0, 0),
sprite = self.factory.from_text("High Scores:", color=sdl2_layer.ext.Color(0, 0, 0),
fontmanager=self.fonts[self.target_size[1]//25])
sprite.position = (self.target_size[0] // 2 - sprite.size[0] // 2, scores_start_y)
self.renderer.copy(sprite, dstrect=sprite.position)
@ -246,7 +246,7 @@ class GameWindow:
self.draw_text(score_text, self.fonts[self.target_size[1]//45],
("center", scores_start_y + 30 + 25 * (i + 1)),
sdl2.ext.Color(0, 0, 0))
sdl2_layer.ext.Color(0, 0, 0))
def start_dialog(self, **kwargs):
"""Display the welcome dialog"""
@ -274,15 +274,15 @@ class GameWindow:
if status_text != self.last_status_text:
self.last_status_text = status_text
font = self.fonts[20]
self.stats_sprite = self.factory.from_text(status_text, color=sdl2.ext.Color(0, 0, 0), fontmanager=font)
self.stats_sprite = self.factory.from_text(status_text, color=sdl2_layer.ext.Color(0, 0, 0), fontmanager=font)
if self.text_width != self.stats_sprite.size[0] or self.text_height != self.stats_sprite.size[1]:
self.text_width, self.text_height = self.stats_sprite.size
# create a background for the status text using texture
self.stats_background = self.factory.from_color(sdl2.ext.Color(255, 255, 255), (self.text_width + 10, self.text_height + 4))
self.stats_background = self.factory.from_color(sdl2_layer.ext.Color(255, 255, 255), (self.text_width + 10, self.text_height + 4))
# self.renderer.fill((3, 3, self.text_width + 10, self.text_height + 4), sdl2.ext.Color(255, 255, 255))
self.renderer.copy(self.stats_background, dstrect=sdl2.SDL_Rect(3, 3, self.text_width + 10, self.text_height + 4))
self.renderer.copy(self.stats_sprite, dstrect=sdl2.SDL_Rect(8, 5, self.text_width, self.text_height))
self.renderer.copy(self.stats_background, dstrect=sdl2_layer.SDL_Rect(3, 3, self.text_width + 10, self.text_height + 4))
self.renderer.copy(self.stats_sprite, dstrect=sdl2_layer.SDL_Rect(8, 5, self.text_width, self.text_height))
def update_ammo(self, ammo, assets):
"""Update and display the ammo count"""
@ -290,17 +290,17 @@ class GameWindow:
if self.ammo_text != ammo_text:
self.ammo_text = ammo_text
font = self.fonts[20]
self.ammo_sprite = self.factory.from_text(ammo_text, color=sdl2.ext.Color(0, 0, 0), fontmanager=font)
self.ammo_sprite = self.factory.from_text(ammo_text, color=sdl2_layer.ext.Color(0, 0, 0), fontmanager=font)
text_width, text_height = self.ammo_sprite.size
self.ammo_background = self.factory.from_color(sdl2.ext.Color(255, 255, 255), (text_width + 10, text_height + 4))
self.ammo_background = self.factory.from_color(sdl2_layer.ext.Color(255, 255, 255), (text_width + 10, text_height + 4))
text_width, text_height = self.ammo_sprite.size
position = (self.target_size[0] - text_width - 10, self.target_size[1] - text_height - 5)
#self.renderer.fill((position[0] - 5, position[1] - 2, text_width + 10, text_height + 4), sdl2.ext.Color(255, 255, 255))
self.renderer.copy(self.ammo_background, dstrect=sdl2.SDL_Rect(position[0] - 5, position[1] - 2, text_width + 10, text_height + 4))
self.renderer.copy(self.ammo_sprite, dstrect=sdl2.SDL_Rect(position[0], position[1], text_width, text_height))
self.renderer.copy(assets["BMP_BOMB0"], dstrect=sdl2.SDL_Rect(position[0]+25, position[1], 20, 20))
self.renderer.copy(assets["BMP_POISON"], dstrect=sdl2.SDL_Rect(position[0]+85, position[1], 20, 20))
self.renderer.copy(assets["BMP_GAS"], dstrect=sdl2.SDL_Rect(position[0]+140, position[1], 20, 20))
self.renderer.copy(self.ammo_background, dstrect=sdl2_layer.SDL_Rect(position[0] - 5, position[1] - 2, text_width + 10, text_height + 4))
self.renderer.copy(self.ammo_sprite, dstrect=sdl2_layer.SDL_Rect(position[0], position[1], text_width, text_height))
self.renderer.copy(assets["BMP_BOMB0"], dstrect=sdl2_layer.SDL_Rect(position[0]+25, position[1], 20, 20))
self.renderer.copy(assets["BMP_POISON"], dstrect=sdl2_layer.SDL_Rect(position[0]+85, position[1], 20, 20))
self.renderer.copy(assets["BMP_GAS"], dstrect=sdl2_layer.SDL_Rect(position[0]+140, position[1], 20, 20))
# ======================
# VIEW & NAVIGATION
# ======================
@ -342,29 +342,29 @@ class GameWindow:
if not self.audio:
return
sound_path = os.path.join("sound", sound_file)
rw = sdl2.SDL_RWFromFile(byteify(sound_path, "utf-8"), b"rb")
rw = sdl2_layer.SDL_RWFromFile(byteify(sound_path, "utf-8"), b"rb")
if not rw:
raise RuntimeError("Failed to open sound file")
_buf = POINTER(sdl2.Uint8)()
_length = sdl2.Uint32()
_buf = POINTER(sdl2_layer.Uint8)()
_length = sdl2_layer.Uint32()
spec = SDL_AudioSpec(freq=22050, aformat=sdl2.AUDIO_U8, channels=1, samples=2048)
if sdl2.SDL_LoadWAV_RW(rw, 1, byref(spec), byref(_buf), byref(_length)) == None:
spec = SDL_AudioSpec(freq=22050, aformat=sdl2_layer.AUDIO_U8, channels=1, samples=2048)
if sdl2_layer.SDL_LoadWAV_RW(rw, 1, byref(spec), byref(_buf), byref(_length)) == None:
raise RuntimeError("Failed to load WAV")
devid = self.audio_devs[tag]
# Clear any queued audio
sdl2.SDL_ClearQueuedAudio(devid)
sdl2_layer.SDL_ClearQueuedAudio(devid)
# Start playing audio
sdl2.SDL_QueueAudio(devid, _buf, _length)
sdl2.SDL_PauseAudioDevice(devid, 0)
sdl2_layer.SDL_QueueAudio(devid, _buf, _length)
sdl2_layer.SDL_PauseAudioDevice(devid, 0)
def stop_sound(self):
"""Stop all audio playback"""
for dev in self.audio_devs.values():
sdl2.SDL_PauseAudioDevice(dev, 1)
sdl2.SDL_ClearQueuedAudio(dev)
sdl2_layer.SDL_PauseAudioDevice(dev, 1)
sdl2_layer.SDL_ClearQueuedAudio(dev)
# ======================
# INPUT METHODS
@ -372,8 +372,8 @@ class GameWindow:
def load_joystick(self):
"""Initialize joystick support"""
sdl2.SDL_Init(sdl2.SDL_INIT_JOYSTICK)
sdl2.SDL_JoystickOpen(0)
sdl2_layer.SDL_Init(sdl2_layer.SDL_INIT_JOYSTICK)
sdl2_layer.SDL_JoystickOpen(0)
# ======================
# MAIN GAME LOOP
@ -382,7 +382,7 @@ class GameWindow:
def mainloop(self, **kwargs):
"""Main game loop handling events and rendering"""
while self.running:
performance_start = sdl2.SDL_GetPerformanceCounter()
performance_start = sdl2_layer.SDL_GetPerformanceCounter()
self.renderer.clear()
# Execute background update if provided
@ -397,30 +397,30 @@ class GameWindow:
self.draw_white_flash()
# Handle SDL events
events = sdl2.ext.get_events()
events = sdl2_layer.ext.get_events()
for event in events:
if event.type == sdl2.SDL_QUIT:
if event.type == sdl2_layer.SDL_QUIT:
self.running = False
elif event.type == sdl2.SDL_KEYDOWN:
elif event.type == sdl2_layer.SDL_KEYDOWN:
# print in file keycode
keycode = event.key.keysym.sym
key = sdl2.SDL_GetKeyName(event.key.keysym.sym).decode('utf-8')
key = sdl2_layer.SDL_GetKeyName(event.key.keysym.sym).decode('utf-8')
key = key.replace(" ", "_")
# Check for Right Ctrl key to trigger white flash
self.trigger(f"keydown_{key}")
elif event.type == sdl2.SDL_KEYUP:
key = sdl2.SDL_GetKeyName(event.key.keysym.sym).decode('utf-8')
elif event.type == sdl2_layer.SDL_KEYUP:
key = sdl2_layer.SDL_GetKeyName(event.key.keysym.sym).decode('utf-8')
key = key.replace(" ", "_")
self.trigger(f"keyup_{key}")
elif event.type == sdl2.SDL_MOUSEMOTION:
elif event.type == sdl2_layer.SDL_MOUSEMOTION:
self.trigger(f"mousemove_{event.motion.x}, {event.motion.y}")
elif event.type == sdl2.SDL_JOYBUTTONDOWN:
elif event.type == sdl2_layer.SDL_JOYBUTTONDOWN:
key = event.jbutton.button
self.trigger(f"joybuttondown_{key}")
elif event.type == sdl2.SDL_JOYBUTTONUP:
elif event.type == sdl2_layer.SDL_JOYBUTTONUP:
key = event.jbutton.button
self.trigger(f"joybuttonup_{key}")
elif event.type == sdl2.SDL_JOYHATMOTION:
elif event.type == sdl2_layer.SDL_JOYHATMOTION:
hat = event.jhat.hat
value = event.jhat.value
self.trigger(f"joyhatmotion_{hat}_{value}")
@ -431,11 +431,11 @@ class GameWindow:
self.renderer.present()
# Calculate performance and delay
self.performance = ((sdl2.SDL_GetPerformanceCounter() - performance_start) /
sdl2.SDL_GetPerformanceFrequency() * 1000)
self.performance = ((sdl2_layer.SDL_GetPerformanceCounter() - performance_start) /
sdl2_layer.SDL_GetPerformanceFrequency() * 1000)
delay = max(0, self.delay - round(self.performance))
sdl2.SDL_Delay(delay)
sdl2_layer.SDL_Delay(delay)
# ======================
# SPECIAL EFFECTS
@ -444,7 +444,7 @@ class GameWindow:
def trigger_white_flash(self):
"""Trigger the white flash effect"""
self.white_flash_active = True
self.white_flash_start_time = sdl2.SDL_GetTicks()
self.white_flash_start_time = sdl2_layer.SDL_GetTicks()
self.white_flash_opacity = 255
def update_white_flash(self):
@ -452,7 +452,7 @@ class GameWindow:
if not self.white_flash_active:
return False
current_time = sdl2.SDL_GetTicks()
current_time = sdl2_layer.SDL_GetTicks()
elapsed_time = current_time - self.white_flash_start_time
if elapsed_time < 500: # First 500ms : full white
@ -472,7 +472,7 @@ class GameWindow:
"""Draw the white flash overlay"""
if self.white_flash_opacity > 0:
# Create a white surface with the current opacity
white_surface = sdl2.SDL_CreateRGBSurface(
white_surface = sdl2_layer.SDL_CreateRGBSurface(
0, self.target_size[0], self.target_size[1], 32,
0x000000FF, # R mask
0x0000FF00, # G mask
@ -482,8 +482,8 @@ class GameWindow:
if white_surface:
# Fill surface with white
sdl2.SDL_FillRect(white_surface, None,
sdl2.SDL_MapRGBA(white_surface.contents.format,
sdl2_layer.SDL_FillRect(white_surface, None,
sdl2_layer.SDL_MapRGBA(white_surface.contents.format,
255, 255, 255, self.white_flash_opacity))
# Convert to texture and draw
@ -491,13 +491,13 @@ class GameWindow:
white_texture.position = (0, 0)
# Enable alpha blending for the texture
sdl2.SDL_SetTextureBlendMode(white_texture.texture, sdl2.SDL_BLENDMODE_BLEND)
sdl2_layer.SDL_SetTextureBlendMode(white_texture.texture, sdl2_layer.SDL_BLENDMODE_BLEND)
# Draw the white overlay
self.renderer.copy(white_texture, dstrect=sdl2.SDL_Rect(0, 0, self.target_size[0], self.target_size[1]))
self.renderer.copy(white_texture, dstrect=sdl2_layer.SDL_Rect(0, 0, self.target_size[0], self.target_size[1]))
# Clean up
sdl2.SDL_FreeSurface(white_surface)
sdl2_layer.SDL_FreeSurface(white_surface)
# ======================
# UTILITY METHODS
@ -509,16 +509,16 @@ class GameWindow:
def full_screen(self, flag):
"""Toggle fullscreen mode"""
sdl2.SDL_SetWindowFullscreen(self.window.window, flag)
sdl2_layer.SDL_SetWindowFullscreen(self.window.window, flag)
def get_perf_counter(self):
"""Get performance counter for timing"""
return sdl2.SDL_GetPerformanceCounter()
return sdl2_layer.SDL_GetPerformanceCounter()
def close(self):
"""Close the game window and cleanup"""
self.running = False
sdl2.ext.quit()
sdl2_layer.ext.quit()
# ======================
# MAIN GAME LOOP
@ -535,7 +535,7 @@ class GameWindow:
size = self.cell_size
# Create RGBA surface for blood splatter
blood_surface = sdl2.SDL_CreateRGBSurface(
blood_surface = sdl2_layer.SDL_CreateRGBSurface(
0, size, size, 32,
0x000000FF, # R mask
0x0000FF00, # G mask
@ -547,7 +547,7 @@ class GameWindow:
return None
# Lock surface for pixel manipulation
sdl2.SDL_LockSurface(blood_surface)
sdl2_layer.SDL_LockSurface(blood_surface)
# Get pixel data
pixels = cast(blood_surface.contents.pixels, POINTER(c_uint32))
@ -613,31 +613,31 @@ class GameWindow:
pixels[ny * pitch + nx] = color
# Unlock surface
sdl2.SDL_UnlockSurface(blood_surface)
sdl2_layer.SDL_UnlockSurface(blood_surface)
return blood_surface
def draw_blood_surface(self, blood_surface, position):
"""Convert blood surface to texture and return it"""
# Create temporary surface for blood texture
temp_surface = sdl2.SDL_CreateRGBSurface(0, self.cell_size, self.cell_size, 32, 0, 0, 0, 0)
temp_surface = sdl2_layer.SDL_CreateRGBSurface(0, self.cell_size, self.cell_size, 32, 0, 0, 0, 0)
if temp_surface is None:
sdl2.SDL_FreeSurface(blood_surface)
sdl2_layer.SDL_FreeSurface(blood_surface)
return None
# Copy blood surface to temporary surface
sdl2.SDL_BlitSurface(blood_surface, None, temp_surface, None)
sdl2.SDL_FreeSurface(blood_surface)
sdl2_layer.SDL_BlitSurface(blood_surface, None, temp_surface, None)
sdl2_layer.SDL_FreeSurface(blood_surface)
# Create texture from temporary surface
texture = self.factory.from_surface(temp_surface)
sdl2.SDL_FreeSurface(temp_surface)
sdl2_layer.SDL_FreeSurface(temp_surface)
return texture
def combine_blood_surfaces(self, existing_surface, new_surface):
"""Combine two blood surfaces by blending them together"""
# Create combined surface
combined_surface = sdl2.SDL_CreateRGBSurface(
combined_surface = sdl2_layer.SDL_CreateRGBSurface(
0, self.cell_size, self.cell_size, 32,
0x000000FF, # R mask
0x0000FF00, # G mask
@ -649,9 +649,9 @@ class GameWindow:
return existing_surface
# Lock surfaces for pixel manipulation
sdl2.SDL_LockSurface(existing_surface)
sdl2.SDL_LockSurface(new_surface)
sdl2.SDL_LockSurface(combined_surface)
sdl2_layer.SDL_LockSurface(existing_surface)
sdl2_layer.SDL_LockSurface(new_surface)
sdl2_layer.SDL_LockSurface(combined_surface)
# Get pixel data
existing_pixels = cast(existing_surface.contents.pixels, POINTER(c_uint32))
@ -704,13 +704,13 @@ class GameWindow:
combined_pixels[idx] = (final_a << 24) | (final_r << 16) | (final_g << 8) | final_b
# Unlock surfaces
sdl2.SDL_UnlockSurface(existing_surface)
sdl2.SDL_UnlockSurface(new_surface)
sdl2.SDL_UnlockSurface(combined_surface)
sdl2_layer.SDL_UnlockSurface(existing_surface)
sdl2_layer.SDL_UnlockSurface(new_surface)
sdl2_layer.SDL_UnlockSurface(combined_surface)
return combined_surface
def free_surface(self, surface):
"""Safely free an SDL surface"""
if surface is not None:
sdl2.SDL_FreeSurface(surface)
sdl2_layer.SDL_FreeSurface(surface)

82
key.py

@ -1,82 +0,0 @@
#!/usr/bin/env python3
import sys
import os
import sdl2
import sdl2.ext
class KeyLogger:
def __init__(self):
# Initialize SDL2
sdl2.ext.init(joystick=True, video=True, audio=False)
# Initialize joystick support
sdl2.SDL_Init(sdl2.SDL_INIT_JOYSTICK)
sdl2.SDL_JoystickOpen(0)
sdl2.SDL_JoystickOpen(1) # Open the first joystick
sdl2.SDL_JoystickEventState(sdl2.SDL_ENABLE)
self.window = sdl2.ext.Window("Key Logger", size=(640, 480))
self.window.show()
self.running = True
self.key_down = True
self.font = sdl2.ext.FontManager("assets/decterm.ttf", size=24)
def run(self):
# Main loop
while self.running:
# Handle SDL events
events = sdl2.ext.get_events()
for event in events:
self.event = event.type
if event.type == sdl2.SDL_KEYDOWN:
keycode = event.key.keysym.sym
# Log keycode to file
self.message = f"Key pressed: {sdl2.SDL_GetKeyName(keycode).decode('utf-8')}"
elif event.type == sdl2.SDL_KEYUP:
keycode = event.key.keysym.sym
# Log keycode to file
self.message = f"Key released: {sdl2.SDL_GetKeyName(keycode).decode('utf-8')}"
elif event.type == sdl2.SDL_JOYBUTTONDOWN:
button = event.jbutton.button
self.message = f"Joystick button {button} pressed"
if button == 9: # Assuming button 0 is the right trigger
self.running = False
elif event.type == sdl2.SDL_JOYBUTTONUP:
button = event.jbutton.button
self.message = f"Joystick button {button} released"
elif event.type == sdl2.SDL_JOYAXISMOTION:
axis = event.jaxis.axis
value = event.jaxis.value
self.message = f"Joystick axis {axis} moved to {value}"
elif event.type == sdl2.SDL_JOYHATMOTION:
hat = event.jhat.hat
value = event.jhat.value
self.message = f"Joystick hat {hat} moved to {value}"
elif event.type == sdl2.SDL_QUIT:
self.running = False
# Update the window
sdl2.ext.fill(self.window.get_surface(), sdl2.ext.Color(34, 0, 33))
greeting = self.font.render("Press any key...", color=sdl2.ext.Color(255, 255, 255))
sdl2.SDL_BlitSurface(greeting, None, self.window.get_surface(), None)
if hasattr(self, 'message'):
text_surface = self.font.render(self.message, color=sdl2.ext.Color(255, 255, 255))
sdl2.SDL_BlitSurface(text_surface, None, self.window.get_surface(), sdl2.SDL_Rect(0, 30, 640, 480))
if hasattr(self, 'event'):
event_surface = self.font.render(f"Event: {self.event}", color=sdl2.ext.Color(255, 255, 255))
sdl2.SDL_BlitSurface(event_surface, None, self.window.get_surface(), sdl2.SDL_Rect(0, 60, 640, 480))
sdl2.SDL_UpdateWindowSurface(self.window.window)
# Refresh the window
self.window.refresh()
sdl2.SDL_Delay(10)
# Check for quit event
if not self.running:
break
# Cleanup
sdl2.ext.quit()
if __name__ == "__main__":
logger = KeyLogger()
logger.run()

89
rats.py

@ -4,9 +4,9 @@ import random
import os
import json
from engine import maze, sdl2 as engine, controls, graphics, unit_manager, scoring
from engine import maze, controls, graphics, pygame_layer as engine, unit_manager, scoring
from units import points
from engine.user_profile_integration import UserProfileIntegration, get_global_leaderboard
from engine.user_profile_integration import UserProfileIntegration
class MiceMaze(
@ -20,21 +20,55 @@ class MiceMaze(
def __init__(self, maze_file):
# Initialize user profile integration
print("[DEBUG] Initializing user profile integration...")
self.profile_integration = UserProfileIntegration()
print(f"[DEBUG] Profile integration initialized. Has current_profile: {self.profile_integration.current_profile is not None}")
if self.profile_integration.current_profile:
print(f"[DEBUG] Current profile: {self.profile_integration.get_profile_name()}")
else:
print("[DEBUG] No profile loaded, will use default settings")
#self.profile_integration = None
self.map = maze.Map(maze_file)
# Load profile-specific settings
self.audio = self.profile_integration.get_setting('sound_enabled', True)
sound_volume = self.profile_integration.get_setting('sound_volume', 50)
# Load profile'-specific settings
if self.profile_integration is None:
self.audio = True
sound_volume = 50
else:
self.audio = self.profile_integration.get_setting('sound_enabled', True)
sound_volume = self.profile_integration.get_setting('sound_volume', 50)
self.cell_size = 40
self.full_screen = False
# Initialize render engine with profile-aware title
player_name = self.profile_integration.get_profile_name()
if self.profile_integration is None:
player_name = "Guest"
else:
player_name = self.profile_integration.get_profile_name()
window_title = f"Mice! - {player_name}"
# If running under Pyodide in the browser, ensure the JS canvas is bound
# to Pyodide's pygame integration _before_ creating the display. This
# avoids cases where pygame.display.set_mode is called before the HTML
# canvas is attached, which would result in a blank canvas in the page.
try:
# 'js' is available under Pyodide as a proxy to the global window
import js
try:
canvas_el = js.document.getElementById('canvas')
if hasattr(js.pyodide, 'canvas') and hasattr(js.pyodide.canvas, 'setCanvas2D'):
js.pyodide.canvas.setCanvas2D(canvas_el)
except Exception:
# Non-fatal: continue with engine initialization
pass
except Exception:
# Not running under Pyodide (native run), ignore
pass
self.render_engine = engine.GameWindow(self.map.width, self.map.height,
self.cell_size, window_title,
key_callback=self.trigger)
@ -44,7 +78,9 @@ class MiceMaze(
self.render_engine.set_volume(sound_volume)
self.load_assets()
self.render_engine.window.show()
# Show window (for pygame, this is implicit; for SDL2 it's explicit)
if hasattr(self.render_engine.window, 'show'):
self.render_engine.window.show()
self.pointer = (random.randint(1, self.map.width-2), random.randint(1, self.map.height-2))
self.scroll_cursor()
self.points = 0
@ -78,7 +114,7 @@ class MiceMaze(
"max": 8
},
"nuclear": {
"count": 1,
"count": 11,
"max": 1
},
"mine": {
@ -126,8 +162,14 @@ class MiceMaze(
return
if self.game_status == "start_menu":
# Create personalized greeting
player_name = self.profile_integration.get_profile_name()
device_id = self.profile_integration.get_device_id()
if self.profile_integration:
player_name = self.profile_integration.get_profile_name()
else:
player_name = "Guest"
if self.profile_integration:
device_id = self.profile_integration.get_device_id()
else:
device_id = "Unknown Device"
greeting_title = f"Welcome to Mice, {player_name}!"
@ -136,7 +178,7 @@ class MiceMaze(
device_line = f"Device: {device_id}"
# Show profile stats if available
if self.profile_integration.current_profile:
if self.profile_integration and self.profile_integration.current_profile:
profile = self.profile_integration.current_profile
stats_line = f"Best Score: {profile['best_score']} | Games: {profile['games_played']}"
full_subtitle = f"{subtitle}\n{device_line}\n{stats_line}"
@ -145,7 +187,7 @@ class MiceMaze(
self.render_engine.dialog(greeting_title,
subtitle=full_subtitle,
image=self.assets["BMP_WEWIN"])
image=self.assets["da"])
return
self.render_engine.delete_tag("unit")
self.render_engine.delete_tag("effect")
@ -165,6 +207,20 @@ class MiceMaze(
self.scroll()
self.render_engine.new_cycle(50, self.update_maze)
def tick(self):
"""Run a single frame tick: update logic + background update (non-blocking).
Intended to be called repeatedly from the browser via requestAnimationFrame.
"""
try:
# Use render_engine.step() to run one frame iteration (draw + flip)
if hasattr(self.render_engine, 'step'):
self.render_engine.step(update=self.update_maze, bg_update=self.draw_maze)
else:
# Fallback: call update_maze directly
self.update_maze()
except Exception:
pass
def run(self):
self.render_engine.mainloop(update=self.update_maze, bg_update=self.draw_maze)
@ -173,7 +229,7 @@ class MiceMaze(
def game_over(self):
if self.game_end[0]:
if not self.combined_scores:
self.combined_scores = get_global_leaderboard(4)
self.combined_scores = self.profile_integration.get_global_leaderboard(4)
global_scores = []
for entry in self.combined_scores:
@ -202,16 +258,21 @@ class MiceMaze(
return True
count_rats = self.count_rats()
if count_rats > 200:
print("[DEBUG GAME_OVER] Loss condition: rats > 200")
print(f"[DEBUG GAME_OVER] Rat count: {count_rats}, Points: {self.points}")
self.render_engine.stop_sound()
self.render_engine.play_sound("WEWIN.WAV")
self.game_end = (True, False)
self.game_status = "paused"
# Track incomplete game in profile
print(f"[DEBUG GAME_OVER] Calling update_game_stats(completed=False)")
self.profile_integration.update_game_stats(self.points, completed=False)
return True
if not count_rats and not any(isinstance(unit, points.Point) for unit in self.units.values()):
print("[DEBUG GAME_OVER] Win condition: all rats and points cleared")
print(f"[DEBUG GAME_OVER] Points earned: {self.points}")
self.render_engine.stop_sound()
self.render_engine.play_sound("VICTORY.WAV")
self.render_engine.play_sound("WELLDONE.WAV", tag="effects")
@ -219,7 +280,9 @@ class MiceMaze(
self.game_status = "paused"
# Save score to both traditional file and user profile
print(f"[DEBUG GAME_OVER] Calling save_score()")
self.save_score()
print(f"[DEBUG GAME_OVER] Calling update_game_stats(completed=True)")
self.profile_integration.update_game_stats(self.points, completed=True)
return True

2
requirements.txt

@ -1,3 +1,5 @@
pysdl2
pygame
Pillow
pyaml
requests

BIN
sound/converted/Death.wav

Binary file not shown.

BIN
units/__pycache__/rat.cpython-313.pyc

Binary file not shown.

BIN
units/__pycache__/unit.cpython-313.pyc

Binary file not shown.

2
units/rat.py

@ -80,7 +80,7 @@ class Rat(Unit):
units.extend(self.game.unit_positions.get(self.position, []))
for unit in units:
if unit.id == self.id or unit.age < AGE_THRESHOLD or self.position != unit.position_before:
if unit.id == self.id or unit.age < AGE_THRESHOLD:
continue
x1, y1, x2, y2 = self.bbox
ox1, oy1, ox2, oy2 = unit.bbox

6
user_profiles.json

@ -3,9 +3,9 @@
"Player1": {
"name": "Player1",
"created_date": "2024-01-15T10:30:00",
"last_played": "2025-08-24T21:49:04.187787",
"games_played": 25,
"total_score": 15420,
"last_played": "2025-10-17T17:00:43.907946",
"games_played": 28,
"total_score": 15555,
"best_score": 980,
"settings": {
"difficulty": "normal",

Loading…
Cancel
Save