qtestaccessible.h 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. /****************************************************************************
  2. **
  3. ** Copyright (C) 2016 The Qt Company Ltd.
  4. ** Contact: https://www.qt.io/licensing/
  5. **
  6. ** This file is part of the QtTest module of the Qt Toolkit.
  7. **
  8. ** $QT_BEGIN_LICENSE:LGPL$
  9. ** Commercial License Usage
  10. ** Licensees holding valid commercial Qt licenses may use this file in
  11. ** accordance with the commercial license agreement provided with the
  12. ** Software or, alternatively, in accordance with the terms contained in
  13. ** a written agreement between you and The Qt Company. For licensing terms
  14. ** and conditions see https://www.qt.io/terms-conditions. For further
  15. ** information use the contact form at https://www.qt.io/contact-us.
  16. **
  17. ** GNU Lesser General Public License Usage
  18. ** Alternatively, this file may be used under the terms of the GNU Lesser
  19. ** General Public License version 3 as published by the Free Software
  20. ** Foundation and appearing in the file LICENSE.LGPL3 included in the
  21. ** packaging of this file. Please review the following information to
  22. ** ensure the GNU Lesser General Public License version 3 requirements
  23. ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
  24. **
  25. ** GNU General Public License Usage
  26. ** Alternatively, this file may be used under the terms of the GNU
  27. ** General Public License version 2.0 or (at your option) the GNU General
  28. ** Public license version 3 or any later version approved by the KDE Free
  29. ** Qt Foundation. The licenses are as published by the Free Software
  30. ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
  31. ** included in the packaging of this file. Please review the following
  32. ** information to ensure the GNU General Public License requirements will
  33. ** be met: https://www.gnu.org/licenses/gpl-2.0.html and
  34. ** https://www.gnu.org/licenses/gpl-3.0.html.
  35. **
  36. ** $QT_END_LICENSE$
  37. **
  38. ****************************************************************************/
  39. #ifndef QTESTACCESSIBLE_H
  40. #define QTESTACCESSIBLE_H
  41. #if 0
  42. // inform syncqt
  43. #pragma qt_no_master_include
  44. #endif
  45. #include <QtCore/qglobal.h>
  46. #ifndef QT_NO_ACCESSIBILITY
  47. #define QVERIFY_EVENT(event) \
  48. QVERIFY(QTestAccessibility::verifyEvent(event))
  49. #include <QtCore/qlist.h>
  50. #include <QtCore/qdebug.h>
  51. #include <QtGui/qaccessible.h>
  52. #include <QtGui/qguiapplication.h>
  53. #include <QtTest/qtest_global.h>
  54. #include <QtTest/qtestsystem.h>
  55. QT_BEGIN_NAMESPACE
  56. class QObject;
  57. // Use pointers since we subclass QAccessibleEvent
  58. typedef QList<QAccessibleEvent*> EventList;
  59. bool operator==(const QAccessibleEvent &l, const QAccessibleEvent &r)
  60. {
  61. if (l.type() != r.type()) {
  62. // qDebug() << "QAccessibleEvent with wrong type: " << qAccessibleEventString(l.type()) << " and " << qAccessibleEventString(r.type());
  63. return false;
  64. }
  65. if (l.object() != r.object() ||
  66. l.child() != r.child()) {
  67. // qDebug() << "QAccessibleEvent for wrong object: " << l.object() << " and " << r.object() << " child: " << l.child() << " and " << r.child();
  68. return false;
  69. }
  70. if (l.type() == QAccessible::StateChanged) {
  71. return static_cast<const QAccessibleStateChangeEvent*>(&l)->changedStates()
  72. == static_cast<const QAccessibleStateChangeEvent*>(&r)->changedStates();
  73. } else if (l.type() == QAccessible::TextCaretMoved) {
  74. return static_cast<const QAccessibleTextCursorEvent*>(&l)->cursorPosition()
  75. == static_cast<const QAccessibleTextCursorEvent*>(&r)->cursorPosition();
  76. } else if (l.type() == QAccessible::TextSelectionChanged) {
  77. const QAccessibleTextSelectionEvent *le = static_cast<const QAccessibleTextSelectionEvent*>(&l);
  78. const QAccessibleTextSelectionEvent *re = static_cast<const QAccessibleTextSelectionEvent*>(&r);
  79. return le->cursorPosition() == re->cursorPosition() &&
  80. le->selectionStart() == re->selectionStart() &&
  81. le->selectionEnd() == re->selectionEnd();
  82. } else if (l.type() == QAccessible::TextInserted) {
  83. const QAccessibleTextInsertEvent *le = static_cast<const QAccessibleTextInsertEvent*>(&l);
  84. const QAccessibleTextInsertEvent *re = static_cast<const QAccessibleTextInsertEvent*>(&r);
  85. return le->cursorPosition() == re->cursorPosition() &&
  86. le->changePosition() == re->changePosition() &&
  87. le->textInserted() == re->textInserted();
  88. } else if (l.type() == QAccessible::TextRemoved) {
  89. const QAccessibleTextRemoveEvent *le = static_cast<const QAccessibleTextRemoveEvent*>(&l);
  90. const QAccessibleTextRemoveEvent *re = static_cast<const QAccessibleTextRemoveEvent*>(&r);
  91. return le->cursorPosition() == re->cursorPosition() &&
  92. le->changePosition() == re->changePosition() &&
  93. le->textRemoved() == re->textRemoved();
  94. } else if (l.type() == QAccessible::TextUpdated) {
  95. const QAccessibleTextUpdateEvent *le = static_cast<const QAccessibleTextUpdateEvent*>(&l);
  96. const QAccessibleTextUpdateEvent *re = static_cast<const QAccessibleTextUpdateEvent*>(&r);
  97. return le->cursorPosition() == re->cursorPosition() &&
  98. le->changePosition() == re->changePosition() &&
  99. le->textInserted() == re->textInserted() &&
  100. le->textRemoved() == re->textRemoved();
  101. } else if (l.type() == QAccessible::ValueChanged) {
  102. const QAccessibleValueChangeEvent *le = static_cast<const QAccessibleValueChangeEvent*>(&l);
  103. const QAccessibleValueChangeEvent *re = static_cast<const QAccessibleValueChangeEvent*>(&r);
  104. return le->value() == re->value();
  105. }
  106. return true;
  107. }
  108. class QTestAccessibility
  109. {
  110. public:
  111. static void initialize()
  112. {
  113. if (!instance()) {
  114. instance() = new QTestAccessibility;
  115. qAddPostRoutine(cleanup);
  116. }
  117. }
  118. static void cleanup()
  119. {
  120. delete instance();
  121. instance() = Q_NULLPTR;
  122. }
  123. static void clearEvents() { eventList().clear(); }
  124. static EventList events() { return eventList(); }
  125. static bool verifyEvent(QAccessibleEvent *ev)
  126. {
  127. for (int i = 0; eventList().isEmpty() && i < 5; ++i)
  128. QTest::qWait(50);
  129. if (eventList().isEmpty()) {
  130. qWarning("Timeout waiting for accessibility event.");
  131. return false;
  132. }
  133. const bool res = *eventList().first() == *ev;
  134. if (!res)
  135. qWarning("%s", qPrintable(msgAccessibilityEventListMismatch(eventList(), ev)));
  136. delete eventList().takeFirst();
  137. return res;
  138. }
  139. static bool containsEvent(QAccessibleEvent *event) {
  140. for (const QAccessibleEvent *ev : qAsConst(eventList())) {
  141. if (*ev == *event)
  142. return true;
  143. }
  144. return false;
  145. }
  146. private:
  147. QTestAccessibility()
  148. {
  149. QAccessible::installUpdateHandler(updateHandler);
  150. QAccessible::installRootObjectHandler(rootObjectHandler);
  151. }
  152. ~QTestAccessibility()
  153. {
  154. QAccessible::installUpdateHandler(Q_NULLPTR);
  155. QAccessible::installRootObjectHandler(Q_NULLPTR);
  156. }
  157. static void rootObjectHandler(QObject *object)
  158. {
  159. // qDebug("rootObjectHandler called %p", object);
  160. if (object) {
  161. QGuiApplication* app = qobject_cast<QGuiApplication*>(object);
  162. if ( !app )
  163. qWarning("root Object is not a QGuiApplication!");
  164. } else {
  165. qWarning("root Object called with 0 pointer");
  166. }
  167. }
  168. static void updateHandler(QAccessibleEvent *event)
  169. {
  170. eventList().append(copyEvent(event));
  171. }
  172. static QAccessibleEvent *copyEvent(QAccessibleEvent *event)
  173. {
  174. QAccessibleEvent *ev;
  175. if (event->type() == QAccessible::StateChanged) {
  176. if (event->object())
  177. ev = new QAccessibleStateChangeEvent(event->object(),
  178. static_cast<QAccessibleStateChangeEvent*>(event)->changedStates());
  179. else
  180. ev = new QAccessibleStateChangeEvent(event->accessibleInterface(),
  181. static_cast<QAccessibleStateChangeEvent*>(event)->changedStates());
  182. } else if (event->type() == QAccessible::TextCaretMoved) {
  183. if (event->object())
  184. ev = new QAccessibleTextCursorEvent(event->object(), static_cast<QAccessibleTextCursorEvent*>(event)->cursorPosition());
  185. else
  186. ev = new QAccessibleTextCursorEvent(event->accessibleInterface(), static_cast<QAccessibleTextCursorEvent*>(event)->cursorPosition());
  187. } else if (event->type() == QAccessible::TextSelectionChanged) {
  188. const QAccessibleTextSelectionEvent *original = static_cast<QAccessibleTextSelectionEvent*>(event);
  189. QAccessibleTextSelectionEvent *sel;
  190. if (event->object())
  191. sel = new QAccessibleTextSelectionEvent(event->object(), original->selectionStart(), original->selectionEnd());
  192. else
  193. sel = new QAccessibleTextSelectionEvent(event->accessibleInterface(), original->selectionStart(), original->selectionEnd());
  194. sel->setCursorPosition(original->cursorPosition());
  195. ev = sel;
  196. } else if (event->type() == QAccessible::TextInserted) {
  197. const QAccessibleTextInsertEvent *original = static_cast<QAccessibleTextInsertEvent*>(event);
  198. QAccessibleTextInsertEvent *ins;
  199. if (original->object())
  200. ins = new QAccessibleTextInsertEvent(event->object(), original->changePosition(), original->textInserted());
  201. else
  202. ins = new QAccessibleTextInsertEvent(event->accessibleInterface(), original->changePosition(), original->textInserted());
  203. ins->setCursorPosition(original->cursorPosition());
  204. ev = ins;
  205. } else if (event->type() == QAccessible::TextRemoved) {
  206. const QAccessibleTextRemoveEvent *original = static_cast<QAccessibleTextRemoveEvent*>(event);
  207. QAccessibleTextRemoveEvent *rem;
  208. if (event->object())
  209. rem = new QAccessibleTextRemoveEvent(event->object(), original->changePosition(), original->textRemoved());
  210. else
  211. rem = new QAccessibleTextRemoveEvent(event->accessibleInterface(), original->changePosition(), original->textRemoved());
  212. rem->setCursorPosition(original->cursorPosition());
  213. ev = rem;
  214. } else if (event->type() == QAccessible::TextUpdated) {
  215. const QAccessibleTextUpdateEvent *original = static_cast<QAccessibleTextUpdateEvent*>(event);
  216. QAccessibleTextUpdateEvent *upd;
  217. if (event->object())
  218. upd = new QAccessibleTextUpdateEvent(event->object(), original->changePosition(), original->textRemoved(), original->textInserted());
  219. else
  220. upd = new QAccessibleTextUpdateEvent(event->accessibleInterface(), original->changePosition(), original->textRemoved(), original->textInserted());
  221. upd->setCursorPosition(original->cursorPosition());
  222. ev = upd;
  223. } else if (event->type() == QAccessible::ValueChanged) {
  224. if (event->object())
  225. ev = new QAccessibleValueChangeEvent(event->object(), static_cast<QAccessibleValueChangeEvent*>(event)->value());
  226. else
  227. ev = new QAccessibleValueChangeEvent(event->accessibleInterface(), static_cast<QAccessibleValueChangeEvent*>(event)->value());
  228. } else if (event->type() == QAccessible::TableModelChanged) {
  229. QAccessibleTableModelChangeEvent *oldEvent = static_cast<QAccessibleTableModelChangeEvent*>(event);
  230. QAccessibleTableModelChangeEvent *newEvent;
  231. if (event->object())
  232. newEvent = new QAccessibleTableModelChangeEvent(event->object(), oldEvent->modelChangeType());
  233. else
  234. newEvent = new QAccessibleTableModelChangeEvent(event->accessibleInterface(), oldEvent->modelChangeType());
  235. newEvent->setFirstRow(oldEvent->firstRow());
  236. newEvent->setFirstColumn(oldEvent->firstColumn());
  237. newEvent->setLastRow(oldEvent->lastRow());
  238. newEvent->setLastColumn(oldEvent->lastColumn());
  239. ev = newEvent;
  240. } else {
  241. if (event->object())
  242. ev = new QAccessibleEvent(event->object(), event->type());
  243. else
  244. ev = new QAccessibleEvent(event->accessibleInterface(), event->type());
  245. }
  246. ev->setChild(event->child());
  247. return ev;
  248. }
  249. static EventList &eventList()
  250. {
  251. static EventList list;
  252. return list;
  253. }
  254. static QTestAccessibility *&instance()
  255. {
  256. static QTestAccessibility *ta = Q_NULLPTR;
  257. return ta;
  258. }
  259. private:
  260. static QString msgAccessibilityEventListMismatch(const EventList &haystack,
  261. const QAccessibleEvent *needle)
  262. {
  263. QString rc;
  264. QDebug str = QDebug(&rc).nospace();
  265. str << "Event " << *needle
  266. << " not found at head of event list of size " << haystack.size() << " :";
  267. for (const QAccessibleEvent *e : haystack)
  268. str << ' ' << *e;
  269. return rc;
  270. }
  271. };
  272. QT_END_NAMESPACE
  273. #endif // QT_NO_ACCESSIBILITY
  274. #endif // QTESTACCESSIBLE_H