jahro-watcher
关于
This Claude Skill automatically adds `[JahroWatch]` attributes to C# fields and properties for real-time runtime monitoring in Unity. It identifies key game state variables and generates performance-safe patterns to replace manual logging. Use it when you need to track variables, inspect game state, or integrate with the JahroWatch system.
快速安装
Claude Code
推荐npx skills add jahro-console/unity-agent-skills -a claude-code/plugin add https://github.com/jahro-console/unity-agent-skillsgit clone https://github.com/jahro-console/unity-agent-skills.git ~/.claude/skills/jahro-watcher在 Claude Code 中复制并粘贴此命令以安装该技能
技能文档
Jahro Watcher
Help users monitor game variables in real-time using Jahro's [JahroWatch] attribute system.
Workflow
- Analyze the user's code — identify fields/properties worth monitoring
- Generate correct
[JahroWatch]attributes - Add registration if needed (instance members require
RegisterObject) - VERIFY — "Enter Play Mode, open the Watcher tab, confirm values update"
Analyzing Code for Watcher Candidates
When the user shares a class, identify members worth monitoring:
Good candidates:
- Game state fields: health, score, level, currency, inventory counts
- Physics values: velocity, position, rotation (especially via properties wrapping Rigidbody)
- Performance metrics: FPS, memory, draw calls
- Enum state fields: game state, player state, AI state
- Key counters: enemy count, player count, item count
Skip these:
- Constants and readonly compile-time values
- Private implementation details that change every frame with no debugging value
- References to other objects (Transform, GameObject) — watch their properties instead
- Collections and dictionaries (arrays are supported, but dictionaries are not)
Suggest replacing Debug.Log polling:
If the user has Debug.Log($"Health: {health}") in Update(), recommend [JahroWatch] instead — it eliminates log spam and provides a clean real-time dashboard.
Attribute Syntax
[JahroWatch("Display Name", "GroupName", "Description for detail modal")]
Constructor: [JahroWatch(string name, string group, string description)]
All parameters are optional. Defaults: name = member name (leading _ stripped), group = "Default", description = "".
Complete example
using JahroConsole;
using UnityEngine;
public class PlayerController : MonoBehaviour
{
[JahroWatch("Health", "Player", "Current hit points")]
public float health = 100f;
[JahroWatch("Stamina", "Player", "Current stamina")]
public float stamina = 50f;
[JahroWatch("Position", "Player", "World position")]
public Vector3 Position => transform.position;
[JahroWatch("Velocity", "Player", "Movement velocity")]
public Vector3 Velocity => GetComponent<Rigidbody>().velocity;
[JahroWatch("Is Grounded", "Player", "Touching ground")]
public bool isGrounded;
void OnEnable() => Jahro.RegisterObject(this);
void OnDisable() => Jahro.UnregisterObject(this);
}
Registration
Instance members — require RegisterObject
void OnEnable() => Jahro.RegisterObject(this);
void OnDisable() => Jahro.UnregisterObject(this);
This same call also registers [JahroCommand] attributes on the class. If the class already has RegisterObject for commands, do not add a second call — one call handles both.
Read references/common-patterns.md for the canonical lifecycle pattern.
Static members — no registration needed
Static fields and properties with [JahroWatch] are discovered via assembly scanning:
public static class GameStats
{
[JahroWatch("Total Score", "Game")]
public static int Score;
[JahroWatch("Session Time", "Game")]
public static float SessionTime => Time.realtimeSinceStartup;
}
Adding watchers to a class that already has commands
If the class already has [JahroCommand] attributes and RegisterObject, just add [JahroWatch] attributes — no registration changes:
public class GameManager : MonoBehaviour
{
// Existing command
[JahroCommand("reset-game", "Game", "Reset game")]
public void ResetGame() { /* ... */ }
// New watchers — just add attributes
[JahroWatch("Player Count", "Game")]
public int playerCount;
[JahroWatch("Game Time", "Game")]
public float gameTime;
// Already present — no changes needed
void OnEnable() => Jahro.RegisterObject(this);
void OnDisable() => Jahro.UnregisterObject(this);
}
Supported Types
| Type | List View Display | Detail Modal |
|---|---|---|
int, float, double, bool | Value as-is | Same |
string | Truncated | Full text |
Vector2 | Compact coords | Coords + magnitude |
Vector3 | Compact coords | Coords + magnitude |
Quaternion | Raw values | Raw + Euler angles |
Transform | Position | Position, rotation, scale, child count |
Rigidbody | Summary | Mass, kinematic, gravity, velocity, angular velocity |
Collider | Summary | Trigger status, material, bounds |
AudioSource | Summary | Clip, volume, loop, pitch, mute |
Camera | Summary | FOV, clip planes, aspect ratio |
Arrays (any T[]) | TypeName[length] | Full contents |
For custom types not in this table, the watcher calls .ToString(). If you need rich display, consider watching individual primitive properties instead.
Read references/api-reference.md for the full type display details.
Performance
Watchers are designed to be safe for development and testing:
- Values are only read when the Watcher UI tab is visible. No continuous polling when the console is closed or on another tab.
- No overhead when disabled. If Jahro is disabled via
JAHRO_DISABLEor auto-disable, watchers are never evaluated. - Be mindful of expensive property getters. A property like
public int Count => expensiveList.Where(...).Count()runs its getter every frame the Watcher is open. Cache expensive computations.
Performance-safe property pattern
private float _cachedFps;
private float _lastFpsUpdate;
void Update()
{
if (Time.time - _lastFpsUpdate > 0.25f)
{
_cachedFps = 1f / Time.unscaledDeltaTime;
_lastFpsUpdate = Time.time;
}
}
[JahroWatch("FPS", "Performance")]
public float FPS => _cachedFps;
Group Organization
By system (recommended default)
"Player" — Health, Stamina, Position, Velocity
"Physics" — Is Grounded, Angular Velocity, Collision Count
"Performance" — FPS, Memory, Draw Calls
"Game" — Game State, Level Progress, Player Count
"AI" — AI State, Target, Path Length
By priority
"Critical" — Health, Frame Time (always need these)
"Gameplay" — Enemy Count, Spawn Timer
"Diagnostics" — GC Allocs, Memory
Watcher UI behavior
- Favorites group always appears at top (user stars individual watchers)
- Custom groups are alphabetically sorted
- Groups are collapsible to reduce clutter
- Tapping a watcher opens a detail modal with description and full type info
Contextual Awareness
| Pattern in code | Suggestion |
|---|---|
Debug.Log in Update() logging variable values | Replace with [JahroWatch] — cleaner, no log spam |
[JahroWatch] already present | Suggest additional watchers, better groups, performance tips |
[JahroCommand] but no watchers | Suggest adding watchers for key state the commands modify |
| Rigidbody or physics-heavy code | Suggest velocity, angular velocity, isGrounded watchers |
| Game manager with state fields | Suggest watching game state enum, counts, timers |
Verification
After generating watchers, always include:
Verify: Enter Play Mode → press ~ → switch to the Watcher tab. Confirm your watched values appear in the correct groups and update in real-time as the game runs. Tap a watcher to see its detail modal.
If watchers appear but don't update, or don't appear at all, suggest the jahro-troubleshooting skill — common causes: missing RegisterObject, Watcher tab not open, object destroyed without unregistering.
GitHub 仓库
相关推荐技能
content-collections
元Content Collections 是一个 TypeScript 优先的构建工具,可将本地 Markdown/MDX 文件转换为类型安全的数据集合。它专为构建博客、文档站和内容密集型 Vite+React 应用而设计,提供基于 Zod 的自动模式验证。该工具涵盖从 Vite 插件配置、MDX 编译到生产环境部署的完整工作流。
polymarket
元这个Claude Skill为开发者提供完整的Polymarket预测市场开发支持,涵盖API调用、交易执行和市场数据分析。关键特性包括实时WebSocket数据流,可监控实时交易、订单和市场动态。开发者可用它构建预测市场应用、实施交易策略并集成实时市场预测功能。
creating-opencode-plugins
元该Skill帮助开发者创建OpenCode插件,用于接入命令、文件、LSP等25+种事件。它提供了插件结构、事件API规范和JavaScript/TypeScript实现模式,适合需要拦截操作、扩展功能或自定义事件处理的场景。开发者可通过它快速构建响应式模块来增强OpenCode AI助手的能力。
sglang
元SGLang是一个专为LLM设计的高性能推理框架,特别适用于需要结构化输出的场景。它通过RadixAttention前缀缓存技术,在处理JSON、正则表达式、工具调用等具有重复前缀的复杂工作流时,能实现极速生成。如果你正在构建智能体或多轮对话系统,并追求远超vLLM的推理性能,SGLang是理想选择。
