gpt4 book ai didi

ios - 聚焦非 UILabel titleView 时,VoiceOver 会读取辅助功能标签两次

转载 作者:塔克拉玛干 更新时间:2023-11-02 20:28:10 26 4
gpt4 key购买 nike

我在使用 VoiceOver 时遇到了一个奇怪的问题。

目标:

  • 设置一个包含多个 UILabelUIStackView 作为我的 navigationItem.titleView
  • 将堆栈 View 标记为可访问性元素并将其 accessibilityLabel 设置为适当的值。
  • 通过在 viewDidAppear(animated:) 中调用 UIAccessibility.post(notification: .screenChanged, argument: navigationItem.titleView) 将堆栈 View 设置为初始 VoiceOver 焦点。

预期结果:

  • 当 View Controller 出现时,焦点似乎位于标题 View 上,并且 VoiceOver 会读取辅助功能标签的内容一次。

实际结果:

  • VoiceOver 开始朗读无障碍标签的内容,然后中途(或有时在读完之后)继续朗读第二遍。

如果我将 navigationItem.titleView 设置为 UILabel 的实例,则不会发生此问题。

有人知道为什么会这样吗?这是 iOS 中的错误吗?

我在这里建立了一个简单的项目来演示这个问题: https://github.com/rzulkoski/Focus-TitleView-Bug

最佳答案

你的标题有二次阅读的原因在你的代码中。

在您的 viewDidLoad 中,您设置了 VoiceOver 自动读出的堆栈 View 可访问性标签,以通知用户更改。

接下来,您通过 viewDidAppear 中的帖子通知此更改,VoiceOver 自然也会读出。

要防止这种行为,只需删除 setupNavigationItem 函数中的 stackView.accessibilityLabel = label.text 并将此代码段添加到您的私有(private)惰性变量标签 init 中:

if (self.view.subviews.contains(stackView)) {
stackView.accessibilityLabel = label.text
}

以这种方式更新 stackView.accessibilityLabel 不会触发 VoiceOver 来通知用户并允许实现您的目的。

但是,我不建议将标题读出作为新页面的第一个元素,除非你 reorder呈现的元素。

VoiceOver 用户自然不会猜到标题前有另一个元素:

  • 他们可能找不到返回上一页的方法。
  • 如果他们获得带有 4 fingers simple-tap 的页面的第一个元素,他们可能会丢失因为他们会得到后退按钮而不是标题。

从技术上讲,您的问题已通过上面的代码解决,但从概念上讲,如果您仍想将标题显示为第一个元素,我建议您对元素重新排序。

==========

编辑(解决方法)

关于技术问题,您的评论是正确的,由于 VoiceOver 的标签阅读,上述解决方案有效。

我在您在初始帖子中提供的 git 分支中提交了一个解决方案。

问题涉及 UIStackView,在这种情况下我无法解释,也无法按原样解决。

为了达到您的目的,我为 stackview 创建了一个 UIAccessibilityELement,它可以完美地访问和公开,无需通过后通知进行双重读取。

我这样做是因为当标签位于时我无法以编程方式获得 stackview 的新大小...也许创建一个 UIStackView 子类并进入其 layoutSubviews 可能是诀窍?

此解决方案应该作为变通方法但我不知道此行为出现在 UIStackview 中的原因。

==========

编辑(解决方案)

问题在于 navigationItemtitleView 的创建方式。实现目标的最佳方式是:

  • 将您的 titleView 初始化为一个简单的 UIView,其框架与 stackview 的框架相同。
  • 在指定其框架及其可访问性属性后,将堆栈 View 添加为 subview 。

在您的代码中执行以下步骤:

  • 在 stackview 属性中添加 .header 特征:

    private lazy var stackView: UIStackView = {
    let stackView = UIStackView(frame: .zero)
    stackView.axis = .vertical
    stackView.alignment = .center
    stackView.distribution = .equalSpacing
    stackView.isAccessibilityElement = true
    stackView.accessibilityTraits = .header
    return stackView
    }()
  • 如下更改“switch...case...”代码部分中的 stackview case:

    case .stackView:
    label.text = "UIStackView"
    label.sizeToFit()
    stackView.addArrangedSubview(label)

    label2.text = subtitle
    label2.sizeToFit()
    stackView.addArrangedSubview(label2)

    stackView.frame.size.width = max(label.frame.width, label2.frame.width)
    stackView.frame.size.height = label.frame.height + label2.frame.height

    stackView.accessibilityLabel = label.text?.appending(", \(label2.text!)")

    navigationItem.titleView = UIView(frame: stackView.frame)
    navigationItem.titleView?.addSubview(stackView)
    }

现在,postNotification 只读出一次堆栈 View 作为屏幕的第一个元素。

关于ios - 聚焦非 UILabel titleView 时,VoiceOver 会读取辅助功能标签两次,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54954861/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com