io.github.ikoskela/precision-desktop
Fixes DPI coordinate scaling for Windows desktop automation MCP servers.
Ask AI about io.github.ikoskela/precision-desktop
Powered by Claude Β· Grounded in docs
I know everything about io.github.ikoskela/precision-desktop. Ask me about installation, configuration, usage, or troubleshooting.
0/500
Reviews
Documentation
precision-desktop
mcp-name: io.github.ikoskela/precision-desktop
A companion MCP server that fixes DPI coordinate scaling for Windows desktop automation.
Windows DPI scaling silently breaks every MCP tool that clicks, types, or hovers on the desktop. precision-desktop detects and corrects the coordinate mismatch so your AI agent's clicks actually land where they should.
The Problem
Windows has two coordinate systems and doesn't tell you which one you're using.
When Windows DPI scaling is set above 100% (which it is on most modern laptops and monitors), different Windows APIs return coordinates in different systems:
| Coordinate System | Used By | Example: Point at 50% across a 4K display |
|---|---|---|
| Physical (pixels) | Mouse events, SetCursorPos, UI Automation* | 1920, 1080 |
| Logical (DPI-scaled) | GetWindowRect, Cursor.Position, .NET, screenshots | 1097, 617 (at 175% scaling) |
* UI Automation returns physical on DPI-aware processes, logical on others β yet another inconsistency.
The ratio between them is your DPI scale factor (e.g., 1.25x, 1.5x, 1.75x, 2.0x).
The problem isn't just "some APIs return logical." It's that different APIs return different coordinate systems with no indication of which one you're getting. There's no flag, no header, no type annotation β just numbers that look identical but mean completely different things.
Three ways this breaks AI agents
1. API mismatch β Click tools accept physical coordinates, but common Windows APIs return logical:
What the AI wants to click: [Button at physical (1920, 1080)]
What GetWindowRect says: [Button at logical (1097, 617)]
Where the click lands: (1097, 617) in physical space
^^^^^^^^^ WRONG - completely different spot
2. Screenshot mismatch β Screen captures are taken at logical resolution, but click tools expect physical coordinates. On a 3840x2400 display at 175% scaling, screenshots are 2194x1371 pixels. When a vision model (GPT-4o, Claude, CogAgent) looks at a screenshot and estimates "the button is at pixel (500, 300)," that's a logical coordinate. Clicking there in physical space misses by hundreds of pixels:
Vision model sees: [Button at (500, 300) in screenshot]
Screenshot space: Logical (2194x1371)
Click-Tool expects: Physical (3840x2400)
Correct click: (875, 525) β needs 1.75x conversion
3. Mixed sources within the same tool β Even a single tool can return both systems. For example, windows-mcp's State-Tool reports element coordinates in physical space (correct for clicking), but its screenshots are captured at logical resolution. An agent combining both β reading element positions from the accessibility tree and estimating positions from screenshots β will silently mix coordinate systems.
This isn't an edge case
- ~1 billion active Windows devices worldwide
- ~30-50% have DPI scaling above 100% β that's 300-500 million machines
- Windows auto-enables scaling >100% on most modern laptops (13-16" screens at 1080p+ get 125-150% by default)
- 47% of PC users now run resolutions above 1080p (Steam Hardware Survey) β 21% at 1440p, 5% at 2560x1600, 4.2% at 4K β and climbing
- The MCP ecosystem has 5,800+ servers and 97M+ monthly SDK downloads β every Windows desktop automation server will hit this
If you've ever watched an AI agent click confidently at exactly the wrong spot, DPI scaling is probably why.
The Solution
precision-desktop is a companion MCP server that sits alongside your desktop automation MCP (like windows-mcp) and provides:
- Calibration β Measure the actual DPI scale factor on this specific machine using known screen landmarks
- Coordinate conversion β Convert between physical and logical systems on demand
- UI element finding β Locate elements by name via Windows UI Automation, returning physical coordinates ready for clicking
- Health checks β Detect stale calibration, verify UI Automation availability, check companion MCP status
- Patch awareness β Track which DPI-aware patches have been applied to the companion MCP
Tools
| Tool | Description |
|---|---|
calibrate | Compute DPI scale factors from 2+ reference points with known physical and logical coordinates |
calibrate_verify | Mark calibration as verified after confirming a test click landed correctly |
get_calibration | Read current calibration state (scale factors, verification status, age) |
convert_coordinates | Convert a coordinate pair between physical and logical systems |
find_ui_element | Find a single UI element by name using Windows UI Automation. Returns physical coordinates |
find_all_ui_elements | Find all UI elements matching a name. Returns list with physical coordinates |
list_ui_elements | List all named interactive elements (buttons, text fields, etc.) in a window |
find_window | Find a window handle (hwnd) by title substring |
health_check | Run environment checks: calibration freshness, UI Automation, companion MCP status |
patch_status | Check which DPI-aware patches are applied to the companion MCP |
Quick Start
1. Install
Clone this repo and install dependencies:
git clone https://github.com/ikoskela/precision-desktop.git
cd precision-desktop
pip install -e .
2. Configure MCP
Add to your Claude Code MCP configuration (.mcp.json or settings):
{
"mcpServers": {
"precision-desktop": {
"command": "python",
"args": ["C:/path/to/precision-desktop/server.py"]
}
}
}
Or if using Claude Desktop, add to claude_desktop_config.json:
{
"mcpServers": {
"precision-desktop": {
"command": "python",
"args": ["C:\\path\\to\\precision-desktop\\server.py"]
}
}
}
3. Calibrate
The AI agent calibrates itself on first use. The flow:
- Agent calls
health_checkβ discovers calibration is missing - Agent gathers reference points β uses a coordinate reading tool (like MPos) or Move-Tool + Cursor.Position to get both physical and logical coordinates at 2+ known screen landmarks
- Agent calls
calibratewith the reference points β computes scale factors - Agent verifies β moves cursor to a known element, confirms it landed correctly, calls
calibrate_verify
Calibration persists in state/calibration.json and only needs to be redone if DPI settings change or the display configuration changes.
Calibration Guide
What you need
Two coordinate readings for the same point:
- Physical coordinates β from Move-Tool, Click-Tool, or State-Tool (these operate in physical space)
- Logical coordinates β from
[System.Windows.Forms.Cursor]::Positionin PowerShell, orGetWindowRectAPI calls
Good calibration landmarks
- Start button (bottom-left of taskbar) β easy to locate precisely
- Date/time (bottom-right of taskbar) β anchors the opposite corner
- Minimize button of any maximized window β well-defined clickable target
Example calibration call
{
"points": [
{
"physical_x": 38,
"physical_y": 2365,
"logical_x": 21,
"logical_y": 1351,
"label": "start_button"
},
{
"physical_x": 3691,
"physical_y": 2332,
"logical_x": 2109,
"logical_y": 1332,
"label": "datetime"
}
]
}
This computes a scale factor (here, ~1.75x) and persists it for future coordinate conversions.
Verification
After calibration, the agent should:
- Use
convert_coordinatesto convert a known logical position to physical - Use Move-Tool to move the cursor there
- Confirm the cursor is on the expected target
- Call
calibrate_verifywithsuccess: true
UI Element Finding
Beyond coordinate conversion, precision-desktop can locate elements directly by name using Windows UI Automation β no coordinates needed.
Agent: find_ui_element(element_name="Save", window_title="Notepad")
β { "name": "Save", "center_x": 450, "center_y": 32, "control_type": "button", ... }
(coordinates are physical, ready for Click-Tool)
This works for:
- Native Windows application controls (buttons, text fields, menus)
- Chrome extension popups and dialogs
- Overlay windows that State-Tool may not see
- Any UI element exposed via Windows UI Automation
Scoped search
You can scope searches to a specific window to avoid finding elements in the wrong application:
find_ui_element(element_name="Submit", window_title="My App")
find_ui_element(element_name="Close", window_handle=12345)
Discovery
Don't know the element name? Use list_ui_elements to see what's available:
list_ui_elements(window_title="Settings")
β [{ "name": "General", "control_type": "tab item", ... },
{ "name": "Apply", "control_type": "button", ... }, ...]
Integration with windows-mcp
precision-desktop is designed to work alongside windows-mcp (or any MCP that provides desktop click/type/scroll tools).
The patching concept
Rather than forking windows-mcp, precision-desktop describes patch intents β what should change in the companion MCP and why. The AI agent reads these intents and applies version-appropriate patches itself.
Current patch intents:
dpi_awarenessβ Add an optionalcoordinate_systemparameter to Click-Tool and Move-Tool that auto-converts logical coordinates to physicalfind_and_clickβ Add an optionalelement_nameparameter to Click-Tool that finds and clicks an element by name (no coordinates needed)
Use patch_status to check which patches are applied.
Environment variable
If windows-mcp is installed in a non-standard location, set the WINDOWS_MCP_PATH environment variable:
WINDOWS_MCP_PATH=C:\path\to\windows-mcp
By default, it looks in the standard Claude Extensions directory (%APPDATA%\Claude\Claude Extensions\ant.dir.cursortouch.windows-mcp).
Architecture
precision-desktop/
βββ server.py # MCP server entry point β tool definitions and routing
βββ calibration.py # DPI calibration: compute, persist, convert coordinates
βββ find_element.py # Windows UI Automation: find elements, list elements, find windows
βββ health_check.py # Environment checks: calibration, UI Automation, companion MCP
βββ patches/
β βββ windows_mcp.py # LLM-adaptive patch intents for windows-mcp
βββ state/
β βββ calibration.json # Persisted calibration data (user-specific, gitignored)
βββ pyproject.toml
How calibration works
- User (or AI agent) provides 2+ points with both physical and logical coordinates
calibration.pycomputes the median scale factor for X and Y axes independently- Checks consistency β if points disagree by more than 2%, flags as inconsistent
- Computes offset (typically 0 for standard DPI scaling, non-zero for multi-monitor setups)
- Persists to
state/calibration.json - Subsequent
convert_coordinatescalls use the persisted factors
How UI Automation finding works
find_element.pyruns PowerShell scripts that loadUIAutomationClientandUIAutomationTypesassemblies- Searches the UI Automation tree for elements matching the requested name
- Returns bounding rectangles in the coordinate system that UI Automation reports (which is physical on DPI-aware processes)
- Results include center coordinates ready for direct use with Click-Tool/Move-Tool
Requirements
- Windows 10/11 (uses Windows UI Automation)
- Python 3.10+
- PowerShell (ships with Windows)
- mcp >= 1.0.0 (MCP SDK)
- MPos or similar coordinate reader (for calibration) β any tool that shows the cursor's screen position in both physical and logical coordinates. MPos is lightweight and portable (no install needed)
