OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2015 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 /** | |
6 * @fileoverview Implements support for live regions in ChromeVox Next. | |
7 */ | |
8 | |
9 goog.provide('LiveRegions'); | |
10 | |
11 goog.require('ChromeVoxState'); | |
12 | |
13 /** | |
14 * ChromeVox2 live region handler. | |
15 * @param {ChromeVoxState} chromeVoxState The ChromeVox state object, | |
16 * keeping track of the current mode and current range. | |
17 * @constructor | |
18 */ | |
19 LiveRegions = function(chromeVoxState) { | |
20 /** | |
21 * @type {ChromeVoxState} | |
22 * @private | |
23 */ | |
24 this.chromeVoxState_ = chromeVoxState; | |
25 | |
26 /** | |
27 * The time the last live region event was output. | |
28 * @type {Date} | |
29 * @private | |
30 */ | |
31 this.lastLiveRegionTime_ = new Date(0); | |
32 | |
33 /** | |
34 * Set of nodes that have been announced as part of a live region since | |
35 * |this.lastLiveRegionTime_|, to prevent duplicate announcements. | |
36 * @type {WeakSet<chrome.automation.AutomationNode>} | |
37 * @private | |
38 */ | |
39 this.liveRegionNodeSet_ = new WeakSet(); | |
40 | |
41 chrome.automation.setTreeChangeObserverMask('liveRegionTreeChanges'); | |
42 chrome.automation.addTreeChangeObserver(this.onTreeChange.bind(this)); | |
43 }; | |
44 | |
45 /** | |
46 * Live region events received in fewer than this many milliseconds will | |
47 * queue, otherwise they'll be output with a category flush. | |
48 * @type {number} | |
49 * @const | |
50 */ | |
51 LiveRegions.LIVE_REGION_QUEUE_TIME_MS = 500; | |
52 | |
53 LiveRegions.prototype = { | |
54 /** | |
55 * Called when the automation tree is changed. | |
56 * @param {chrome.automation.TreeChange} treeChange | |
Peter Lundblad
2015/11/24 11:04:31
Suggest using goog.scope like in some other files
dmazzoni
2015/11/30 22:00:47
Done.
| |
57 */ | |
58 onTreeChange: function(treeChange) { | |
59 var node = treeChange.target; | |
60 if (!node.containerLiveStatus) | |
61 return; | |
62 | |
63 var mode = this.chromeVoxState_.getMode(); | |
64 var currentRange = this.chromeVoxState_.getCurrentRange(); | |
65 | |
66 if (mode === ChromeVoxMode.CLASSIC || !cvox.ChromeVox.isActive) | |
67 return; | |
68 | |
69 if (!currentRange) | |
70 return; | |
71 | |
72 if (!AutomationUtil.isInSameWebpage(node, currentRange.start.node)) | |
Peter Lundblad
2015/11/24 11:04:31
Doesn't classic announce live region updates from
dmazzoni
2015/11/30 22:00:47
Good catch.
I switched it but made it an option s
| |
73 return; | |
74 | |
75 var type = treeChange.type; | |
76 var relevant = node.containerLiveRelevant; | |
77 if (relevant.indexOf('additions') >= 0 && | |
78 (type == 'nodeCreated' || type == 'subtreeCreated')) { | |
79 this.outputLiveRegionChange_(node, null); | |
80 } | |
81 | |
82 if (relevant.indexOf('text') >= 0 && type == 'nodeChanged') | |
83 this.outputLiveRegionChange_(node, null); | |
84 | |
85 if (relevant.indexOf('removals') >= 0 && type == 'nodeRemoved') | |
86 this.outputLiveRegionChange_(node, '@live_regions_removed'); | |
87 }, | |
88 | |
89 /** | |
90 * Given a node that needs to be spoken as part of a live region | |
91 * change and an additional optional format string, output the | |
92 * live region description. | |
93 * @param {!chrome.automation.AutomationNode} node The changed node. | |
94 * @param {?string} opt_prependFormatStr If set, a format string for | |
Peter Lundblad
2015/11/24 11:04:31
Optional parameters should have an = suffix.
dmazzoni
2015/11/30 22:00:47
Done.
| |
95 * cvox2.Output to prepend to the output. | |
96 * @private | |
97 */ | |
98 outputLiveRegionChange_: function(node, opt_prependFormatStr) { | |
99 if (node.containerLiveBusy) | |
100 return; | |
101 | |
102 if (node.containerLiveAtomic && !node.liveAtomic) { | |
103 if (node.parent) | |
104 this.outputLiveRegionChange_(node.parent, opt_prependFormatStr); | |
105 return; | |
106 } | |
107 | |
108 var range = cursors.Range.fromNode(node); | |
109 var output = new Output(); | |
110 if (opt_prependFormatStr) { | |
Peter Lundblad
2015/11/24 11:04:31
nit: bracket inconsistency.
dmazzoni
2015/11/30 22:00:47
Done.
| |
111 output.format(opt_prependFormatStr); | |
112 } | |
113 output.withSpeech(range, range, Output.EventType.NAVIGATE); | |
114 | |
115 if (!output.hasSpeech && node.liveAtomic) { | |
Peter Lundblad
2015/11/24 11:04:31
Ditto.
dmazzoni
2015/11/30 22:00:47
Done.
| |
116 output.format('$descendants', node); | |
117 } | |
118 | |
119 output.withSpeechCategory(cvox.TtsCategory.LIVE); | |
120 | |
121 if (!output.hasSpeech) | |
122 return; | |
123 | |
124 // Enqueue live region updates that were received at approximately | |
125 // the same time, otherwise flush previous live region updates. | |
126 var currentTime = new Date(); | |
127 var queueTime = LiveRegions.LIVE_REGION_QUEUE_TIME_MS; | |
128 if (currentTime - this.lastLiveRegionTime_ > queueTime) { | |
129 this.liveRegionNodeSet_ = new WeakSet(); | |
130 output.withQueueMode(cvox.QueueMode.CATEGORY_FLUSH); | |
131 this.lastLiveRegionTime_ = currentTime; | |
132 } else { | |
133 output.withQueueMode(cvox.QueueMode.QUEUE); | |
134 } | |
135 | |
136 var parent = node; | |
137 while (parent) { | |
138 if (this.liveRegionNodeSet_.has(parent)) | |
139 return; | |
140 parent = parent.parent; | |
141 } | |
142 | |
143 this.liveRegionNodeSet_.add(node); | |
144 output.go(); | |
145 }, | |
146 }; | |
OLD | NEW |