Version 5.4.3.2, tag libreoffice-5.4.3.2
[LibreOffice.git] / vcl / osx / a11ywrapper.mm
blob15123c826854e34dbe668f99ec07c265ae947e13
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3  * This file is part of the LibreOffice project.
4  *
5  * This Source Code Form is subject to the terms of the Mozilla Public
6  * License, v. 2.0. If a copy of the MPL was not distributed with this
7  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8  *
9  * This file incorporates work covered by the following license notice:
10  *
11  *   Licensed to the Apache Software Foundation (ASF) under one or more
12  *   contributor license agreements. See the NOTICE file distributed
13  *   with this work for additional information regarding copyright
14  *   ownership. The ASF licenses this file to you under the Apache
15  *   License, Version 2.0 (the "License"); you may not use this file
16  *   except in compliance with the License. You may obtain a copy of
17  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
18  */
21 #include "osx/salinst.h"
22 #include "osx/saldata.hxx"
24 #include "osx/a11ywrapper.h"
25 #include "osx/a11ylistener.hxx"
26 #include "osx/a11yfactory.h"
27 #include "osx/a11yfocustracker.hxx"
29 #include "quartz/utils.h"
31 #include "a11yfocuslistener.hxx"
32 #include "a11yactionwrapper.h"
33 #include "a11ycomponentwrapper.h"
34 #include "a11yselectionwrapper.h"
35 #include "a11ytablewrapper.h"
36 #include "a11ytextwrapper.h"
37 #include "a11yvaluewrapper.h"
38 #include "a11yrolehelper.h"
40 #include <com/sun/star/accessibility/AccessibleRole.hpp>
41 #include <com/sun/star/accessibility/AccessibleStateType.hpp>
42 #include <com/sun/star/accessibility/XAccessibleEventBroadcaster.hpp>
43 #include <com/sun/star/awt/Size.hpp>
44 #include <com/sun/star/accessibility/XAccessibleRelationSet.hpp>
45 #include <com/sun/star/accessibility/AccessibleRelationType.hpp>
46 #include <com/sun/star/lang/DisposedException.hpp>
48 using namespace ::com::sun::star::accessibility;
49 using namespace ::com::sun::star::awt;
50 using namespace ::com::sun::star::lang;
51 using namespace ::com::sun::star::uno;
53 @interface SalFrameWindow : NSWindow
56 -(Reference<XAccessibleContext>)accessibleContext;
57 @end
59 static BOOL isPopupMenuOpen = NO;
61 static std::ostream &operator<<(std::ostream &s, NSObject *obj) {
62     return s << [[obj description] UTF8String];
65 @implementation AquaA11yWrapper : NSView
67 #pragma mark -
68 #pragma mark Init and dealloc
70 -(id)initWithAccessibleContext: (Reference < XAccessibleContext >) rxAccessibleContext {
71     self = [ super init ];
72     if ( self ) {
73         [ self setDefaults: rxAccessibleContext ];
74     }
75     return self;
78 -(void) setDefaults: (Reference < XAccessibleContext >) rxAccessibleContext {
79     mpReferenceWrapper = new ReferenceWrapper;
80     mActsAsRadioGroup = NO;
81     mpReferenceWrapper -> rAccessibleContext = rxAccessibleContext;
82     mIsTableCell = NO;
83     // Querying all supported interfaces
84     try {
85         // XAccessibleComponent
86         mpReferenceWrapper->rAccessibleComponent.set( rxAccessibleContext, UNO_QUERY );
87         // XAccessibleExtendedComponent
88         mpReferenceWrapper->rAccessibleExtendedComponent.set( rxAccessibleContext, UNO_QUERY );
89         // XAccessibleSelection
90         mpReferenceWrapper->rAccessibleSelection.set( rxAccessibleContext, UNO_QUERY );
91         // XAccessibleTable
92         mpReferenceWrapper->rAccessibleTable.set( rxAccessibleContext, UNO_QUERY );
93         // XAccessibleText
94         mpReferenceWrapper->rAccessibleText.set( rxAccessibleContext, UNO_QUERY );
95         // XAccessibleEditableText
96         mpReferenceWrapper->rAccessibleEditableText.set( rxAccessibleContext, UNO_QUERY );
97         // XAccessibleValue
98         mpReferenceWrapper->rAccessibleValue.set( rxAccessibleContext, UNO_QUERY );
99         // XAccessibleAction
100         mpReferenceWrapper->rAccessibleAction.set( rxAccessibleContext, UNO_QUERY );
101         // XAccessibleTextAttributes
102         mpReferenceWrapper->rAccessibleTextAttributes.set( rxAccessibleContext, UNO_QUERY );
103         // XAccessibleMultiLineText
104         mpReferenceWrapper->rAccessibleMultiLineText.set( rxAccessibleContext, UNO_QUERY );
105         // XAccessibleTextMarkup
106         mpReferenceWrapper->rAccessibleTextMarkup.set( rxAccessibleContext, UNO_QUERY );
107         // XAccessibleEventBroadcaster
108         #if 0
109         /* #i102033# NSAccessibility does not seemt to know an equivalent for transient children.
110            That means we need to cache this, else e.g. tree list boxes are not accessible (moreover
111            it crashes by notifying dead objects - which would seemt o be another bug)
113            FIXME:
114            Unfortunately this can increase memory consumption drastically until the non transient parent
115            is destroyed an finally all the transients are released.
116         */
117         if ( ! rxAccessibleContext -> getAccessibleStateSet() -> contains ( AccessibleStateType::TRANSIENT ) )
118         #endif
119         {
120             Reference< XAccessibleEventBroadcaster > xBroadcaster(rxAccessibleContext, UNO_QUERY);
121             if( xBroadcaster.is() ) {
122                 /*
123                  * We intentionally do not hold a reference to the event listener in the wrapper object,
124                  * but let the listener control the life cycle of the wrapper instead ..
125                  */
126                 xBroadcaster->addAccessibleEventListener( new AquaA11yEventListener( self, rxAccessibleContext -> getAccessibleRole() ) );
127             }
128         }
129         // TABLE_CELL
130         if ( rxAccessibleContext -> getAccessibleRole() == AccessibleRole::TABLE_CELL ) {
131             mIsTableCell = YES;
132         }
133     } catch ( const Exception ) {
134     }
137 -(void)dealloc {
138     if ( mpReferenceWrapper ) {
139         delete mpReferenceWrapper;
140     }
141     [ super dealloc ];
144 #pragma mark -
145 #pragma mark Utility Section
147 // generates selectors for attribute name AXAttributeNameHere
148 // (getter without parameter) attributeNameHereAttribute
149 // (getter with parameter)    attributeNameHereAttributeForParameter:
150 // (setter)                   setAttributeNameHereAttributeForElement:to:
151 -(SEL)selectorForAttribute:(NSString *)attribute asGetter:(BOOL)asGetter withGetterParameter:(BOOL)withGetterParameter {
152     SEL selector = static_cast<SEL>(nil);
153     NSAutoreleasePool * pool = [ [ NSAutoreleasePool alloc ] init ];
154     @try {
155         // step 1: create method name from attribute name
156         NSMutableString * methodName = [ NSMutableString string ];
157         if ( ! asGetter ) {
158             [ methodName appendString: @"set" ];
159         }
160         NSRange aRange = { 2, 1 };
161         NSString * firstChar = [ attribute substringWithRange: aRange ]; // drop leading "AX" and get first char
162         if ( asGetter ) {
163             [ methodName appendString: [ firstChar lowercaseString ] ];
164         } else {
165             [ methodName appendString: firstChar ];
166         }
167         [ methodName appendString: [ attribute substringFromIndex: 3 ] ]; // append rest of attribute name
168         // append rest of method name
169         [ methodName appendString: @"Attribute" ];
170         if ( ! asGetter ) {
171             [ methodName appendString: @"ForElement:to:" ];
172         } else if ( asGetter && withGetterParameter ) {
173             [ methodName appendString: @"ForParameter:" ];
174         }
175         // step 2: create selector
176         selector = NSSelectorFromString ( methodName );
177     } @catch ( id exception ) {
178         selector = static_cast<SEL>(nil);
179     }
180     [ pool release ];
181     return selector;
184 -(Reference < XAccessible >)getFirstRadioButtonInGroup {
185     Reference < XAccessibleRelationSet > rxAccessibleRelationSet = [ self accessibleContext ] -> getAccessibleRelationSet();
186     if( rxAccessibleRelationSet.is() )
187     {
188         AccessibleRelation relationMemberOf = rxAccessibleRelationSet -> getRelationByType ( AccessibleRelationType::MEMBER_OF );
189         if ( relationMemberOf.RelationType == AccessibleRelationType::MEMBER_OF && relationMemberOf.TargetSet.hasElements() )
190             return Reference < XAccessible > ( relationMemberOf.TargetSet[0], UNO_QUERY );
191     }
192     return Reference < XAccessible > ();
195 -(BOOL)isFirstRadioButtonInGroup {
196     Reference < XAccessible > rFirstMateAccessible = [ self getFirstRadioButtonInGroup ];
197     if ( rFirstMateAccessible.is() && rFirstMateAccessible -> getAccessibleContext().get() == [ self accessibleContext ] ) {
198         return YES;
199     }
200     return NO;
203 #pragma mark -
204 #pragma mark Attribute Value Getters
205 // ( called via Reflection by accessibilityAttributeValue )
208     Radiobutton grouping is done differently in NSAccessibility and the UNO-API. In UNO related radio buttons share an entry in their
209     RelationSet. In NSAccessibility the relationship is axpressed through the hierarchy. A AXRadioGroup contains two or more AXRadioButton
210     objects. Since this group is not available in the UNO hierarchy, an extra wrapper is used for it. This wrapper shares almost all
211     attributes with the first radio button of the group, except for the role, subrole, role description, parent and children attributes.
212     So in this five methods there is a special treatment for radio buttons and groups.
215 -(id)roleAttribute {
216     if ( mActsAsRadioGroup ) {
217         return NSAccessibilityRadioGroupRole;
218     }
219     else {
220         return [ AquaA11yRoleHelper getNativeRoleFrom: [ self accessibleContext ] ];
221     }
224 -(id)subroleAttribute {
225     if ( mActsAsRadioGroup ) {
226         return @"";
227     } else {
228         NSString * subRole = [ AquaA11yRoleHelper getNativeSubroleFrom: [ self accessibleContext ] -> getAccessibleRole() ];
229         if ( ! [ subRole isEqualToString: @"" ] ) {
230             return subRole;
231         } else {
232             [ subRole release ];
233             SAL_WNODEPRECATED_DECLARATIONS_PUSH
234                 //TODO: 10.10 accessibilityAttributeValue:
235             return [ super accessibilityAttributeValue: NSAccessibilitySubroleAttribute ];
236             SAL_WNODEPRECATED_DECLARATIONS_POP
237         }
238     }
241 -(id)titleAttribute {
242     return CreateNSString ( [ self accessibleContext ] -> getAccessibleName() );
245 -(id)descriptionAttribute {
246     if ( [ self accessibleContext ] -> getAccessibleRole() == AccessibleRole::COMBO_BOX ) {
247         return [ self titleAttribute ];
248     } else if ( [ self accessibleExtendedComponent ] ) {
249         return [ AquaA11yComponentWrapper descriptionAttributeForElement: self ];
250     } else {
251         return CreateNSString ( [ self accessibleContext ] -> getAccessibleDescription() );
252     }
255 -(id)enabledAttribute {
256     if ( [ self accessibleContext ] -> getAccessibleStateSet().is() ) {
257         return [ NSNumber numberWithBool: [ self accessibleContext ] -> getAccessibleStateSet() -> contains ( AccessibleStateType::ENABLED ) ];
258     } else {
259         return nil;
260     }
263 -(id)focusedAttribute {
264     if ( [ self accessibleContext ] -> getAccessibleRole() == AccessibleRole::COMBO_BOX ) {
265         id isFocused = nil;
266         Reference < XAccessible > rxParent = [ self accessibleContext ] -> getAccessibleParent();
267         if ( rxParent.is() ) {
268             Reference < XAccessibleContext > rxContext = rxParent -> getAccessibleContext();
269             if ( rxContext.is() && rxContext -> getAccessibleStateSet().is() ) {
270                 isFocused = [ NSNumber numberWithBool: rxContext -> getAccessibleStateSet() -> contains ( AccessibleStateType::FOCUSED ) ];
271             }
272         }
273         return isFocused;
274     } else if ( [ self accessibleContext ] -> getAccessibleStateSet().is() ) {
275         return [ NSNumber numberWithBool: [ self accessibleContext ] -> getAccessibleStateSet() -> contains ( AccessibleStateType::FOCUSED ) ];
276     } else {
277         return nil;
278     }
281 -(id)parentAttribute {
282     if ( [ self accessibleContext ] -> getAccessibleRole() == AccessibleRole::RADIO_BUTTON && ! mActsAsRadioGroup ) {
283         Reference < XAccessible > rxAccessible = [ self getFirstRadioButtonInGroup ];
284         if ( rxAccessible.is() && rxAccessible -> getAccessibleContext().is() ) {
285             Reference < XAccessibleContext > rxAccessibleContext = rxAccessible -> getAccessibleContext();
286             id parent_wrapper = [ AquaA11yFactory wrapperForAccessibleContext: rxAccessibleContext createIfNotExists: YES asRadioGroup: YES ];
287             [ parent_wrapper autorelease ];
288             return NSAccessibilityUnignoredAncestor( parent_wrapper );
289         }
290         return nil;
291     }
292     try {
293         Reference< XAccessible > xParent( [ self accessibleContext ] -> getAccessibleParent() );
294         if ( xParent.is() ) {
295             Reference< XAccessibleContext > xContext( xParent -> getAccessibleContext() );
296             if ( xContext.is() ) {
297                 id parent_wrapper = [ AquaA11yFactory wrapperForAccessibleContext: xContext ];
298                 [ parent_wrapper autorelease ];
299                 return NSAccessibilityUnignoredAncestor( parent_wrapper );
300             }
301         }
302     } catch (const Exception&) {
303     }
305     OSL_ASSERT( false );
306     return nil;
309 -(id)childrenAttribute {
310     if ( mActsAsRadioGroup ) {
311         NSMutableArray * children = [ [ NSMutableArray alloc ] init ];
312         Reference < XAccessibleRelationSet > rxAccessibleRelationSet = [ self accessibleContext ] -> getAccessibleRelationSet();
313         AccessibleRelation relationMemberOf = rxAccessibleRelationSet -> getRelationByType ( AccessibleRelationType::MEMBER_OF );
314         if ( relationMemberOf.RelationType == AccessibleRelationType::MEMBER_OF && relationMemberOf.TargetSet.hasElements() ) {
315             for ( int index = 0; index < relationMemberOf.TargetSet.getLength(); index++ ) {
316                 Reference < XAccessible > rMateAccessible( relationMemberOf.TargetSet[index], UNO_QUERY );
317                 if ( rMateAccessible.is() ) {
318                     Reference< XAccessibleContext > rMateAccessibleContext( rMateAccessible -> getAccessibleContext() );
319                     if ( rMateAccessibleContext.is() ) {
320                         id wrapper = [ AquaA11yFactory wrapperForAccessibleContext: rMateAccessibleContext ];
321                         [ children addObject: wrapper ];
322                         [ wrapper release ];
323                     }
324                 }
325             }
326         }
327         return children;
328     } else if ( [ self accessibleTable ] )
329     {
330         AquaA11yTableWrapper* pTable = [self isKindOfClass: [AquaA11yTableWrapper class]] ? (AquaA11yTableWrapper*)self : nil;
331         return [ AquaA11yTableWrapper childrenAttributeForElement: pTable ];
332     } else {
333         try {
334             NSMutableArray * children = [ [ NSMutableArray alloc ] init ];
335             Reference< XAccessibleContext > xContext( [ self accessibleContext ] );
337             sal_Int32 cnt = xContext -> getAccessibleChildCount();
338             for ( sal_Int32 i = 0; i < cnt; i++ ) {
339                 Reference< XAccessible > xChild( xContext -> getAccessibleChild( i ) );
340                 if( xChild.is() ) {
341                     Reference< XAccessibleContext > xChildContext( xChild -> getAccessibleContext() );
342                     // the menubar is already accessible (including Apple- and Application-Menu) through NSApplication => omit it here
343                     if ( xChildContext.is() && AccessibleRole::MENU_BAR != xChildContext -> getAccessibleRole() ) {
344                         id wrapper = [ AquaA11yFactory wrapperForAccessibleContext: xChildContext ];
345                         [ children addObject: wrapper ];
346                         [ wrapper release ];
347                     }
348                 }
349             }
351             // if not already acting as RadioGroup now is the time to replace RadioButtons with RadioGroups and remove RadioButtons
352             if ( ! mActsAsRadioGroup ) {
353                 NSEnumerator * enumerator = [ children objectEnumerator ];
354                 AquaA11yWrapper * element;
355                 while ( ( element = ( (AquaA11yWrapper *) [ enumerator nextObject ] ) ) ) {
356                     if ( [ element accessibleContext ] -> getAccessibleRole() == AccessibleRole::RADIO_BUTTON ) {
357                         if ( [ element isFirstRadioButtonInGroup ] ) {
358                             id wrapper = [ AquaA11yFactory wrapperForAccessibleContext: [ element accessibleContext ] createIfNotExists: YES asRadioGroup: YES ];
359                             [ children replaceObjectAtIndex: [ children indexOfObjectIdenticalTo: element ] withObject: wrapper ];
360                         }
361                         [ children removeObject: element ];
362                     }
363                 }
364             }
366             [ children autorelease ];
367             return NSAccessibilityUnignoredChildren( children );
368         } catch (const Exception &e) {
369             // TODO: Log
370             return nil;
371         }
372     }
375 -(id)windowAttribute {
376     // go upstairs until reaching the broken connection
377     AquaA11yWrapper * aWrapper = self;
378     int loops = 0;
379     while ( [ aWrapper accessibleContext ] -> getAccessibleParent().is() ) {
380         AquaA11yWrapper *aTentativeParentWrapper = [ AquaA11yFactory wrapperForAccessibleContext: [ aWrapper accessibleContext ] -> getAccessibleParent() -> getAccessibleContext() ];
381         // Quick-and-dirty fix for infinite loop after fixing crash in
382         // fdo#47275
383         if ( aTentativeParentWrapper == aWrapper )
384             break;
385         // Even dirtier fix for infinite loop in fdo#55156
386         if ( loops++ == 100 )
387             break;
388         aWrapper = aTentativeParentWrapper;
389         [ aWrapper autorelease ];
390     }
391     // get associated NSWindow
392     NSWindow* theWindow = [ aWrapper windowForParent ];
393     return theWindow;
396 -(id)topLevelUIElementAttribute {
397     return [ self windowAttribute ];
400 -(id)sizeAttribute {
401     if ( [ self accessibleComponent ] ) {
402         return [ AquaA11yComponentWrapper sizeAttributeForElement: self ];
403     } else {
404         return nil;
405     }
408 -(id)positionAttribute {
409     if ( [ self accessibleComponent ] ) {
410         return [ AquaA11yComponentWrapper positionAttributeForElement: self ];
411     } else {
412         return nil;
413     }
416 -(id)helpAttribute {
417     return CreateNSString ( [ self accessibleContext ] -> getAccessibleDescription() );
420 -(id)roleDescriptionAttribute {
421     if ( mActsAsRadioGroup ) {
422         return [ AquaA11yRoleHelper getRoleDescriptionFrom: NSAccessibilityRadioGroupRole with: @"" ];
423         } else if( [ self accessibleContext ] -> getAccessibleRole() == AccessibleRole::RADIO_BUTTON ) {
424                 // FIXME: VO should read this because of hierarchy, this is just a workaround
425                 // get parent and its children
426                 AquaA11yWrapper * parent = [ self parentAttribute ];
427                 NSArray * children = [ parent childrenAttribute ];
428                 // find index of self
429                 int index = 1;
430                 NSEnumerator * enumerator = [ children objectEnumerator ];
431                 AquaA11yWrapper * child = nil;
432                 while ( ( child = [ enumerator nextObject ] ) ) {
433                         if ( self == child ) {
434                                 break;
435                         }
436                         index++;
437                 }
438                 // build string
439                 NSNumber * nIndex = [ NSNumber numberWithInt: index ];
440                 NSNumber * nGroupsize = [ NSNumber numberWithInt: [ children count ] ];
441                 NSMutableString * value = [ [ NSMutableString alloc ] init ];
442                 [ value appendString: @"radio button " ];
443                 [ value appendString: [ nIndex stringValue ] ];
444                 [ value appendString: @" of " ];
445                 [ value appendString: [ nGroupsize stringValue ] ];
446                 // clean up and return string
447                 [ nIndex release ];
448                 [ nGroupsize release ];
449                 [ children release ];
450                 return value;
451     } else {
452         return [ AquaA11yRoleHelper getRoleDescriptionFrom:
453                 [ AquaA11yRoleHelper getNativeRoleFrom: [ self accessibleContext ] ]
454                 with: [ AquaA11yRoleHelper getNativeSubroleFrom: [ self accessibleContext ] -> getAccessibleRole() ] ];
455     }
458 -(id)valueAttribute {
459     if ( [ [ self roleAttribute ] isEqualToString: NSAccessibilityMenuItemRole ] ) {
460         return nil;
461     } else if ( [ self accessibleText ] ) {
462         return [ AquaA11yTextWrapper valueAttributeForElement: self ];
463     } else if ( [ self accessibleValue ] ) {
464         return [ AquaA11yValueWrapper valueAttributeForElement: self ];
465     } else {
466         return nil;
467     }
470 -(id)minValueAttribute {
471     if ( [ self accessibleValue ] ) {
472         return [ AquaA11yValueWrapper minValueAttributeForElement: self ];
473     } else {
474         return nil;
475     }
478 -(id)maxValueAttribute {
479     if ( [ self accessibleValue ] ) {
480         return [ AquaA11yValueWrapper maxValueAttributeForElement: self ];
481     } else {
482         return nil;
483     }
486 -(id)contentsAttribute {
487     return [ self childrenAttribute ];
490 -(id)selectedChildrenAttribute {
491     return [ AquaA11ySelectionWrapper selectedChildrenAttributeForElement: self ];
494 -(id)numberOfCharactersAttribute {
495     if ( [ self accessibleText ] ) {
496         return [ AquaA11yTextWrapper numberOfCharactersAttributeForElement: self ];
497     } else {
498         return nil;
499     }
502 -(id)selectedTextAttribute {
503     if ( [ self accessibleText ] ) {
504         return [ AquaA11yTextWrapper selectedTextAttributeForElement: self ];
505     } else {
506         return nil;
507     }
510 -(id)selectedTextRangeAttribute {
511     if ( [ self accessibleText ] ) {
512         return [ AquaA11yTextWrapper selectedTextRangeAttributeForElement: self ];
513     } else {
514         return nil;
515     }
518 -(id)visibleCharacterRangeAttribute {
519     if ( [ self accessibleText ] ) {
520         return [ AquaA11yTextWrapper visibleCharacterRangeAttributeForElement: self ];
521     } else {
522         return nil;
523     }
526 -(id)tabsAttribute {
527     return self; // TODO ???
530 -(id)sharedTextUIElementsAttribute {
531     if ( [ self accessibleText ] ) {
532         return [ AquaA11yTextWrapper sharedTextUIElementsAttributeForElement: self ];
533     } else {
534         return nil;
535     }
538 -(id)sharedCharacterRangeAttribute {
539     if ( [ self accessibleText ] ) {
540         return [ AquaA11yTextWrapper sharedCharacterRangeAttributeForElement: self ];
541     } else {
542         return nil;
543     }
546 -(id)expandedAttribute {
547     return [ NSNumber numberWithBool: [ self accessibleContext ] -> getAccessibleStateSet() -> contains ( AccessibleStateType::EXPANDED ) ];
550 -(id)selectedAttribute {
551     return [ NSNumber numberWithBool: [ self accessibleContext ] -> getAccessibleStateSet() -> contains ( AccessibleStateType::SELECTED ) ];
554 -(id)stringForRangeAttributeForParameter:(id)range {
555     if ( [ self accessibleText ] ) {
556         return [ AquaA11yTextWrapper stringForRangeAttributeForElement: self forParameter: range ];
557     } else {
558         return nil;
559     }
562 -(id)attributedStringForRangeAttributeForParameter:(id)range {
563     if ( [ self accessibleText ] ) {
564         return [ AquaA11yTextWrapper attributedStringForRangeAttributeForElement: self forParameter: range ];
565     } else {
566         return nil;
567     }
570 -(id)rangeForIndexAttributeForParameter:(id)index {
571     if ( [ self accessibleText ] ) {
572         return [ AquaA11yTextWrapper rangeForIndexAttributeForElement: self forParameter: index ];
573     } else {
574         return nil;
575     }
578 -(id)rangeForPositionAttributeForParameter:(id)point {
579     if ( [ self accessibleText ] ) {
580         return [ AquaA11yTextWrapper rangeForPositionAttributeForElement: self forParameter: point ];
581     } else {
582         return nil;
583     }
586 -(id)boundsForRangeAttributeForParameter:(id)range {
587     if ( [ self accessibleText ] ) {
588         return [ AquaA11yTextWrapper boundsForRangeAttributeForElement: self forParameter: range ];
589     } else {
590         return nil;
591     }
594 -(id)styleRangeForIndexAttributeForParameter:(id)index {
595     if ( [ self accessibleText ] ) {
596         return [ AquaA11yTextWrapper styleRangeForIndexAttributeForElement: self forParameter: index ];
597     } else {
598         return nil;
599     }
602 -(id)rTFForRangeAttributeForParameter:(id)range {
603     if ( [ self accessibleText ] ) {
604         return [ AquaA11yTextWrapper rTFForRangeAttributeForElement: self forParameter: range ];
605     } else {
606         return nil;
607     }
610 -(id)orientationAttribute {
611     NSString * orientation = nil;
612     Reference < XAccessibleStateSet > stateSet = [ self accessibleContext ] -> getAccessibleStateSet();
613     if ( stateSet -> contains ( AccessibleStateType::HORIZONTAL ) ) {
614         orientation = NSAccessibilityHorizontalOrientationValue;
615     } else if ( stateSet -> contains ( AccessibleStateType::VERTICAL ) ) {
616         orientation = NSAccessibilityVerticalOrientationValue;
617     }
618     return orientation;
621 -(id)titleUIElementAttribute {
622     if ( [ self accessibleContext ] -> getAccessibleRelationSet().is() ) {
623         NSString * title = [ self titleAttribute ];
624         id titleElement = nil;
625         if ( [ title length ] == 0 ) {
626             AccessibleRelation relationLabeledBy = [ self accessibleContext ] -> getAccessibleRelationSet() -> getRelationByType ( AccessibleRelationType::LABELED_BY );
627             if ( relationLabeledBy.RelationType == AccessibleRelationType::LABELED_BY && relationLabeledBy.TargetSet.hasElements()  ) {
628                 Reference < XAccessible > rxAccessible ( relationLabeledBy.TargetSet[0], UNO_QUERY );
629                 titleElement = [ AquaA11yFactory wrapperForAccessibleContext: rxAccessible -> getAccessibleContext() ];
630             }
631         }
632         if ( title ) {
633             [ title release ];
634         }
635         return titleElement;
636     } else {
637         return nil;
638     }
641 -(id)servesAsTitleForUIElementsAttribute {
642     if ( [ self accessibleContext ] -> getAccessibleRelationSet().is() ) {
643         id titleForElement = nil;
644         AccessibleRelation relationLabelFor = [ self accessibleContext ] -> getAccessibleRelationSet() -> getRelationByType ( AccessibleRelationType::LABEL_FOR );
645         if ( relationLabelFor.RelationType == AccessibleRelationType::LABEL_FOR && relationLabelFor.TargetSet.hasElements() ) {
646             Reference < XAccessible > rxAccessible ( relationLabelFor.TargetSet[0], UNO_QUERY );
647             titleForElement = [ AquaA11yFactory wrapperForAccessibleContext: rxAccessible -> getAccessibleContext() ];
648         }
649         return titleForElement;
650     } else {
651         return nil;
652     }
655 -(id)lineForIndexAttributeForParameter:(id)index {
656     if ( [ self accessibleMultiLineText ] ) {
657         return [ AquaA11yTextWrapper lineForIndexAttributeForElement: self forParameter: index ];
658     } else {
659         return nil;
660     }
663 -(id)rangeForLineAttributeForParameter:(id)line {
664     if ( [ self accessibleMultiLineText ] ) {
665         return [ AquaA11yTextWrapper rangeForLineAttributeForElement: self forParameter: line ];
666     } else {
667         return nil;
668     }
671 #pragma mark -
672 #pragma mark Accessibility Protocol
674 -(id)accessibilityAttributeValue:(NSString *)attribute {
675     SAL_INFO("vcl.a11y", "[" << self << " accessibilityAttributeValue:" << attribute << "]");
676     // #i90575# guard NSAccessibility protocol against unwanted access
677     if ( isPopupMenuOpen ) {
678         return nil;
679     }
681     id value = nil;
682     // if we are no longer in the wrapper repository, we have been disposed
683     AquaA11yWrapper * theWrapper = [ AquaA11yFactory wrapperForAccessibleContext: [ self accessibleContext ] createIfNotExists: NO ];
684     if ( theWrapper || mIsTableCell ) {
685         try {
686             SEL methodSelector = [ self selectorForAttribute: attribute asGetter: YES withGetterParameter: NO ];
687             if ( [ self respondsToSelector: methodSelector ] ) {
688                 value = [ self performSelector: methodSelector ];
689             }
690         } catch ( const DisposedException & e ) {
691             mIsTableCell = NO; // just to be sure
692             [ AquaA11yFactory removeFromWrapperRepositoryFor: [ self accessibleContext ] ];
693             return nil;
694         } catch ( const Exception & e ) {
695             // empty
696         }
697     }
698     if ( theWrapper ) {
699         [ theWrapper release ]; // the above called method calls retain on the returned Wrapper
700     }
701     return value;
704 -(BOOL)accessibilityIsIgnored {
705     SAL_INFO("vcl.a11y", "[" << self << " accessibilityIsIgnored]");
706     // #i90575# guard NSAccessibility protocol against unwanted access
707     if ( isPopupMenuOpen ) {
708         return NO;
709     }
710     BOOL ignored = NO;
711     sal_Int16 nRole = [ self accessibleContext ] -> getAccessibleRole();
712     switch ( nRole ) {
713         //case AccessibleRole::PANEL:
714         case AccessibleRole::FRAME:
715         case AccessibleRole::ROOT_PANE:
716         case AccessibleRole::SEPARATOR:
717         case AccessibleRole::FILLER:
718         case AccessibleRole::DIALOG:
719             ignored = YES;
720             break;
721         default:
722             ignored = ! ( [ self accessibleContext ] -> getAccessibleStateSet() -> contains ( AccessibleStateType::VISIBLE ) );
723             break;
724     }
725     return ignored; // TODO: to be completed
728 -(NSArray *)accessibilityAttributeNames {
729     SAL_INFO("vcl.a11y", "[" << self << " accessibilityAttributeNames]");
730     // #i90575# guard NSAccessibility protocol against unwanted access
731     if ( isPopupMenuOpen ) {
732         return nil;
733     }
734     NSString * nativeSubrole = nil;
735     NSString * title = nil;
736     NSMutableArray * attributeNames = nil;
737     sal_Int32 nAccessibleChildren = 0;
738     try {
739         // Default Attributes
740         attributeNames = [ NSMutableArray arrayWithObjects:
741             NSAccessibilityRoleAttribute,
742             NSAccessibilityDescriptionAttribute,
743             NSAccessibilityParentAttribute,
744             NSAccessibilityWindowAttribute,
745             NSAccessibilityHelpAttribute,
746             NSAccessibilityTopLevelUIElementAttribute,
747             NSAccessibilityRoleDescriptionAttribute,
748             nil ];
749         nativeSubrole = (NSString *) [ AquaA11yRoleHelper getNativeSubroleFrom: [ self accessibleContext ] -> getAccessibleRole() ];
750         title = (NSString *) [ self titleAttribute ];
751         Reference < XAccessibleRelationSet > rxRelationSet = [ self accessibleContext ] -> getAccessibleRelationSet();
752         // Special Attributes depending on attribute values
753         if ( nativeSubrole && ! [ nativeSubrole isEqualToString: @"" ] ) {
754             [ attributeNames addObject: NSAccessibilitySubroleAttribute ];
755         }
756         try
757         {
758             nAccessibleChildren = [ self accessibleContext ] -> getAccessibleChildCount();
759             if (  nAccessibleChildren > 0 ) {
760                 [ attributeNames addObject: NSAccessibilityChildrenAttribute ];
761         }
762         }
763         catch( DisposedException& ) {}
764         catch( RuntimeException& ) {}
766         if ( title && ! [ title isEqualToString: @"" ] ) {
767             [ attributeNames addObject: NSAccessibilityTitleAttribute ];
768         }
769         if ( [ title length ] == 0 && rxRelationSet.is() && rxRelationSet -> containsRelation ( AccessibleRelationType::LABELED_BY ) ) {
770             [ attributeNames addObject: NSAccessibilityTitleUIElementAttribute ];
771         }
772         if ( rxRelationSet.is() && rxRelationSet -> containsRelation ( AccessibleRelationType::LABEL_FOR ) ) {
773             [ attributeNames addObject: NSAccessibilityServesAsTitleForUIElementsAttribute ];
774         }
775         // Special Attributes depending on interface
776         if( [self accessibleContext ] -> getAccessibleRole() == AccessibleRole::TABLE )
777             [AquaA11yTableWrapper addAttributeNamesTo: attributeNames object: self];
779         if ( [ self accessibleText ] ) {
780             [ AquaA11yTextWrapper addAttributeNamesTo: attributeNames ];
781         }
782         if ( [ self accessibleComponent ] ) {
783             [ AquaA11yComponentWrapper addAttributeNamesTo: attributeNames ];
784         }
785         if ( [ self accessibleSelection ] ) {
786             [ AquaA11ySelectionWrapper addAttributeNamesTo: attributeNames ];
787         }
788         if ( [ self accessibleValue ] ) {
789             [ AquaA11yValueWrapper addAttributeNamesTo: attributeNames ];
790         }
791         [ nativeSubrole release ];
792         [ title release ];
793         return attributeNames;
794     } catch ( DisposedException & e ) { // Object is no longer available
795         if ( nativeSubrole ) {
796             [ nativeSubrole release ];
797         }
798         if ( title ) {
799             [ title release ];
800         }
801         if ( attributeNames ) {
802             [ attributeNames release ];
803         }
804         [ AquaA11yFactory removeFromWrapperRepositoryFor: [ self accessibleContext ] ];
805         return [ [ NSArray alloc ] init ];
806     }
809 -(BOOL)accessibilityIsAttributeSettable:(NSString *)attribute {
810     SAL_INFO("vcl.a11y", "[" << self << " accessibilityAttributeIsSettable:" << attribute << "]");
811     BOOL isSettable = NO;
812     if ( [ self accessibleText ] ) {
813         isSettable = [ AquaA11yTextWrapper isAttributeSettable: attribute forElement: self ];
814     }
815     if ( ! isSettable && [ self accessibleComponent ] ) {
816         isSettable = [ AquaA11yComponentWrapper isAttributeSettable: attribute forElement: self ];
817     }
818     if ( ! isSettable && [ self accessibleSelection ] ) {
819         isSettable = [ AquaA11ySelectionWrapper isAttributeSettable: attribute forElement: self ];
820     }
821     if ( ! isSettable && [ self accessibleValue ] ) {
822         isSettable = [ AquaA11yValueWrapper isAttributeSettable: attribute forElement: self ];
823     }
824     return isSettable; // TODO: to be completed
827 -(NSArray *)accessibilityParameterizedAttributeNames {
828     SAL_INFO("vcl.a11y", "[" << self << " accessibilityParameterizedAttributeNames]");
829     NSMutableArray * attributeNames = [ [ NSMutableArray alloc ] init ];
830     // Special Attributes depending on interface
831     if ( [ self accessibleText ] ) {
832         [ AquaA11yTextWrapper addParameterizedAttributeNamesTo: attributeNames ];
833     }
834     return attributeNames; // TODO: to be completed
837 -(id)accessibilityAttributeValue:(NSString *)attribute forParameter:(id)parameter {
838     SAL_INFO("vcl.a11y", "[" << self << " accessibilityAttributeValue:" << attribute << " forParameter:" << ((NSObject*)parameter) << "]");
839     SEL methodSelector = [ self selectorForAttribute: attribute asGetter: YES withGetterParameter: YES ];
840     if ( [ self respondsToSelector: methodSelector ] ) {
841         return [ self performSelector: methodSelector withObject: parameter ];
842     }
843     return nil; // TODO: to be completed
846 -(BOOL)accessibilitySetOverrideValue:(id)value forAttribute:(NSString *)attribute
848     SAL_INFO("vcl.a11y", "[" << self << " accessibilitySetOverrideValue:" << ((NSObject*)value) << " forAttribute:" << attribute << "]");
849     (void)value;
850     (void)attribute;
851     return NO; // TODO
854 -(void)accessibilitySetValue:(id)value forAttribute:(NSString *)attribute {
855     SAL_INFO("vcl.a11y", "[" << self << " accessibilitySetValue:" << ((NSObject*)value) << " forAttribute:" << attribute << "]");
856     SEL methodSelector = [ self selectorForAttribute: attribute asGetter: NO withGetterParameter: NO ];
857     if ( [ AquaA11yComponentWrapper respondsToSelector: methodSelector ] ) {
858         [ AquaA11yComponentWrapper performSelector: methodSelector withObject: self withObject: value ];
859     }
860     if ( [ AquaA11yTextWrapper respondsToSelector: methodSelector ] ) {
861         [ AquaA11yTextWrapper performSelector: methodSelector withObject: self withObject: value ];
862     }
863     if ( [ AquaA11ySelectionWrapper respondsToSelector: methodSelector ] ) {
864         [ AquaA11ySelectionWrapper performSelector: methodSelector withObject: self withObject: value ];
865     }
866     if ( [ AquaA11yValueWrapper respondsToSelector: methodSelector ] ) {
867         [ AquaA11yValueWrapper performSelector: methodSelector withObject: self withObject: value ];
868     }
871 -(id)accessibilityFocusedUIElement {
872     SAL_INFO("vcl.a11y", "[" << self << " accessibilityFocusedUIElement]");
873     // #i90575# guard NSAccessibility protocol against unwanted access
874     if ( isPopupMenuOpen ) {
875         return nil;
876     }
878     // as this seems to be the first API call on a newly created SalFrameView object,
879     // make sure self gets registered in the repository ..
880     [ self accessibleContext ];
882     AquaA11yWrapper * focusedUIElement = AquaA11yFocusListener::get()->getFocusedUIElement();
883 //    AquaA11yWrapper * ancestor = focusedUIElement;
885       // Make sure the focused object is a descendant of self
886 //    do  {
887 //       if( self == ancestor )
888              return focusedUIElement;
890 //       ancestor = [ ancestor accessibilityAttributeValue: NSAccessibilityParentAttribute ];
891 //    }  while( nil != ancestor );
893     return self;
896 -(NSString *)accessibilityActionDescription:(NSString *)action {
897     SAL_INFO("vcl.a11y", "[" << self << " accessibilityActionDescription:" << action << "]");
898     return NSAccessibilityActionDescription(action);
901 -(AquaA11yWrapper *)actionResponder {
902     AquaA11yWrapper * wrapper = nil;
903     // get some information
904     NSString * role = (NSString *) [ self accessibilityAttributeValue: NSAccessibilityRoleAttribute ];
905     id enabledAttr = [ self enabledAttribute ];
906     BOOL enabled = [ enabledAttr boolValue ];
907     NSView * parent = (NSView *) [ self accessibilityAttributeValue: NSAccessibilityParentAttribute ];
908     AquaA11yWrapper * parentAsWrapper = nil;
909     if ( [ parent isKindOfClass: [ AquaA11yWrapper class ] ] ) {
910         parentAsWrapper = (AquaA11yWrapper *) parent;
911     }
912     SAL_WNODEPRECATED_DECLARATIONS_PUSH
913         //TODO: 10.10 accessibilityAttributeValue:
914     NSString * parentRole = (NSString *) [ parent accessibilityAttributeValue: NSAccessibilityRoleAttribute ];
915     SAL_WNODEPRECATED_DECLARATIONS_POP
916     // if we are a textarea inside a combobox, then the combobox is the action responder
917     if ( enabled
918       && [ role isEqualToString: NSAccessibilityTextAreaRole ]
919       && [ parentRole isEqualToString: NSAccessibilityComboBoxRole ]
920       && parentAsWrapper ) {
921         wrapper = parentAsWrapper;
922     } else if ( enabled && [ self accessibleAction ] ) {
923         wrapper = self ;
924     }
925     [ parentRole release ];
926     [ enabledAttr release ];
927     [ role release ];
928     return wrapper;
931 -(void)accessibilityPerformAction:(NSString *)action {
932     SAL_INFO("vcl.a11y", "[" << self << " accessibilityPerformAction:" << action << "]");
933     AquaA11yWrapper * actionResponder = [ self actionResponder ];
934     if ( actionResponder ) {
935         [ AquaA11yActionWrapper doAction: action ofElement: actionResponder ];
936     }
939 -(NSArray *)accessibilityActionNames {
940     SAL_INFO("vcl.a11y", "[" << self << " accessibilityActionNames]");
941     NSArray * actionNames = nil;
942     AquaA11yWrapper * actionResponder = [ self actionResponder ];
943     if ( actionResponder ) {
944         actionNames = [ AquaA11yActionWrapper actionNamesForElement: actionResponder ];
945     } else {
946         actionNames = [ [ NSArray alloc ] init ];
947     }
948     return actionNames;
951 #pragma mark -
952 #pragma mark Hit Test
954 -(BOOL)isViewElement:(NSObject *)viewElement hitByPoint:(NSPoint)point {
955     BOOL hit = NO;
956     NSAutoreleasePool * pool = [ [ NSAutoreleasePool alloc ] init ];
957     SAL_WNODEPRECATED_DECLARATIONS_PUSH
958         //TODO: 10.10 accessibilityAttributeValue:
959     NSValue * position = [ viewElement accessibilityAttributeValue: NSAccessibilityPositionAttribute ];
960     NSValue * size = [ viewElement accessibilityAttributeValue: NSAccessibilitySizeAttribute ];
961     SAL_WNODEPRECATED_DECLARATIONS_POP
962     if ( position && size ) {
963         float minX = [ position pointValue ].x;
964         float minY = [ position pointValue ].y;
965         float maxX = minX + [ size sizeValue ].width;
966         float maxY = minY + [ size sizeValue ].height;
967         if ( minX < point.x && maxX > point.x && minY < point.y && maxY > point.y ) {
968             hit = YES;
969         }
970     }
971     [ pool release ];
972     return hit;
975 Reference < XAccessibleContext > hitTestRunner ( css::awt::Point point,
976                                                  Reference < XAccessibleContext > const & rxAccessibleContext ) {
977     Reference < XAccessibleContext > hitChild;
978     Reference < XAccessibleContext > emptyReference;
979     try {
980         Reference < XAccessibleComponent > rxAccessibleComponent ( rxAccessibleContext, UNO_QUERY );
981         if ( rxAccessibleComponent.is() ) {
982             css::awt::Point location = rxAccessibleComponent -> getLocationOnScreen();
983             css::awt::Point hitPoint ( point.X - location.X , point.Y - location.Y);
984             Reference < XAccessible > rxAccessible = rxAccessibleComponent -> getAccessibleAtPoint ( hitPoint );
985             if ( rxAccessible.is() && rxAccessible -> getAccessibleContext().is() &&
986                  rxAccessible -> getAccessibleContext() -> getAccessibleChildCount() == 0 ) {
987                 hitChild = rxAccessible -> getAccessibleContext();
988             }
989         }
991         // iterate the hirerachy looking doing recursive hit testing.
992         // apparently necessary as a special treatment for e.g. comboboxes
993         if ( !hitChild.is() ) {
994             bool bSafeToIterate = true;
995             sal_Int32 nCount = rxAccessibleContext -> getAccessibleChildCount();
997             if ( nCount < 0 || nCount > SAL_MAX_UINT16 /* slow enough for anyone */ )
998                 bSafeToIterate = false;
999             else { // manages descendants is an horror from the a11y standards guys.
1000                 Reference< XAccessibleStateSet > xStateSet;
1001                 xStateSet = rxAccessibleContext -> getAccessibleStateSet();
1002                 if (xStateSet.is() && xStateSet -> contains(AccessibleStateType::MANAGES_DESCENDANTS ) )
1003                     bSafeToIterate = false;
1004             }
1006             if( bSafeToIterate ) {
1007                 for ( int i = 0; i < rxAccessibleContext -> getAccessibleChildCount(); i++ ) {
1008                     Reference < XAccessible > rxAccessibleChild = rxAccessibleContext -> getAccessibleChild ( i );
1009                     if ( rxAccessibleChild.is() && rxAccessibleChild -> getAccessibleContext().is() && rxAccessibleChild -> getAccessibleContext() -> getAccessibleRole() != AccessibleRole::LIST ) {
1010                         Reference < XAccessibleContext > myHitChild = hitTestRunner ( point, rxAccessibleChild -> getAccessibleContext() );
1011                         if ( myHitChild.is() ) {
1012                             hitChild = myHitChild;
1013                             break;
1014                         }
1015                     }
1016                 }
1017             }
1018         }
1019     } catch ( RuntimeException ) {
1020         return emptyReference;
1021     }
1022     return hitChild;
1025 -(id)accessibilityHitTest:(NSPoint)point {
1026     SAL_INFO("vcl.a11y", "[" << self << " accessibilityHitTest:" << point << "]");
1027     static id wrapper = nil;
1028     if ( nil != wrapper ) {
1029         [ wrapper release ];
1030         wrapper = nil;
1031     }
1032     Reference < XAccessibleContext > hitChild;
1033     NSRect screenRect = [ [ NSScreen mainScreen ] frame ];
1034     css::awt::Point hitPoint ( static_cast<long>(point.x) , static_cast<long>(screenRect.size.height - point.y) );
1035     // check child windows first
1036     NSWindow * window = (NSWindow *) [ self accessibilityAttributeValue: NSAccessibilityWindowAttribute ];
1037     NSArray * childWindows = [ window childWindows ];
1038     if ( [ childWindows count ] > 0 ) {
1039         NSWindow * element = nil;
1040         NSEnumerator * enumerator = [ childWindows objectEnumerator ];
1041         while ( ( element = [ enumerator nextObject ] ) && !hitChild.is() ) {
1042             if ( [ element isKindOfClass: [ SalFrameWindow class ] ] && [ self isViewElement: element hitByPoint: point ] ) {
1043                 // we have a child window that is hit
1044                 Reference < XAccessibleRelationSet > relationSet = [ ( ( SalFrameWindow * ) element ) accessibleContext ] -> getAccessibleRelationSet();
1045                 if ( relationSet.is() && relationSet -> containsRelation ( AccessibleRelationType::SUB_WINDOW_OF )) {
1046                     // we have a valid relation to the parent element
1047                     AccessibleRelation relation = relationSet -> getRelationByType ( AccessibleRelationType::SUB_WINDOW_OF );
1048                     for ( int i = 0; i < relation.TargetSet.getLength() && !hitChild.is(); i++ ) {
1049                         Reference < XAccessible > rxAccessible ( relation.TargetSet [ i ], UNO_QUERY );
1050                         if ( rxAccessible.is() && rxAccessible -> getAccessibleContext().is() ) {
1051                             // hit test for children of parent
1052                             hitChild = hitTestRunner ( hitPoint, rxAccessible -> getAccessibleContext() );
1053                         }
1054                     }
1055                 }
1056             }
1057         }
1058     }
1059     // nothing hit yet, so check ourself
1060     if ( ! hitChild.is() ) {
1061         if ( !mpReferenceWrapper ) {
1062             [ self setDefaults: [ self accessibleContext ] ];
1063         }
1064         hitChild = hitTestRunner ( hitPoint, mpReferenceWrapper -> rAccessibleContext );
1065     }
1066     if ( hitChild.is() ) {
1067         wrapper = [ AquaA11yFactory wrapperForAccessibleContext: hitChild ];
1068     }
1069     if ( wrapper ) {
1070         [ wrapper retain ]; // TODO: retain only when transient ?
1071     }
1072     return wrapper;
1075 #pragma mark -
1076 #pragma mark Access Methods
1078 -(XAccessibleAction *)accessibleAction {
1079     return mpReferenceWrapper -> rAccessibleAction.get();
1082 -(XAccessibleContext *)accessibleContext {
1083     return mpReferenceWrapper -> rAccessibleContext.get();
1086 -(XAccessibleComponent *)accessibleComponent {
1087     return mpReferenceWrapper -> rAccessibleComponent.get();
1090 -(XAccessibleExtendedComponent *)accessibleExtendedComponent {
1091     return mpReferenceWrapper -> rAccessibleExtendedComponent.get();
1094 -(XAccessibleSelection *)accessibleSelection {
1095     return mpReferenceWrapper -> rAccessibleSelection.get();
1098 -(XAccessibleTable *)accessibleTable {
1099     return mpReferenceWrapper -> rAccessibleTable.get();
1102 -(XAccessibleText *)accessibleText {
1103     return mpReferenceWrapper -> rAccessibleText.get();
1106 -(XAccessibleEditableText *)accessibleEditableText {
1107     return mpReferenceWrapper -> rAccessibleEditableText.get();
1110 -(XAccessibleValue *)accessibleValue {
1111     return mpReferenceWrapper -> rAccessibleValue.get();
1114 -(XAccessibleTextAttributes *)accessibleTextAttributes {
1115     return mpReferenceWrapper -> rAccessibleTextAttributes.get();
1118 -(XAccessibleMultiLineText *)accessibleMultiLineText {
1119     return mpReferenceWrapper -> rAccessibleMultiLineText.get();
1122 -(XAccessibleTextMarkup *)accessibleTextMarkup {
1123     return mpReferenceWrapper -> rAccessibleTextMarkup.get();
1126 -(NSWindow*)windowForParent {
1127     return [self window];
1130 -(void)setActsAsRadioGroup:(BOOL)actsAsRadioGroup {
1131     mActsAsRadioGroup = actsAsRadioGroup;
1134 -(BOOL)actsAsRadioGroup {
1135     return mActsAsRadioGroup;
1138 +(void)setPopupMenuOpen:(BOOL)popupMenuOpen {
1139     isPopupMenuOpen = popupMenuOpen;
1142 @end
1144 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */