xref: /trunk/main/sw/source/core/access/acchyperlink.cxx (revision cf6516809c57e1bb0a940545cca99cdad54d4ce2)
1efeef26fSAndrew Rist /**************************************************************
2cdf0e10cSrcweir  *
3efeef26fSAndrew Rist  * Licensed to the Apache Software Foundation (ASF) under one
4efeef26fSAndrew Rist  * or more contributor license agreements.  See the NOTICE file
5efeef26fSAndrew Rist  * distributed with this work for additional information
6efeef26fSAndrew Rist  * regarding copyright ownership.  The ASF licenses this file
7efeef26fSAndrew Rist  * to you under the Apache License, Version 2.0 (the
8efeef26fSAndrew Rist  * "License"); you may not use this file except in compliance
9efeef26fSAndrew Rist  * with the License.  You may obtain a copy of the License at
10cdf0e10cSrcweir  *
11efeef26fSAndrew Rist  *   http://www.apache.org/licenses/LICENSE-2.0
12cdf0e10cSrcweir  *
13efeef26fSAndrew Rist  * Unless required by applicable law or agreed to in writing,
14efeef26fSAndrew Rist  * software distributed under the License is distributed on an
15efeef26fSAndrew Rist  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16efeef26fSAndrew Rist  * KIND, either express or implied.  See the License for the
17efeef26fSAndrew Rist  * specific language governing permissions and limitations
18efeef26fSAndrew Rist  * under the License.
19cdf0e10cSrcweir  *
20efeef26fSAndrew Rist  *************************************************************/
21efeef26fSAndrew Rist 
22efeef26fSAndrew Rist 
23cdf0e10cSrcweir 
24cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
25cdf0e10cSrcweir #include "precompiled_sw.hxx"
26cdf0e10cSrcweir #include <comphelper/accessiblekeybindinghelper.hxx>
27cdf0e10cSrcweir #include <swurl.hxx>
28cdf0e10cSrcweir #include <vos/mutex.hxx>
29cdf0e10cSrcweir #include <vcl/svapp.hxx>
30cdf0e10cSrcweir #include <ndtxt.hxx>
31cdf0e10cSrcweir #include <txtinet.hxx>
32cdf0e10cSrcweir #include <accpara.hxx>
33cdf0e10cSrcweir #include <acchyperlink.hxx>
34cdf0e10cSrcweir 
35*ca62e2c2SSteve Yin #include <comphelper/processfactory.hxx>
36*ca62e2c2SSteve Yin #ifndef _COM_SUN_STAR_FRAME_XDESKTOP_HPP_
37*ca62e2c2SSteve Yin #include <com/sun/star/frame/XDesktop.hpp>
38*ca62e2c2SSteve Yin #endif
39*ca62e2c2SSteve Yin #ifndef _COM_SUN_STAR_FRAME_XCOMPONENTLOADER_HPP_
40*ca62e2c2SSteve Yin #include <com/sun/star/frame/XComponentLoader.hpp>
41*ca62e2c2SSteve Yin #endif
42*ca62e2c2SSteve Yin #ifndef _COM_SUN_STAR_DOCUMENT_XLINKTARGETSUPPLIER_HPP_
43*ca62e2c2SSteve Yin #include <com/sun/star/document/XLinkTargetSupplier.hpp>
44*ca62e2c2SSteve Yin #endif
45*ca62e2c2SSteve Yin 
46cdf0e10cSrcweir using namespace ::com::sun::star;
47cdf0e10cSrcweir using namespace ::com::sun::star::accessibility;
48cdf0e10cSrcweir using ::rtl::OUString;
49*ca62e2c2SSteve Yin using ::com::sun::star::lang::IndexOutOfBoundsException;
50cdf0e10cSrcweir 
SwAccessibleHyperlink(sal_uInt16 nHPos,SwAccessibleParagraph * p,sal_Int32 nStt,sal_Int32 nEnd)51cdf0e10cSrcweir SwAccessibleHyperlink::SwAccessibleHyperlink( sal_uInt16 nHPos,
52cdf0e10cSrcweir     SwAccessibleParagraph *p, sal_Int32 nStt, sal_Int32 nEnd ) :
53cdf0e10cSrcweir     nHintPos( nHPos ),
54cdf0e10cSrcweir     xPara( p ),
55cdf0e10cSrcweir     nStartIdx( nStt ),
56cdf0e10cSrcweir     nEndIdx( nEnd )
57cdf0e10cSrcweir {
58cdf0e10cSrcweir }
59cdf0e10cSrcweir 
GetTxtAttr() const60cdf0e10cSrcweir const SwTxtAttr *SwAccessibleHyperlink::GetTxtAttr() const
61cdf0e10cSrcweir {
62cdf0e10cSrcweir     const SwTxtAttr *pTxtAttr = 0;
63cdf0e10cSrcweir     if( xPara.isValid() && xPara->GetMap() )
64cdf0e10cSrcweir     {
65cdf0e10cSrcweir         const SwTxtNode *pTxtNd = xPara->GetTxtNode();
66cdf0e10cSrcweir         const SwpHints *pHints = pTxtNd->GetpSwpHints();
67cdf0e10cSrcweir         if( pHints && nHintPos < pHints->Count() )
68cdf0e10cSrcweir         {
69cdf0e10cSrcweir             const SwTxtAttr *pHt = (*pHints)[nHintPos];
70cdf0e10cSrcweir             if( RES_TXTATR_INETFMT == pHt->Which() )
71cdf0e10cSrcweir                 pTxtAttr = pHt;
72cdf0e10cSrcweir         }
73cdf0e10cSrcweir     }
74cdf0e10cSrcweir 
75cdf0e10cSrcweir     return pTxtAttr;
76cdf0e10cSrcweir }
77cdf0e10cSrcweir 
78cdf0e10cSrcweir 
79cdf0e10cSrcweir // XAccessibleAction
getAccessibleActionCount()80cdf0e10cSrcweir sal_Int32 SAL_CALL SwAccessibleHyperlink::getAccessibleActionCount()
81cdf0e10cSrcweir         throw (uno::RuntimeException)
82cdf0e10cSrcweir {
83cdf0e10cSrcweir      return isValid() ? 1 : 0;
84cdf0e10cSrcweir }
85cdf0e10cSrcweir 
doAccessibleAction(sal_Int32 nIndex)86cdf0e10cSrcweir sal_Bool SAL_CALL SwAccessibleHyperlink::doAccessibleAction( sal_Int32 nIndex )
87cdf0e10cSrcweir         throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
88cdf0e10cSrcweir {
89cdf0e10cSrcweir     vos::OGuard aGuard(Application::GetSolarMutex());
90cdf0e10cSrcweir 
91cdf0e10cSrcweir     sal_Bool bRet = sal_False;
92cdf0e10cSrcweir 
93*ca62e2c2SSteve Yin     if(nIndex != 0)
94*ca62e2c2SSteve Yin         throw new IndexOutOfBoundsException;
95cdf0e10cSrcweir     const SwTxtAttr *pTxtAttr = GetTxtAttr();
96*ca62e2c2SSteve Yin     if( pTxtAttr /*&& 0 == nIndex*/ )
97cdf0e10cSrcweir     {
98cdf0e10cSrcweir         const SwFmtINetFmt& rINetFmt = pTxtAttr->GetINetFmt();
99cdf0e10cSrcweir         if( rINetFmt.GetValue().Len() )
100cdf0e10cSrcweir         {
101cdf0e10cSrcweir             ViewShell *pVSh = xPara->GetShell();
102cdf0e10cSrcweir             if( pVSh )
103cdf0e10cSrcweir             {
104cdf0e10cSrcweir                 LoadURL( rINetFmt.GetValue(), pVSh, URLLOAD_NOFILTER,
105cdf0e10cSrcweir                          &rINetFmt.GetTargetFrame() );
106cdf0e10cSrcweir                 ASSERT( pTxtAttr == rINetFmt.GetTxtINetFmt(),
107cdf0e10cSrcweir                         "lost my txt attr" );
108cdf0e10cSrcweir                 const SwTxtINetFmt* pTxtAttr2 = rINetFmt.GetTxtINetFmt();
109cdf0e10cSrcweir                 if( pTxtAttr2 )
110cdf0e10cSrcweir                 {
111cdf0e10cSrcweir                     const_cast<SwTxtINetFmt*>(pTxtAttr2)->SetVisited(true);
112cdf0e10cSrcweir                     const_cast<SwTxtINetFmt*>(pTxtAttr2)->SetVisitedValid(true);
113cdf0e10cSrcweir                 }
114cdf0e10cSrcweir                 bRet = sal_True;
115cdf0e10cSrcweir             }
116cdf0e10cSrcweir         }
117cdf0e10cSrcweir     }
118cdf0e10cSrcweir 
119cdf0e10cSrcweir     return bRet;
120cdf0e10cSrcweir }
121cdf0e10cSrcweir 
getAccessibleActionDescription(sal_Int32 nIndex)122cdf0e10cSrcweir OUString SAL_CALL SwAccessibleHyperlink::getAccessibleActionDescription(
123cdf0e10cSrcweir         sal_Int32 nIndex )
124cdf0e10cSrcweir         throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
125cdf0e10cSrcweir {
126cdf0e10cSrcweir     OUString sDesc;
127cdf0e10cSrcweir 
128*ca62e2c2SSteve Yin     if(nIndex != 0)
129*ca62e2c2SSteve Yin         throw new IndexOutOfBoundsException;
130cdf0e10cSrcweir     const SwTxtAttr *pTxtAttr = GetTxtAttr();
131*ca62e2c2SSteve Yin     if( pTxtAttr /*&& 0 == nIndex*/ )
132cdf0e10cSrcweir     {
133cdf0e10cSrcweir         const SwFmtINetFmt& rINetFmt = pTxtAttr->GetINetFmt();
134cdf0e10cSrcweir         sDesc = OUString( rINetFmt.GetValue() );
135cdf0e10cSrcweir     }
136cdf0e10cSrcweir     return sDesc;
137cdf0e10cSrcweir }
138cdf0e10cSrcweir 
139cdf0e10cSrcweir uno::Reference< XAccessibleKeyBinding > SAL_CALL
getAccessibleActionKeyBinding(sal_Int32)140*ca62e2c2SSteve Yin     SwAccessibleHyperlink::getAccessibleActionKeyBinding( sal_Int32 )
141cdf0e10cSrcweir     throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
142cdf0e10cSrcweir {
143cdf0e10cSrcweir     uno::Reference< XAccessibleKeyBinding > xKeyBinding;
144cdf0e10cSrcweir 
145*ca62e2c2SSteve Yin     if( isValid() /*&& 0 == nIndex*/ )
146cdf0e10cSrcweir     {
147cdf0e10cSrcweir         ::comphelper::OAccessibleKeyBindingHelper* pKeyBindingHelper =
148cdf0e10cSrcweir             new ::comphelper::OAccessibleKeyBindingHelper();
149cdf0e10cSrcweir         xKeyBinding = pKeyBindingHelper;
150cdf0e10cSrcweir 
151cdf0e10cSrcweir         awt::KeyStroke aKeyStroke;
152cdf0e10cSrcweir         aKeyStroke.Modifiers = 0;
153cdf0e10cSrcweir         aKeyStroke.KeyCode = KEY_RETURN;
154cdf0e10cSrcweir         aKeyStroke.KeyChar = 0;
155cdf0e10cSrcweir         aKeyStroke.KeyFunc = 0;
156cdf0e10cSrcweir         pKeyBindingHelper->AddKeyBinding( aKeyStroke );
157cdf0e10cSrcweir     }
158cdf0e10cSrcweir 
159cdf0e10cSrcweir     return xKeyBinding;
160cdf0e10cSrcweir }
161cdf0e10cSrcweir 
162cdf0e10cSrcweir // XAccessibleHyperlink
getAccessibleActionAnchor(sal_Int32 nIndex)163cdf0e10cSrcweir uno::Any SAL_CALL SwAccessibleHyperlink::getAccessibleActionAnchor(
164*ca62e2c2SSteve Yin         sal_Int32 nIndex)
165cdf0e10cSrcweir         throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
166cdf0e10cSrcweir {
167*ca62e2c2SSteve Yin     uno::Any aRet;
168*ca62e2c2SSteve Yin     if(nIndex != 0)
169*ca62e2c2SSteve Yin         throw new IndexOutOfBoundsException;
170*ca62e2c2SSteve Yin     //End Added.
171*ca62e2c2SSteve Yin     ::rtl::OUString text = OUString( xPara->GetString() );
172*ca62e2c2SSteve Yin     ::rtl::OUString retText =  text.copy(nStartIdx, nEndIdx - nStartIdx);
173*ca62e2c2SSteve Yin     aRet <<= retText;
174*ca62e2c2SSteve Yin     return aRet;
175cdf0e10cSrcweir }
176cdf0e10cSrcweir 
getAccessibleActionObject(sal_Int32 nIndex)177cdf0e10cSrcweir uno::Any SAL_CALL SwAccessibleHyperlink::getAccessibleActionObject(
178*ca62e2c2SSteve Yin             sal_Int32 nIndex )
179cdf0e10cSrcweir     throw (lang::IndexOutOfBoundsException, uno::RuntimeException)
180cdf0e10cSrcweir {
181*ca62e2c2SSteve Yin     if(nIndex != 0)
182*ca62e2c2SSteve Yin         throw new IndexOutOfBoundsException;
183*ca62e2c2SSteve Yin     //End Added.
184*ca62e2c2SSteve Yin     const SwTxtAttr *pTxtAttr = GetTxtAttr();
185*ca62e2c2SSteve Yin     ::rtl::OUString retText;
186*ca62e2c2SSteve Yin     if( pTxtAttr /*&& 0 == nIndex*/ )
187*ca62e2c2SSteve Yin     {
188*ca62e2c2SSteve Yin         const SwFmtINetFmt& rINetFmt = pTxtAttr->GetINetFmt();
189*ca62e2c2SSteve Yin         retText = OUString( rINetFmt.GetValue() );
190*ca62e2c2SSteve Yin     }
191*ca62e2c2SSteve Yin     uno::Any aRet;
192*ca62e2c2SSteve Yin     aRet <<= retText;
193*ca62e2c2SSteve Yin     return aRet;
194cdf0e10cSrcweir }
195cdf0e10cSrcweir 
getStartIndex()196cdf0e10cSrcweir sal_Int32 SAL_CALL SwAccessibleHyperlink::getStartIndex()
197cdf0e10cSrcweir         throw (uno::RuntimeException)
198cdf0e10cSrcweir {
199cdf0e10cSrcweir     return nStartIdx;
200cdf0e10cSrcweir }
201cdf0e10cSrcweir 
getEndIndex()202cdf0e10cSrcweir sal_Int32 SAL_CALL SwAccessibleHyperlink::getEndIndex()
203cdf0e10cSrcweir         throw (uno::RuntimeException)
204cdf0e10cSrcweir {
205cdf0e10cSrcweir     return nEndIdx;
206cdf0e10cSrcweir }
207cdf0e10cSrcweir 
isValid()208cdf0e10cSrcweir sal_Bool SAL_CALL SwAccessibleHyperlink::isValid(  )
209cdf0e10cSrcweir         throw (uno::RuntimeException)
210cdf0e10cSrcweir {
211cdf0e10cSrcweir     vos::OGuard aGuard(Application::GetSolarMutex());
212*ca62e2c2SSteve Yin     //  return xPara.isValid();
213*ca62e2c2SSteve Yin     if (xPara.isValid())
214*ca62e2c2SSteve Yin     {
215*ca62e2c2SSteve Yin         const SwTxtAttr *pTxtAttr = GetTxtAttr();
216*ca62e2c2SSteve Yin         ::rtl::OUString sText;
217*ca62e2c2SSteve Yin         if( pTxtAttr )
218*ca62e2c2SSteve Yin         {
219*ca62e2c2SSteve Yin             const SwFmtINetFmt& rINetFmt = pTxtAttr->GetINetFmt();
220*ca62e2c2SSteve Yin             sText = OUString( rINetFmt.GetValue() );
221*ca62e2c2SSteve Yin             ::rtl::OUString sToken = ::rtl::OUString::createFromAscii("#");
222*ca62e2c2SSteve Yin             sal_Int32 nPos = sText.indexOf(sToken);
223*ca62e2c2SSteve Yin             if (nPos==0)//document link
224*ca62e2c2SSteve Yin             {
225*ca62e2c2SSteve Yin                 uno::Reference< lang::XMultiServiceFactory > xFactory( ::comphelper::getProcessServiceFactory() );
226*ca62e2c2SSteve Yin                 if( ! xFactory.is() )
227*ca62e2c2SSteve Yin                     return sal_False;
228*ca62e2c2SSteve Yin                 uno::Reference< com::sun::star::frame::XDesktop > xDesktop( xFactory->createInstance( OUString::createFromAscii( "com.sun.star.frame.Desktop" ) ),
229*ca62e2c2SSteve Yin                     uno::UNO_QUERY );
230*ca62e2c2SSteve Yin                 if( !xDesktop.is() )
231*ca62e2c2SSteve Yin                     return sal_False;
232*ca62e2c2SSteve Yin                 uno::Reference< lang::XComponent > xComp;
233*ca62e2c2SSteve Yin                 xComp = xDesktop->getCurrentComponent();
234*ca62e2c2SSteve Yin                 if( !xComp.is() )
235*ca62e2c2SSteve Yin                     return sal_False;
236*ca62e2c2SSteve Yin                 uno::Reference< com::sun::star::document::XLinkTargetSupplier >  xLTS(xComp, uno::UNO_QUERY);
237*ca62e2c2SSteve Yin                 if ( !xLTS.is())
238*ca62e2c2SSteve Yin                     return sal_False;
239*ca62e2c2SSteve Yin 
240*ca62e2c2SSteve Yin                 uno::Reference< ::com::sun::star::container::XNameAccess > xLinks = xLTS->getLinks();
241*ca62e2c2SSteve Yin                 uno::Reference< ::com::sun::star::container::XNameAccess > xSubLinks;
242*ca62e2c2SSteve Yin                 const uno::Sequence< OUString > aNames( xLinks->getElementNames() );
243*ca62e2c2SSteve Yin                 const sal_uLong nLinks = aNames.getLength();
244*ca62e2c2SSteve Yin                 const OUString* pNames = aNames.getConstArray();
245*ca62e2c2SSteve Yin 
246*ca62e2c2SSteve Yin                 for( sal_uLong i = 0; i < nLinks; i++ )
247*ca62e2c2SSteve Yin                 {
248*ca62e2c2SSteve Yin                     uno::Any aAny;
249*ca62e2c2SSteve Yin                     OUString aLink( *pNames++ );
250*ca62e2c2SSteve Yin                     aAny = xLinks->getByName( aLink );
251*ca62e2c2SSteve Yin                     aAny >>= xSubLinks;
252*ca62e2c2SSteve Yin                     if (xSubLinks->hasByName(sText.copy(1)) )
253*ca62e2c2SSteve Yin                         return sal_True;
254*ca62e2c2SSteve Yin                 }
255*ca62e2c2SSteve Yin             }
256*ca62e2c2SSteve Yin             else//internet
257*ca62e2c2SSteve Yin                 return sal_True;
258*ca62e2c2SSteve Yin         }
259*ca62e2c2SSteve Yin     }//xpara valid
260*ca62e2c2SSteve Yin     return sal_False;
261cdf0e10cSrcweir }
262cdf0e10cSrcweir 
Invalidate()263cdf0e10cSrcweir void SwAccessibleHyperlink::Invalidate()
264cdf0e10cSrcweir {
265cdf0e10cSrcweir     vos::OGuard aGuard(Application::GetSolarMutex());
266cdf0e10cSrcweir     xPara = 0;
267cdf0e10cSrcweir }
268