iOS 无障碍化(适老化)适配总结

原文地址: iOS 无障碍化(适老化)适配总结

VoiceOver 和 Accessibility

iOS 开发中主要讨论的是 UIAccessibility 的 API 在 VoiceOver 上的运用.

旁白使用手册-在 iPhone 上学习旁白手势

检测当前是否开启无障碍模式

系统通知:UIAccessibilityVoiceOverStatusDidChangeNotification

系统方法:UIAccessibilityIsVoiceOverRunning()

iOS代码适配示例

UILabel *titleLabel = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, 100, 100)];
// 描述控件是什么 第一个播放
titleLabel.accessibilityLabel = @"标签";
// 元素的特征 如按钮、链接等 第二个播放
titleLabel.accessibilityTraits = UIAccessibilityTraitStaticText;
// 附加说明 第三个播放, 只在真机播放, 且有一秒延迟
titleLabel.accessibilityHint = @"提示";
// 元素的frame(基于屏幕坐标系, 一般不用填设置), 当系统聚焦到当前元素时, 会有一个黑色的外框, 该值就是聚焦框的大小. 当元素过小时可以通过设置该 frame 使得容易点击, 这个不会改变 app 的 UI. 默认为 CGRectZero.
titleLabel.accessibilityFrame = CGRectMake(0, 0, 100, 100);
// 元素的值 用在UISlider,UITextField等组件上 用来描述元素的值
titleLabel.accessibilityValue = @"60%";
// VoiceOver会把这几个属性连接起来, 朗读顺序为label→value(可选)→traits→hint.

需要适配的场景

由于系统控件是默认处理好的,而且VoiceOver的默认阅读顺序通常也没什么大问题,因此需要开发者专门去兼容的场景有如下

自定义 View 组件无障碍化

当子元素需要配置成 accessible, 而你的视图容器不需要配置成 accessible, 下面两个方法可以解决

  1. 直接设置 accessibilityElements 属性 (iOS8+), 使用在代码里搜下吧, 很简单.
  2. (很老的 API 了, 不推荐使用)视图需要实现 UIAccessibilityContainer.

只有背景图片无文字的控件

需要给出描述和对应控件的属性, 例如返回按钮/关闭按钮

self.backButton.accessibilityLabel = @"返回";
self.closeButton.accessibilityLabel = @"关闭";
self.closeButton.accessibilityTraits = UIAccessibilityTraitButton;

读取时间的无障碍

24 小时制的时刻系统无障碍会按字符一个个读出,需要转化成 12 小时制的写法,才会正常读取时间,例如 22:58 要写成 10.58PM.

某些控件的无障碍元素默认是关闭的, 需要开启

UIView/UIImageViewisAccessibilityElement 默认为 NO
而 UILabel/UIButton/UISwitch/UICollectionViewCell/UITableViewCell/UIPageControl 等组件默认为 YES.
其中 UILabel 比较离谱, 偶现 setText, 然后 po isAccessibilityElement 为 YES, 但是不响应的情况, 最好都显式设置 isAccessibilityElement = YES 吧.

父 View 如果为 AccessibilityElement, 子 element 将不响应 VoiceOver

self.bottomView.isAccessibilityElement = YES;
self.bottomView.accessibilityLabel = @"XXX";

比如一个View中有多个Label,那么每一个下面的Label单独访问可能意义不大,那么就可以将这个View设置成可以访问的,然后将其accessibilityLabel设置为所有子Label的 accessibilityLabel的合并值.

无障碍控件点击区域过小

类似下图:图片和文字搭配, 需要扩大无障碍点击范围:设置accessibilityFrame,通过CGRectUnion(frame1,frame2);
需要注意的是 accessibilityFrame 是相对屏幕的坐标系的, 使用 [UIView convertRect:toView:] 来转一次才能设置, 如果用了自动布局会很麻烦.

UI 组件实际功能不符合时

使用 button 只做点击作用的时候(不需要选中态),需要设置对应的控件属性 accessibilityTraits

// 1. 不需要播报"已选中"
button.accessibilityTraits &= ~UIAccessibilityTraitSelected;
// 2. 纯文本
button.accessibilityTraits = UIAccessibilityTraitStaticText;

使用 laber / view / imageView, 实现自定义响应事件时,设置对应控件属性 accessibilityTraits 为 UIAccessibilityTraitButton.

hidden 元素

有时把某个 view 设成 hidden 的时候, UI 上已经不展示了, 但是VoiceOver仍然可以读到.
此时可以使用

UIAccessibilityPostNotification(UIAccessibilityLayoutChangedNotification, nil)

强制更新VoiceOver的表现.

主动播报

toast 弹出或者刷新成功等场景需要主动播报, 如下:

UIAccessibilityPostNotification(UIAccessibilityAnnouncementNotification, voiceText);

弹窗

弹窗后屏蔽弹窗下面的元素

popupView.accessibilityViewIsModal = YES;//当前view才能响应无障碍播报
UIAccessibilityPostNotification(UIAccessibilityScreenChangedNotification, popupView.accessibilityElements.firstObject);

焦点乱跳

焦点乱跳有两个可能

  1. UICollectionView/UITableView reloadData 时候焦点会跳到最后一个 Cell, 解决办法为自行实现 cell 的更新, 减少直接 reloadData 调用; 如果这个 Cell 是满屏的, 可以设置当前 Cell 的 accessibilityViewIsModal 简单解决.
  2. 某个 UI 元素被加入到 accessibilityElements 但是又被标记为不可用 isAccessibilityElement = NO, 在手指左划时会出现焦点乱跳(但是右划正常)

UICollectionViewDelegate 异常

在无障碍开启时, UICollectionViewDelegate 部分回调异常, 表现为 Cell 提前预加载了, 不在这些方法中执行关键逻辑即可, 目前已知下面两个回调会有异常

- (void)collectionView:(UICollectionView *)collectionView willDisplayCell:(UICollectionViewCell *)cell forItemAtIndexPath:(NSIndexPath *)indexPath;

- (void)collectionView:(UICollectionView *)collectionView didEndDisplayingCell:(UICollectionViewCell *)cell forItemAtIndexPath:(NSIndexPath *)indexPath;

Cell 横划删除 在无障碍下无法使用

使用自定义操作来代替

UIAccessibilityCustomAction * action = [[UIAccessibilityCustomAction alloc] initWithName:@"测试" target:self selector:@selector(testToast)];
UIAccessibilityCustomAction * action1 = [[UIAccessibilityCustomAction alloc] initWithName:@"测试1" target:self selector:@selector(testToast1)];
UIAccessibilityCustomAction * action2 = [[UIAccessibilityCustomAction alloc] initWithName:@"测试2" target:self selector:@selector(testToast2)];

self.accessibilityCustomActions = @[action, action1, action2];

触发方法:

  1. 单击选中该 UI 元素
  2. 单指上下划, 第一次播报 "测试", 再单指上下划第二次播报 "测试2", 继续划会有第三次播报 "测试3", 第四次播报"激活" (初始状态).
  3. 然后在某一次播报后单指双击, 就可以触发对应的 selector

需要播报"已选中"

// 设置"已选中"播报
view.accessibilityTraits |= UIAccessibilityTraitSelected;
// 移除"已选中"播报
view.accessibilityTraits &= ~UIAccessibilityTraitSelected;
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容