1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
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/.
9 * This file incorporates work covered by the following license notice:
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 .
21 #include <osx/salinst.h>
22 #include <osx/a11yfactory.h>
23 #include <osx/a11yfocustracker.hxx>
25 #include "a11yfocuslistener.hxx"
26 #include "a11yrolehelper.h"
27 #include "a11ywrapperbutton.h"
28 #include "a11ywrapperstatictext.h"
29 #include "a11ywrappertextarea.h"
30 #include "a11ywrappercheckbox.h"
31 #include "a11ywrappercombobox.h"
32 #include "a11ywrappergroup.h"
33 #include "a11ywrapperlist.h"
34 #include "a11ywrapperradiobutton.h"
35 #include "a11ywrapperradiogroup.h"
36 #include "a11ywrapperrow.h"
37 #include "a11ywrapperscrollarea.h"
38 #include "a11ywrapperscrollbar.h"
39 #include "a11ywrappersplitter.h"
40 #include "a11ywrappertabgroup.h"
41 #include "a11ywrappertoolbar.h"
42 #include "a11ytablewrapper.h"
44 #include <com/sun/star/accessibility/AccessibleStateType.hpp>
46 using namespace ::com::sun::star::accessibility;
47 using namespace ::com::sun::star::uno;
49 static bool enabled = false;
51 @implementation AquaA11yFactory : NSObject
54 #pragma mark Wrapper Repository
56 +(NSMutableDictionary *)allWrapper {
57 static NSMutableDictionary * mdAllWrapper = nil;
58 if ( mdAllWrapper == nil ) {
59 mdAllWrapper = [ [ [ NSMutableDictionary alloc ] init ] retain ];
60 // initialize keyboard focus tracker
61 rtl::Reference< AquaA11yFocusListener > listener( AquaA11yFocusListener::get() );
62 TheAquaA11yFocusTracker().setFocusListener(listener);
68 +(NSValue *)keyForAccessibleContext: (Reference < XAccessibleContext >) rxAccessibleContext {
69 return [ NSValue valueWithPointer: rxAccessibleContext.get() ];
72 +(NSValue *)keyForAccessibleContextAsRadioGroup: (Reference < XAccessibleContext >) rxAccessibleContext {
73 return [ NSValue valueWithPointer: ( rxAccessibleContext.get() + 2 ) ];
76 +(AquaA11yWrapper *)wrapperForAccessible: (Reference < XAccessible >) rxAccessible {
77 if ( rxAccessible.is() ) {
78 Reference< XAccessibleContext > xAccessibleContext = rxAccessible->getAccessibleContext();
79 if( xAccessibleContext.is() ) {
80 return [ AquaA11yFactory wrapperForAccessibleContext: xAccessibleContext ];
86 +(AquaA11yWrapper *)wrapperForAccessibleContext: (Reference < XAccessibleContext >) rxAccessibleContext {
87 return [ AquaA11yFactory wrapperForAccessibleContext: rxAccessibleContext createIfNotExists: YES asRadioGroup: NO ];
90 +(AquaA11yWrapper *)wrapperForAccessibleContext: (Reference < XAccessibleContext >) rxAccessibleContext createIfNotExists:(BOOL) bCreate {
91 return [ AquaA11yFactory wrapperForAccessibleContext: rxAccessibleContext createIfNotExists: bCreate asRadioGroup: NO ];
94 +(AquaA11yWrapper *)wrapperForAccessibleContext: (Reference < XAccessibleContext >) rxAccessibleContext createIfNotExists:(BOOL) bCreate asRadioGroup:(BOOL) asRadioGroup{
95 NSMutableDictionary * dAllWrapper = [ AquaA11yFactory allWrapper ];
98 nKey = [ AquaA11yFactory keyForAccessibleContextAsRadioGroup: rxAccessibleContext ];
100 nKey = [ AquaA11yFactory keyForAccessibleContext: rxAccessibleContext ];
102 AquaA11yWrapper * aWrapper = static_cast<AquaA11yWrapper *>([ dAllWrapper objectForKey: nKey ]);
103 if ( aWrapper != nil ) {
105 } else if ( bCreate ) {
106 NSString * nativeRole = [ AquaA11yRoleHelper getNativeRoleFrom: rxAccessibleContext.get() ];
108 if ( [ nativeRole isEqualToString: NSAccessibilityButtonRole ] ) {
109 aWrapper = [ [ AquaA11yWrapperButton alloc ] initWithAccessibleContext: rxAccessibleContext ];
110 } else if ( [ nativeRole isEqualToString: NSAccessibilityTextAreaRole ] ) {
111 aWrapper = [ [ AquaA11yWrapperTextArea alloc ] initWithAccessibleContext: rxAccessibleContext ];
112 } else if ( [ nativeRole isEqualToString: NSAccessibilityStaticTextRole ] ) {
113 aWrapper = [ [ AquaA11yWrapperStaticText alloc ] initWithAccessibleContext: rxAccessibleContext ];
114 } else if ( [ nativeRole isEqualToString: NSAccessibilityComboBoxRole ] ) {
115 aWrapper = [ [ AquaA11yWrapperComboBox alloc ] initWithAccessibleContext: rxAccessibleContext ];
116 } else if ( [ nativeRole isEqualToString: NSAccessibilityGroupRole ] ) {
117 aWrapper = [ [ AquaA11yWrapperGroup alloc ] initWithAccessibleContext: rxAccessibleContext ];
118 } else if ( [ nativeRole isEqualToString: NSAccessibilityToolbarRole ] ) {
119 aWrapper = [ [ AquaA11yWrapperToolbar alloc ] initWithAccessibleContext: rxAccessibleContext ];
120 } else if ( [ nativeRole isEqualToString: NSAccessibilityScrollAreaRole ] ) {
121 aWrapper = [ [ AquaA11yWrapperScrollArea alloc ] initWithAccessibleContext: rxAccessibleContext ];
122 } else if ( [ nativeRole isEqualToString: NSAccessibilityTabGroupRole ] ) {
123 aWrapper = [ [ AquaA11yWrapperTabGroup alloc ] initWithAccessibleContext: rxAccessibleContext ];
124 } else if ( [ nativeRole isEqualToString: NSAccessibilityScrollBarRole ] ) {
125 aWrapper = [ [ AquaA11yWrapperScrollBar alloc ] initWithAccessibleContext: rxAccessibleContext ];
126 } else if ( [ nativeRole isEqualToString: NSAccessibilityCheckBoxRole ] ) {
127 aWrapper = [ [ AquaA11yWrapperCheckBox alloc ] initWithAccessibleContext: rxAccessibleContext ];
128 } else if ( [ nativeRole isEqualToString: NSAccessibilityRadioGroupRole ] ) {
129 aWrapper = [ [ AquaA11yWrapperRadioGroup alloc ] initWithAccessibleContext: rxAccessibleContext ];
130 } else if ( [ nativeRole isEqualToString: NSAccessibilityRadioButtonRole ] ) {
131 aWrapper = [ [ AquaA11yWrapperRadioButton alloc ] initWithAccessibleContext: rxAccessibleContext ];
132 } else if ( [ nativeRole isEqualToString: NSAccessibilityRowRole ] ) {
133 aWrapper = [ [ AquaA11yWrapperRow alloc ] initWithAccessibleContext: rxAccessibleContext ];
134 } else if ( [ nativeRole isEqualToString: NSAccessibilityListRole ] ) {
135 aWrapper = [ [ AquaA11yWrapperList alloc ] initWithAccessibleContext: rxAccessibleContext ];
136 } else if ( [ nativeRole isEqualToString: NSAccessibilitySplitterRole ] ) {
137 aWrapper = [ [ AquaA11yWrapperSplitter alloc ] initWithAccessibleContext: rxAccessibleContext ];
138 } else if ( [ nativeRole isEqualToString: NSAccessibilityTableRole ] ) {
139 aWrapper = [ [ AquaA11yTableWrapper alloc ] initWithAccessibleContext: rxAccessibleContext ];
141 aWrapper = [ [ AquaA11yWrapper alloc ] initWithAccessibleContext: rxAccessibleContext ];
143 [ nativeRole release ];
144 [ aWrapper setActsAsRadioGroup: asRadioGroup ];
146 /* #i102033# NSAccessibility does not seemt to know an equivalent for transient children.
147 That means we need to cache this, else e.g. tree list boxes are not accessible (moreover
148 it crashes by notifying dead objects - which would seemt o be another bug)
151 Unfortunately this can increase memory consumption drastically until the non transient parent
152 is destroyed and finally all the transients are released.
154 if ( ! (rxAccessibleContext -> getAccessibleStateSet() & AccessibleStateType::TRANSIENT ) )
157 [ dAllWrapper setObject: aWrapper forKey: nKey ];
163 +(void)insertIntoWrapperRepository: (AquaA11yWrapper *) element forAccessibleContext: (Reference < XAccessibleContext >) rxAccessibleContext {
164 NSMutableDictionary * dAllWrapper = [ AquaA11yFactory allWrapper ];
165 [ dAllWrapper setObject: element forKey: [ AquaA11yFactory keyForAccessibleContext: rxAccessibleContext ] ];
168 +(void)removeFromWrapperRepositoryFor: (css::uno::Reference < css::accessibility::XAccessibleContext >) rxAccessibleContext {
169 // TODO: when RADIO_BUTTON search for associated RadioGroup-wrapper and delete that as well
170 AquaA11yWrapper * theWrapper = [ AquaA11yFactory wrapperForAccessibleContext: rxAccessibleContext createIfNotExists: NO ];
171 if ( theWrapper != nil ) {
172 NSAccessibilityPostNotification( theWrapper, NSAccessibilityUIElementDestroyedNotification );
173 [ [ AquaA11yFactory allWrapper ] removeObjectForKey: [ AquaA11yFactory keyForAccessibleContext: rxAccessibleContext ] ];
174 [ theWrapper release ];
178 +(void)registerWrapper: (AquaA11yWrapper *) theWrapper {
179 if ( enabled && theWrapper ) {
180 // insertIntoWrapperRepository gets called from SalFrameView itself to bootstrap the bridge initially
181 [ theWrapper accessibleContext ];
185 +(void)revokeWrapper: (AquaA11yWrapper *) theWrapper {
186 if ( enabled && theWrapper ) {
187 [ AquaA11yFactory removeFromWrapperRepositoryFor: [ theWrapper accessibleContext ] ];
193 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */