.agents/skills/mobile-accessibility/SKILL.md
Mobile accessibility reference data for React Native, Expo, iOS, and Android auditing. Covers accessibilityLabel, accessibilityRole, accessibilityHint, touch target sizes (44x44pt minimum), screen reader compatibility, and platform-specific semantics. Use when reviewing any React Native or native mobile code for accessibility.
npx skillsauth add dodyg/blue-nile-pds mobile-accessibilityInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
This skill provides mobile accessibility reference data for React Native, Expo, iOS, and Android auditing. Used by mobile-accessibility.agent.md.
| Prop | Type | Values / Notes | WCAG SC | Required |
|------|------|---------------|---------|---------|
| accessible | boolean | true marks the view as an accessibility node | 4.1.2 | Conditional |
| accessibilityLabel | string | Human-readable name - overrides all child text | 1.1.1, 4.1.2 | Yes (all interactive + image elements) |
| accessibilityLabelledBy | string / string[] | References ID(s) of labelling element | 1.3.1 | For form inputs |
| accessibilityRole | string (see roles table) | Communicates element type to AT | 4.1.2 | Yes (all interactive elements) |
| accessibilityHint | string | Additional context spoken after label + role | 1.3.3 | When action isn't obvious |
| accessibilityState | object | {checked, disabled, expanded, selected, busy} | 4.1.2 | State-bearing elements |
| accessibilityValue | object | {min, max, now, text} | 1.3.1 | Sliders, steppers, progress bars |
| accessibilityActions | array | [{name, label}] - defines custom actions | 4.1.3 | Context menus, long-press alternatives |
| onAccessibilityAction | function | Handles custom action triggers | 4.1.3 | Paired with accessibilityActions |
| accessibilityLiveRegion | string | 'none' / 'polite' / 'assertive' | 4.1.3 | Dynamic content updates |
| accessibilityViewIsModal | boolean | true traps VoiceOver focus inside modal | 1.3.4 | Modals, drawers, sheets |
| accessibilityElementsHidden | boolean | iOS - hides element and children from VoiceOver | 1.1.1 | Decorative elements |
| importantForAccessibility | string | Android - 'auto' / 'yes' / 'no' / 'no-hide-descendants' | 1.1.1 | Decorative / grouped elements |
| accessibilityIgnoresInvertColors | boolean | iOS - preserves colors in Inverted Colors mode | - | Images, video |
| aria-label | string | RN 0.73+ alias for accessibilityLabel | 1.1.1, 4.1.2 | Preferred in new code |
| aria-labelledby | string | RN 0.73+ alias for accessibilityLabelledBy | 1.3.1 | Form inputs |
| aria-describedby | string | RN 0.73+ alias for accessibilityHint | 1.3.3 | Additional description |
| aria-role | string | RN 0.73+ alias for accessibilityRole | 4.1.2 | All interactive elements |
| aria-checked | boolean / 'mixed' | RN 0.73+ alias for accessibilityState.checked | 4.1.2 | Checkboxes |
| aria-disabled | boolean | RN 0.73+ alias for accessibilityState.disabled | 4.1.2 | Disabled elements |
| aria-expanded | boolean | RN 0.73+ alias for accessibilityState.expanded | 4.1.2 | Accordions, dropdowns |
| aria-selected | boolean | RN 0.73+ alias for accessibilityState.selected | 4.1.2 | Tabs, list items |
| aria-busy | boolean | RN 0.73+ alias for accessibilityState.busy | 4.1.3 | Loading elements |
| aria-hidden | boolean | RN 0.73+ - maps to importantForAccessibility / accessibilityElementsHidden | 1.1.1 | Decorative content |
| aria-live | string | RN 0.73+ alias for accessibilityLiveRegion | 4.1.3 | Dynamic content |
| aria-modal | boolean | RN 0.73+ alias for accessibilityViewIsModal | 1.3.4 | Modals |
| Role | Maps to (iOS) | Maps to (Android) | Use For |
|------|-------------|-----------------|--------|
| 'button' | UIAccessibilityTraitButton | AccessibilityNodeInfo.ROLE_BUTTON | Buttons, submission triggers |
| 'link' | UIAccessibilityTraitLink | AccessibilityNodeInfo.ROLE_LINK | Navigation links, external URLs |
| 'search' | - | ROLE_SEARCH | Search bars |
| 'image' | UIAccessibilityTraitImage | ROLE_IMAGE | Images (when accessible=true) |
| 'imagebutton' | UIAccessibilityTraitImage+Button | ROLE_BUTTON | Icon buttons |
| 'header' | UIAccessibilityTraitHeader | ROLE_HEADING | Headings |
| 'text' | UIAccessibilityTraitStaticText | ROLE_LABEL | Static text |
| 'adjustable' | UIAccessibilityTraitAdjustable | ROLE_SCROLL_VIEW | Sliders |
| 'checkbox' | - | ROLE_CHECKBOX | Checkboxes |
| 'combobox' | - | ROLE_DROP_DOWN_LIST | Dropdowns |
| 'menu' | - | ROLE_MENU | Menus |
| 'menuitem' | - | ROLE_MENU_ITEM | Menu items |
| 'menubar' | - | ROLE_MENU_BAR | Menu bars |
| 'progressbar' | UIAccessibilityTraitUpdatesFrequently | ROLE_PROGRESS_BAR | Progress indicators |
| 'radio' | - | ROLE_RADIO_BUTTON | Radio buttons |
| 'radiogroup' | - | - | Radio button groups |
| 'scrollbar' | - | ROLE_SCROLL_BAR | Scrollbars |
| 'spinbutton' | - | ROLE_SCROLL_VIEW | Steppers, number inputs |
| 'switch' | - | ROLE_SWITCH | Toggle switches |
| 'tab' | - | ROLE_TAB | Tab elements |
| 'tablist' | - | ROLE_TAB_LIST | Tab containers |
| 'timer' | UIAccessibilityTraitUpdatesFrequently | - | Countdown timers |
| 'toolbar' | - | ROLE_TOOL_BAR | Toolbars |
| 'grid' | - | ROLE_GRID | Data grids |
| 'list' | - | ROLE_LIST | Lists |
| 'listitem' | - | ROLE_LIST_ITEM | List items |
| 'summary' | UIAccessibilityTraitSummaryElement | - | Summary/status views |
| 'alert' | UIAccessibilityTraitCausesPageTurn | ROLE_ALERT | Alert dialogs |
| 'none' | UIAccessibilityTraitNone | ROLE_NONE | Suppress role |
| Platform | Minimum Size | Recommended | Standard | |----------|-------------|-------------|---------| | iOS | 44 x 44 pt | 44 x 44 pt | HIG | | Android | 48 x 48 dp | 48 x 48 dp | Material Design | | Web mobile | 44 x 44 CSS px (AAA) | 44 x 44 CSS px | WCAG 2.5.5 | | Web mobile (AA, 2.2) | 24 x 24 CSS px with spacing | 44 x 44 CSS px | WCAG 2.5.8 |
// Violation: TouchableOpacity below minimum
const styles = StyleSheet.create({
closeBtn: { width: 24, height: 24 }, // FAIL - below 44pt
iconBtn: { width: 32, height: 32 }, // FAIL - below 44pt
navBtn: { padding: 4 }, // CONDITIONAL - depends on content size
compliant: { width: 44, height: 44 }, // PASS
compliantWithPadding: { padding: 12 }, // PASS if content >= 20pt
});
| Modifier | Purpose |
|----------|---------|
| .accessibilityLabel("...") | Overrides spoken name |
| .accessibilityHint("...") | Spoken usage hint (after pause) |
| .accessibilityValue("...") | Spoken current value |
| .accessibilityHidden(true) | Removes from VoiceOver tree |
| .accessibilityElement(children: .combine) | Merges child elements into one node |
| .accessibilityElement(children: .contain) | Groups children as sub-elements |
| .accessibilityElement(children: .ignore) | Container becomes accessible, children hidden |
| .accessibilityAddTraits(.isButton) | Adds role trait |
| .accessibilityRemoveTraits(.isImage) | Removes wrong role trait |
| .accessibilityInputLabels(["..."]) | Voice Control activation labels |
| .accessibilitySortPriority(n) | Overrides VoiceOver reading order (higher = earlier) |
| .accessibilityAction(named: "...", {}) | Custom action in the Actions rotor |
| .accessibilityActivationPoint(CGPoint) | Override activation tap point |
| .accessibilityCustomContent("label", "value") | Extra info in Accessibility Inspector |
isButton, isHeader, isLink, isImage, isStaticText, isSelected, isKeyboardKey, isSearchField, playsSound, isModal, updatesFrequently, startsMediaSession, allowsDirectInteraction, causesPageTurn, isTabBar, isSummaryElement
| Property | Type | Notes |
|----------|------|-------|
| isAccessibilityElement | Bool | Set true on custom views |
| accessibilityLabel | String? | Overrides spoken name |
| accessibilityHint | String? | Spoken after pause |
| accessibilityValue | String? | Current value (sliders, progress) |
| accessibilityTraits | UIAccessibilityTraits | Bitfield of traits (.button, .header, etc.) |
| accessibilityFrame | CGRect | Determines VoiceOver focus rect |
| accessibilityActivate() | func | Override activation behavior |
| accessibilityElements | [Any]? | Set container's VoiceOver child order |
| shouldGroupAccessibilityChildren | Bool | Groups all children into single node |
| accessibilityViewIsModal | Bool | true = VoiceOver trapped inside |
| accessibilityElementsHidden | Bool | Hides all children from VoiceOver |
| Modifier / Property | Purpose |
|--------------------|---------|
| semantics { contentDescription = "..." } | Accessible name |
| semantics { role = Role.Button } | Element role |
| semantics { stateDescription = "..." } | Current state text |
| semantics { heading() } | Marks as heading |
| semantics { selected = true/false } | Selected state |
| semantics { toggleableState = ToggleableState.On } | Toggle state |
| semantics { onClick(label = "...", action = {...}) } | Click action with label |
| semantics { disabled() } | Disabled state |
| semantics { focused = true } | Force focus |
| semantics { liveRegion = LiveRegion.Polite } | Live region announcements |
| semantics { invisibleToUser() } | Hide from TalkBack |
| semantics { mergeDescendants = true } | Merge child semantics into one node |
| clearAndSetSemantics { ... } | Replace all descendant semantics |
| Modifier.semantics(mergeDescendants = true) { } | Short merge pattern |
Role.Button, Role.Checkbox, Role.DropdownList, Role.Image, Role.RadioButton, Role.Switch, Role.Tab
// VIOLATION
<TouchableOpacity onPress={close}>
<Icon name="x" size={20} />
</TouchableOpacity>
// FIX
<TouchableOpacity
onPress={close}
accessibilityRole="button"
accessibilityLabel="Close"
>
<Icon name="x" size={20} aria-hidden />
</TouchableOpacity>
// VIOLATION
<Image source={productImage} style={styles.product} />
// FIX - informational image
<Image
source={productImage}
style={styles.product}
accessibilityLabel="Blue suede shoes, size 10"
/>
// FIX - decorative image
<Image
source={decorativeBackground}
style={styles.bg}
accessible={false}
importantForAccessibility="no"
/>
// VIOLATION - placeholder is not a label
<TextInput placeholder="Email" value={email} onChangeText={setEmail} />
// FIX
<View>
<Text nativeID="emailLabel">Email address</Text>
<TextInput
value={email}
onChangeText={setEmail}
accessibilityLabelledBy="emailLabel"
accessibilityHint="Enter your email address"
keyboardType="email-address"
autoComplete="email"
/>
</View>
// VIOLATION
<TouchableOpacity onPress={toggle}>
<Image source={checked ? checkedIcon : uncheckedIcon} />
<Text>Accept terms</Text>
</TouchableOpacity>
// FIX
<TouchableOpacity
onPress={toggle}
accessibilityRole="checkbox"
accessibilityState={{ checked }}
accessibilityLabel="Accept terms and conditions"
>
<Image source={checked ? checkedIcon : uncheckedIcon} accessible={false} />
<Text>Accept terms</Text>
</TouchableOpacity>
// VIOLATION - custom modal without VoiceOver trap
<View style={styles.modal}>
<Text>Are you sure?</Text>
<Button title="Confirm" onPress={confirm} />
</View>
// FIX - use Modal component (traps focus automatically) or set accessibilityViewIsModal
<Modal
visible={visible}
transparent
accessibilityViewIsModal={true} // traps VoiceOver
onRequestClose={close} // Android back button
>
<View style={styles.overlay}>
<Text>Are you sure?</Text>
<Button title="Confirm" onPress={confirm} />
<Button title="Cancel" onPress={close} />
</View>
</Modal>
// VIOLATION
Image(
painter = painterResource(id = R.drawable.product),
contentDescription = null // null = decorative, but wrong for informational image
)
// FIX - informational
Image(
painter = painterResource(id = R.drawable.product),
contentDescription = stringResource(R.string.product_image_description)
)
// FIX - truly decorative
Image(
painter = painterResource(id = R.drawable.divider),
contentDescription = null,
modifier = Modifier.semantics { invisibleToUser() }
)
Xcode -> Xcode menu -> Open Developer Tool -> Accessibility Inspector
- Run audit: Click "Audit" tab -> "Run Audit" button
- Inspect: "Inspection" tab -> hover over elements in Simulator
- Keyboard: Use +F7 to toggle VoiceOver in Simulator
# Install (Play Store or adb)
adb install com.google.android.apps.accessibility.auditor
# Enable TalkBack via ADB (for CI)
adb shell settings put secure enabled_accessibility_services \
com.google.android.marvin.talkback/.TalkBackService
# Check accessibility node tree
adb shell uiautomator dump /sdcard/ui_dump.xml
adb pull /sdcard/ui_dump.xml
npm install --save-dev @testing-library/react-native
import { render, screen, fireEvent } from '@testing-library/react-native';
test('button has accessible name and role', () => {
render(<SubmitButton onPress={jest.fn()} />);
const btn = screen.getByRole('button', { name: /submit/i });
expect(btn).toBeTruthy();
});
test('checkbox updates state', () => {
render(<TermsCheckbox />);
const checkbox = screen.getByRole('checkbox', { name: /accept terms/i });
expect(checkbox).toHaveAccessibilityState({ checked: false });
fireEvent.press(checkbox);
expect(checkbox).toHaveAccessibilityState({ checked: true });
});
# .maestro/accessibility-checks.yaml
appId: com.example.myapp
---
- launchApp
- assertVisible:
label: "Submit form"
- tapOn:
label: "Close"
- assertNotVisible:
label: "Are you sure?"
testing
Get best practices for TUnit unit testing, including data-driven tests
development
Severity scoring, scorecard computation, confidence levels, and remediation tracking for web accessibility audits. Use when computing page accessibility scores (0-100 with A-F grades), tracking remediation progress across audits, or generating cross-page comparison scorecards.
development
Web content discovery, URL crawling, and page inventory for accessibility audits. Use when scanning web pages, crawling sites for audit scope, or building page inventories for multi-page audits.
development
Audit report formatting, severity scoring, scorecard computation, and compliance export for document accessibility audits. Use when generating DOCUMENT-ACCESSIBILITY-AUDIT.md reports, computing document severity scores (0-100 with A-F grades), creating VPAT/ACR compliance exports, or formatting remediation priorities.