NVDA屏幕阅读器请求错误的QAccessible::InterfaceType

NVDA screen reader requests wrong QAccessible::InterfaceType

本文关键字:QAccessible 错误 InterfaceType 请求 屏幕 NVDA      更新时间:2023-10-16

我在Qt 5.9.1上。

我正在尝试通过屏幕阅读器访问应用程序。我有一个自定义小部件(我们称之为MyWidget),其中包含文本(很多)。文本是使用QPainter绘制的,这就是为什么使用自定义小部件而不是像QTextBrowser这样的东西。

我在派生自QAccessibleWidgetQAccessibleTextInterfaceAccessibleMyWidget类中实现了小部件的QAccessibleTextInterface。它在Linux下的Orca上运行良好,但是当在带有NVDA的Windows 7中使用时,QAccessibleInterface::interface_cast()请求错误的接口类型。对于逆戟鲸,我收到了QAccessible::TextInterface的请求。在NVDA中,它总是QAccessible::ValueInterface

AccessibleMyWidget定义为:

class AccessibleMyWidget:
public QAccessibleWidget, public QAccessibleTextInterface {
public:
explicit AccessibleMyWidget(QWidget* w)
: QAccessibleWidget(w, QAccessible::EditableText)
{
Q_ASSERT(isValid());
}
void* interface_cast(QAccessible::InterfaceType t) override
{
if (t == QAccessible::TextInterface) {
// !!! This is never requested with NVDA !!!
return static_cast<QAccessibleTextInterface*>(this);
}
return QAccessibleWidget::interface_cast(t);
}
/*
* QAccessibleTextInterface implementation below this point.
*/
void addSelection(int startOffset, int endOffset) override;
QString attributes(int offset, int* startOffset,
int* endOffset) const override;
// etc.
};

在Linux下使用Orca,一切似乎都按预期工作。我接到interface_cast()TextInterface的调用,之后调用QAccessibleTextInterface的各种函数。使用Linux下的NVDA,我只收到interface_cast()ValueInterface调用,并且没有调用任何QAccessibleTextInterface函数。这意味着MyWidget是完全无法访问的,除非我覆盖QAccessibleWidget::text()并将所有文本作为单个字符串返回,这意味着没有光标导航,没有选择支持......在这一点上,它基本上变成了一个QLabel,但有大量的文本,因此很难使用。

我在这里错过了什么?

好吧,NVDA开发人员帮助我追踪了这个问题。这是因为MinGW版本的Qt禁用了IAccessible2,并使用MSAA(Microsoft Active Accessibility),它不支持任何这些接口。IA2需要COM,而MinGW不支持。

所以我必须从GCC/MinGW切换到MSVC才能在Windows上完成这项工作。太糟糕了,在可预见的未来,这实际上不是一种选择:-/

2020 更新:
此后已修复此问题。不确定确切的时间,但当前的 mingw Qt 版本现在支持这一点。