Free SKILL.md scraped from GitHub. Clone the repo or copy the file directly into your Claude Code skills directory.
npx versuz@latest install community-access-accessibility-agents-gemini-extensions-a11y-agents-skills-nvda-addon-specialistgit clone https://github.com/Community-Access/accessibility-agents.gitcp accessibility-agents/SKILL.MD ~/.claude/skills/community-access-accessibility-agents-gemini-extensions-a11y-agents-skills-nvda-addon-specialist/SKILL.md# NVDA Addon Development Specialist
Expert in NVDA screen reader addon development covering architecture, APIs, plugin types (globalPlugins, appModules, synthDrivers, brailleDisplayDrivers), manifest format, event/script handling, NVDAObject overlays, tree interceptors, addon packaging, Add-on Store submission, testing with NVDA, and internationalization. Grounded in the official NVDA source code (github.com/nvaccess/nvda) and community development guides.
## Core Principles
1. **Source code is the authority.** Every claim verified against the NVDA source.
2. **Never block the main thread.** Blocking calls freeze all speech, braille, and input.
3. **Always call `nextHandler()`.** Event handlers must pass events downstream.
4. **Use the `@script` decorator.** Modern pattern, not legacy `__gestures` dicts.
5. **Test with the real screen reader.** Verify addons with NVDA itself.
6. **Package for the Add-on Store.** Follow the official submission process.
## NVDA 2026.1 Architecture Transition
NVDA 2026.1 is a **major architecture transition** and an **add-on API compatibility breaking release**. All addons must be re-tested and have their manifests updated.
### 64-bit Transition
- **NVDA is now built with Python 3.13, 64-bit.** The 32-bit era is over.
- **32-bit Windows is no longer supported.** Windows 10 (Version 1507) 64-bit is the new minimum.
- **Windows 10 on ARM is dropped.** ARM64 support targets Windows 11 only (via ARM64EC libraries).
- **No backward-compatibility layer for 32-bit native libraries.** Addons shipping 32-bit `.dll` files or using 32-bit `ctypes` bindings will break. Recompile all native code as 64-bit.
- `NVDAHelper.localLib` changed from `ctypes.CDLL` to a module - use `.dll` attribute for the CDLL object.
- The Microsoft Universal C Runtime is no longer bundled.
### SAPI Restructuring
- `sapi5` now refers to 64-bit SAPI 5 voices.
- Use `sapi5_32` to access 32-bit SAPI 5 voices (no audio ducking support).
- `sapi4` removed entirely - use `sapi4_32` instead (no audio ducking support).
### Key API Breaking Changes
- **`versionInfo` split:** `copyrightYears` and `url` moved to `buildVersion` module.
- **`winUser`, `winKernel`, `winGDI`, `shellapi`, `hwIo.hid.hidDll`** symbols moved to `winBindings.*` submodules.
- **Screen Curtain:** `visionEnhancementProviders.screenCurtain` replaced with `screenCurtain` subpackage.
- **MathPlayer removed:** `comInterfaces.MathPlayer` and `mathPres.mathPlayer` are gone.
- **`ftdi2` refactored** into a package with snake_case functions, new enums, and typed FFI bindings.
- **`gui.nvdaControls.TabbableScrolledPanel` removed** - use `wx.lib.scrolledpanel.ScrolledPanel`.
- **`typing_extensions` removed** -- Python 3.13 has native support.
- **License changed** to GPL-2-or-later.
### Deprecations (Still Present, Will Be Removed)
- `NVDAHelper.versionedLibPath` - use `NVDAState.ReadPaths.versionedLibX86Path`
- `NVDAHelper.coreArchLibPath` - use `NVDAState.ReadPaths.coreArchLibPath`
- `winVersion.WIN81` - Windows 8.1 is no longer supported
### Manifest Version Guidance
Use the following table to choose the right `minimumNVDAVersion` and `lastTestedNVDAVersion` values for your addon's manifest.ini.
| Scenario | `minimumNVDAVersion` | `lastTestedNVDAVersion` |
|----------|---------------------|------------------------|
| New addon | `2025.1.0` | `2026.1.0` |
| Broad compatibility (Python 3 required) | `2019.3.0` | `2026.1.0` |
| Widest safe range | `2024.1.0` | `2026.1.0` |
**Absolute minimum for Python 3:** `2019.3.0` -- this is the first NVDA release that requires Python 3. Never set `minimumNVDAVersion` below `2019.3.0`.
**Important:** Addons using native (C/C++) DLLs must set `minimumNVDAVersion` to `2026.1.0` if they ship 64-bit binaries, since earlier NVDA versions are 32-bit and cannot load 64-bit DLLs.
### 2026.1 Sources
Based on the [NVDA 2026.1 changelog](https://github.com/nvaccess/nvda/blob/master/user_docs/en/changes.md#20261) and the following GitHub issues:
- 64-bit Python 3.13: [#18591](https://github.com/nvaccess/nvda/issues/18591), [#19111](https://github.com/nvaccess/nvda/issues/19111)
- 32-bit and ARM deprecation: [#18684](https://github.com/nvaccess/nvda/issues/18684)
- NVDAHelper/localLib changes: [#18207](https://github.com/nvaccess/nvda/issues/18207)
- ARM64EC libraries: [#18570](https://github.com/nvaccess/nvda/issues/18570)
- Universal C Runtime removal: [#19508](https://github.com/nvaccess/nvda/issues/19508)
- SAPI 4/5 restructuring: [#19432](https://github.com/nvaccess/nvda/issues/19432)
- versionInfo split: [#18682](https://github.com/nvaccess/nvda/issues/18682)
- winBindings migration: [#18860](https://github.com/nvaccess/nvda/issues/18860), [#18883](https://github.com/nvaccess/nvda/issues/18883), [#18896](https://github.com/nvaccess/nvda/issues/18896)
- Screen Curtain refactor: [#19177](https://github.com/nvaccess/nvda/issues/19177)
- MathPlayer removal: [#19239](https://github.com/nvaccess/nvda/issues/19239)
- ftdi2 refactor: [#19105](https://github.com/nvaccess/nvda/issues/19105)
- TabbableScrolledPanel removal: [#17751](https://github.com/nvaccess/nvda/issues/17751)
- typing_extensions removal: [#18689](https://github.com/nvaccess/nvda/issues/18689)
## NVDA Architecture
### Event Chain
```
API Handler (IAccessible/UIA/JAB)
-> eventHandler.executeEvent()
-> Global Plugins .event_*()
-> App Module .event_*()
-> Tree Interceptor .event_*()
-> NVDAObject .event_*()
```
### Script Resolution Order
```
1. gesture.scriptableObject
2. Global Plugins (all, in order)
3. App Module (focused app)
4. Braille Display Driver
5. Vision Enhancement Providers
6. Tree Interceptor
7. Focused NVDAObject
8. Focus Ancestors (if canPropagate=True)
9. globalCommands
```
## Addon Types
### Global Plugins
- Location: `addon/globalPlugins/yourAddon.py`
- Base: `globalPluginHandler.GlobalPlugin`
- System-wide commands and event handlers
### App Modules
- Location: `addon/appModules/appname.py` (named after executable)
- Base: `appModuleHandler.AppModule`
- Per-application accessibility support
### Synth Drivers
- Location: `addon/synthDrivers/mySynth.py`
- Base: `synthDriverHandler.SynthDriver`
- `check()`, `speak()`, `cancel()`, `supportedSettings`
### Braille Display Drivers
- Location: `addon/brailleDisplayDrivers/myDisplay.py`
- Base: `braille.BrailleDisplayDriver`
- `check()`, `display()`, `numCells`
## The @script Decorator
```python
from scriptHandler import script
@script(
description=_("Announces the current time"),
category="My Addon",
gesture="kb:NVDA+shift+t",
speakOnDemand=True,
)
def script_announceTime(self, gesture):
import ui, time
ui.message(time.strftime("%H:%M:%S"))
```
Parameters: `description`, `category`, `gesture`/`gestures`, `canPropagate`, `bypassInputHelp`, `allowInSleepMode`, `resumeSayAllMode`, `speakOnDemand`.
## Addon File Structure
```
myAddon/
addon/
globalPlugins/
appModules/
synthDrivers/
brailleDisplayDrivers/
doc/en/readme.md
locale/en/LC_MESSAGES/
installTasks.py
uninstallTasks.py
manifest.ini
buildVars.py
sconstruct
```
### manifest.ini
```ini
name = myAddon
summary = My Addon Display Name
description = What the addon does.
author = Your Name <email@example.com>
url = https://github.com/yourname/myAddon
version = 1.0.0
minimumNVDAVersion = 2025.1.0
lastTestedNVDAVersion = 2026.1.0
```
**Note:** The lowest allowed `minimumNVDAVersion` for Python 3 addons is `2019.3.0`. For addons shipping native 64-bit DLLs, use `2026.1.0` as the minimum.
## Common Patterns
### Dynamic Announcements
```python
import ui, braille
ui.message("Download complete")
braille.handler.message("Download complete")
```
### Timer-Based Monitoring
```python
import wx
class GlobalPlugin(globalPluginHandler.GlobalPlugin):
def __init__(self):
super().__init__()
self._timer = wx.CallLater(1000, self._checkStatus)
def terminate(self):
if self._timer:
self._timer.Stop()
```
### Anti-Patterns
- **Monkey-patching** core modules -- use extension points or event handlers
- **Main thread blocking** -- use `threading.Thread` + `wx.CallAfter()`
- **Bare except** -- use specific exceptions, log errors
## Detection Rules
| ID | Severity | What to Flag |
|---|---|---|
| NVDA-001 | Critical | Missing `nextHandler()` call in event handler |
| NVDA-002 | Critical | Main thread blocking (sleep, sync I/O) |
| NVDA-003 | Serious | Missing `addonHandler.initTranslation()` |
| NVDA-004 | Serious | Missing `terminate()` cleanup |
| NVDA-005 | Serious | Incorrect manifest version format |
| NVDA-006 | Moderate | Monkey-patching core modules |
| NVDA-007 | Moderate | Script without `@script` decorator |
| NVDA-008 | Moderate | Missing script description |
| NVDA-009 | Moderate | Hardcoded gesture conflicts |
| NVDA-010 | Serious | UI updates from background thread |
| NVDA-011 | Moderate | Missing `check()` classmethod on drivers |
| NVDA-012 | Minor | Bare `except:` clause |
| NVDA-013 | Serious | Incompatible API version range |
| NVDA-014 | Minor | Missing SHA256 for store submission |
| NVDA-015 | Moderate | Not using `config.conf.spec` |
| NVDA-016 | Serious | Secure mode vulnerability |
| NVDA-017 | Critical | 32-bit native library on 64-bit NVDA |
| NVDA-018 | Serious | `minimumNVDAVersion` below `2019.3.0` |
## Cross-Team Routing
| Need | Route To |
|------|----------|
| wxPython GUI / sizers / events | `wxpython-specialist` |
| Screen reader testing (NVDA, JAWS) | `desktop-a11y-testing-coach` |
| Platform a11y APIs (UIA, MSAA, NSAccessibility) | `desktop-a11y-specialist` |
| Build a11y scanner / rule engine | `a11y-tool-builder` |
| Web accessibility audit | `web-accessibility-wizard` |
| Document accessibility audit | `document-accessibility-wizard` |