1 /************************************************************** 2 * 3 * Licensed to the Apache Software Foundation (ASF) under one 4 * or more contributor license agreements. See the NOTICE file 5 * distributed with this work for additional information 6 * regarding copyright ownership. The ASF licenses this file 7 * to you under the Apache License, Version 2.0 (the 8 * "License"); you may not use this file except in compliance 9 * with the License. You may obtain a copy of the License at 10 * 11 * http://www.apache.org/licenses/LICENSE-2.0 12 * 13 * Unless required by applicable law or agreed to in writing, 14 * software distributed under the License is distributed on an 15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16 * KIND, either express or implied. See the License for the 17 * specific language governing permissions and limitations 18 * under the License. 19 * 20 *************************************************************/ 21 22 23 24 import java.util.Vector; 25 26 import com.sun.star.uno.AnyConverter; 27 import com.sun.star.uno.UnoRuntime; 28 import com.sun.star.uno.Type; 29 import com.sun.star.accessibility.*; 30 31 /** Handle all the events send from accessibility objects. The events 32 denoting new or removed top windows are handled as well. 33 34 It does not implement any listener interface as does the 35 EventListenerProxy class because it is interested only in a sub set of 36 the event types. 37 */ 38 public class EventHandler 39 { 40 public EventHandler () 41 { 42 mnTopWindowCount = 0; 43 maListenerProxy = new EventListenerProxy (this); 44 maConnectionTask = new ConnectionTask (maListenerProxy); 45 maObjectDisplays = new Vector (); 46 } 47 48 public synchronized void addObjectDisplay (IAccessibleObjectDisplay aDisplay) 49 { 50 maObjectDisplays.add (aDisplay); 51 } 52 53 54 public void finalize () 55 { 56 // When it is running then cancel the timer that tries to connect to 57 // the Office. 58 if (maConnectionTask != null) 59 maConnectionTask.cancel(); 60 } 61 62 63 64 public void disposing (com.sun.star.lang.EventObject aEvent) 65 { 66 // Ignored: We are not holding references to accessibility objects. 67 } 68 69 70 71 72 /** This method is called back when a new top level window has been opened. 73 */ 74 public void windowOpened (XAccessible xAccessible) 75 { 76 if (xAccessible != null) 77 { 78 // Update the counter of currently open top level windows 79 // observed by this object. 80 mnTopWindowCount += 1; 81 82 XAccessibleContext xContext = xAccessible.getAccessibleContext(); 83 if (xContext != null) 84 { 85 MessageArea.println ("new top level window has accessible name " 86 + xContext.getAccessibleName()); 87 88 // Register at all accessible objects of the new window. 89 new RegistrationThread ( 90 maListenerProxy, 91 xContext, 92 true, 93 true); 94 } 95 else 96 MessageArea.println ("new top level window is not accessible."); 97 } 98 else 99 MessageArea.println ("new top level window is not accessible."); 100 } 101 102 103 104 105 public void windowClosed (XAccessible xAccessible) 106 { 107 mnTopWindowCount -= 1; 108 MessageArea.println ("window closed, " + mnTopWindowCount + " still open"); 109 if (mnTopWindowCount == 0) 110 { 111 // This was the last window. Wait for a new connection. 112 MessageArea.println ("lost connection to office"); 113 new ConnectionTask (maListenerProxy); 114 } 115 if (xAccessible != null) 116 new RegistrationThread ( 117 maListenerProxy, 118 xAccessible.getAccessibleContext(), 119 false, 120 true); 121 } 122 123 124 125 126 /** Print a message that the given object just received the focus. Call 127 all accessible object diplays and tell them to update. 128 */ 129 private synchronized void focusGained (XAccessibleContext xContext) 130 { 131 if (xContext != null) 132 { 133 MessageArea.println ("focusGained: " + xContext.getAccessibleName() 134 + " with role " 135 + NameProvider.getRoleName (xContext.getAccessibleRole())); 136 137 // Tell the object displays to update their views. 138 for (int i=0; i<maObjectDisplays.size(); i++) 139 { 140 IAccessibleObjectDisplay aDisplay = 141 (IAccessibleObjectDisplay)maObjectDisplays.get(i); 142 if (aDisplay != null) 143 aDisplay.setAccessibleObject (xContext); 144 } 145 146 // Remember the currently focused object. 147 mxFocusedObject = xContext; 148 } 149 else 150 MessageArea.println ("focusGained: null"); 151 } 152 153 154 155 156 /** Print a message that the given object just lost the focus. Call 157 all accessible object diplays and tell them to update. 158 */ 159 private synchronized void focusLost (XAccessibleContext xContext) 160 { 161 if (xContext != null) 162 { 163 MessageArea.println ("focusLost: " 164 + xContext.getAccessibleName() 165 + " with role " 166 + NameProvider.getRoleName (xContext.getAccessibleRole())); 167 168 // Tell the object displays to update their views. 169 for (int i=0; i<maObjectDisplays.size(); i++) 170 { 171 IAccessibleObjectDisplay aDisplay = 172 (IAccessibleObjectDisplay)maObjectDisplays.get(i); 173 if (aDisplay != null) 174 aDisplay.setAccessibleObject (null); 175 } 176 mxFocusedObject = null; 177 } 178 else 179 MessageArea.println ("focusLost: null"); 180 } 181 182 183 184 185 /** Handle a change of the caret position. Ignore this on all objects 186 but the one currently focused. 187 */ 188 private void handleCaretEvent (XAccessibleContext xContext, 189 long nOldPosition, long nNewPosition) 190 { 191 if (xContext == mxFocusedObject) 192 MessageArea.println ("caret moved from " + nOldPosition + " to " + nNewPosition); 193 } 194 195 196 197 198 /** Print a message that a state has been changed. 199 @param xContext 200 The accessible context of the object whose state has changed. 201 @param nOldState 202 When not zero then this value describes a state that has been reset. 203 @param nNewValue 204 When not zero then this value describes a state that has been set. 205 */ 206 private void handleStateChange (XAccessibleContext xContext, short nOldState, short nNewState) 207 { 208 // Determine which state has changed and what is its new value. 209 short nState; 210 boolean aNewValue; 211 if (nOldState >= 0) 212 { 213 nState = nOldState; 214 aNewValue = false; 215 } 216 else 217 { 218 nState = nNewState; 219 aNewValue = true; 220 } 221 222 // Print a message about the changed state. 223 MessageArea.print ("setting state " + NameProvider.getStateName(nState) 224 + " to " + aNewValue); 225 if (xContext != null) 226 { 227 MessageArea.println (" at " + xContext.getAccessibleName() + " with role " 228 + NameProvider.getRoleName(xContext.getAccessibleRole())); 229 } 230 else 231 MessageArea.println (" at null"); 232 233 // Further handling of some states 234 switch (nState) 235 { 236 case AccessibleStateType.FOCUSED: 237 if (aNewValue) 238 focusGained (xContext); 239 else 240 focusLost (xContext); 241 } 242 } 243 244 245 246 247 /** Handle a child event that describes the creation of removal of a 248 single child. 249 */ 250 private void handleChildEvent ( 251 XAccessibleContext aOldChild, 252 XAccessibleContext aNewChild) 253 { 254 if (aOldChild != null) 255 // Remove event listener from the child and all of its descendants. 256 new RegistrationThread (maListenerProxy, aOldChild, false, false); 257 else if (aNewChild != null) 258 // Add event listener to the new child and all of its descendants. 259 new RegistrationThread (maListenerProxy, aNewChild, true, false); 260 } 261 262 263 264 265 /** Handle the change of some visible data of an object. 266 */ 267 private void handleVisibleDataEvent (XAccessibleContext xContext) 268 { 269 // The given object may affect the visible appearance of the focused 270 // object even when the two are not identical when the given object 271 // is an ancestor of the focused object. 272 // In order to not check this we simply call an update on the 273 // focused object. 274 if (mxFocusedObject != null) 275 for (int i=0; i<maObjectDisplays.size(); i++) 276 { 277 IAccessibleObjectDisplay aDisplay = 278 (IAccessibleObjectDisplay)maObjectDisplays.get(i); 279 if (aDisplay != null) 280 aDisplay.updateAccessibleObject (mxFocusedObject); 281 } 282 } 283 284 285 286 287 /** Print some information about an event that is not handled by any 288 more specialized handler. 289 */ 290 private void handleGenericEvent ( 291 int nEventId, 292 Object aSource, 293 Object aOldValue, 294 Object aNewValue) 295 { 296 // Print event to message area. 297 MessageArea.print ("received event " 298 + NameProvider.getEventName (nEventId) + " from "); 299 XAccessibleContext xContext = objectToContext (aSource); 300 if (xContext != null) 301 MessageArea.print (xContext.getAccessibleName()); 302 else 303 MessageArea.print ("null"); 304 MessageArea.println (" / " 305 + NameProvider.getRoleName(xContext.getAccessibleRole())); 306 } 307 308 309 310 /** This is the main method for handling accessibility events. It is 311 assumed that it is not called directly from the Office but from a 312 listener proxy that runs in a separate thread so that calls back to 313 the Office do not result in dead-locks. 314 */ 315 public void notifyEvent (com.sun.star.accessibility.AccessibleEventObject aEvent) 316 { 317 try // Guard against disposed objects. 318 { 319 switch (aEvent.EventId) 320 { 321 case AccessibleEventId.CHILD: 322 handleChildEvent ( 323 objectToContext (aEvent.OldValue), 324 objectToContext (aEvent.NewValue)); 325 break; 326 327 case AccessibleEventId.STATE_CHANGED: 328 { 329 short nOldState = -1; 330 short nNewState = -1; 331 try 332 { 333 if (AnyConverter.isShort (aEvent.NewValue)) 334 nNewState = AnyConverter.toShort (aEvent.NewValue); 335 if (AnyConverter.isShort (aEvent.OldValue)) 336 nOldState = AnyConverter.toShort (aEvent.OldValue); 337 } 338 catch (com.sun.star.lang.IllegalArgumentException e) 339 {} 340 handleStateChange ( 341 objectToContext (aEvent.Source), 342 nOldState, 343 nNewState); 344 } 345 break; 346 347 case AccessibleEventId.VISIBLE_DATA_CHANGED: 348 case AccessibleEventId.BOUNDRECT_CHANGED: 349 handleVisibleDataEvent (objectToContext (aEvent.Source)); 350 break; 351 352 case AccessibleEventId.CARET_CHANGED: 353 try 354 { 355 handleCaretEvent ( 356 objectToContext (aEvent.Source), 357 AnyConverter.toLong(aEvent.OldValue), 358 AnyConverter.toLong(aEvent.NewValue)); 359 } 360 catch (com.sun.star.lang.IllegalArgumentException e) 361 {} 362 break; 363 364 default: 365 handleGenericEvent (aEvent.EventId, 366 aEvent.Source, aEvent.OldValue, aEvent.NewValue); 367 break; 368 } 369 } 370 catch (com.sun.star.lang.DisposedException e) 371 {} 372 } 373 374 375 376 377 /** Convert the given object into an accessible context. The object is 378 interpreted as UNO Any and may contain either an XAccessible or 379 XAccessibleContext reference. 380 @return 381 The returned value is null when the given object can not be 382 converted to an XAccessibleContext reference. 383 */ 384 private XAccessibleContext objectToContext (Object aObject) 385 { 386 XAccessibleContext xContext = null; 387 XAccessible xAccessible = null; 388 try 389 { 390 xAccessible = (XAccessible)AnyConverter.toObject( 391 new Type(XAccessible.class), aObject); 392 } 393 catch (com.sun.star.lang.IllegalArgumentException e) 394 {} 395 if (xAccessible != null) 396 xContext = xAccessible.getAccessibleContext(); 397 else 398 try 399 { 400 xContext = (XAccessibleContext)AnyConverter.toObject( 401 new Type(XAccessibleContext.class), aObject); 402 } 403 catch (com.sun.star.lang.IllegalArgumentException e) 404 {} 405 return xContext; 406 } 407 408 409 410 411 /** The proxy that runs in a seperate thread and allows to call back to 412 the Office without running into dead-locks. 413 */ 414 private EventListenerProxy maListenerProxy; 415 416 /** The currently focused object. A value of null means that no object 417 has the focus. 418 */ 419 private XAccessibleContext mxFocusedObject; 420 421 /** Keep track of the currently open top windows to start a registration 422 loop when the last window (and the Office) is closed. 423 */ 424 private long mnTopWindowCount; 425 426 /** A list of objects that can display accessible objects in specific 427 ways such as showing a graphical representation or some textual 428 descriptions. 429 */ 430 private Vector maObjectDisplays; 431 432 /** The timer task that attempts in regular intervals to connect to a 433 running Office application. 434 */ 435 private ConnectionTask maConnectionTask; 436 } 437