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 // MARKER(update_precomp.py): autogen include statement, do not remove 25 #include "precompiled_comphelper.hxx" 26 27 #include <deque> 28 29 #if defined( OS2 ) || defined( UNX ) 30 #include <wchar.h> 31 #endif 32 #include <osl/mutex.hxx> 33 #ifndef _VOS_DIAGNOSE_HXX_ 34 #include <vos/diagnose.hxx> 35 #endif 36 #include <vos/macros.hxx> 37 #include <comphelper/eventattachermgr.hxx> 38 #include <com/sun/star/beans/XIntrospection.hpp> 39 #include <com/sun/star/io/XObjectInputStream.hpp> 40 #include <com/sun/star/io/XPersistObject.hpp> 41 #include <com/sun/star/io/XObjectOutputStream.hpp> 42 #include <com/sun/star/io/XMarkableStream.hpp> 43 #include <com/sun/star/lang/XInitialization.hpp> 44 #include <com/sun/star/lang/XMultiServiceFactory.hpp> 45 #include <com/sun/star/reflection/XIdlClass.hpp> 46 #include <com/sun/star/reflection/XIdlReflection.hpp> 47 #include <com/sun/star/reflection/XIdlMethod.hpp> 48 #include <com/sun/star/script/XTypeConverter.hpp> 49 #include <com/sun/star/script/XEngineListener.hpp> 50 #include <com/sun/star/script/XEventAttacher.hpp> 51 #include <com/sun/star/script/XEventAttacherManager.hpp> 52 #include <com/sun/star/script/XScriptListener.hpp> 53 #include <cppuhelper/weak.hxx> 54 #include <cppuhelper/interfacecontainer.hxx> 55 #include <cppuhelper/implbase1.hxx> 56 #include <cppuhelper/implbase2.hxx> 57 58 using namespace com::sun::star::uno; 59 using namespace com::sun::star::io; 60 using namespace com::sun::star::lang; 61 using namespace com::sun::star::beans; 62 using namespace com::sun::star::script; 63 using namespace com::sun::star::reflection; 64 using namespace cppu; 65 using namespace osl; 66 using namespace rtl; 67 68 namespace comphelper 69 { 70 71 //----------------------------------------------------------------------------- 72 struct AttachedObject_Impl 73 { 74 Reference< XInterface > xTarget; 75 Sequence< Reference< XEventListener > > aAttachedListenerSeq; 76 Any aHelper; 77 78 bool operator<( const AttachedObject_Impl & ) const; 79 bool operator==( const AttachedObject_Impl & ) const; 80 }; 81 82 struct AttacherIndex_Impl 83 { 84 #ifdef DEQUE_OK 85 ::std::deque< ScriptEventDescriptor > aEventList; 86 #else 87 Sequence< ScriptEventDescriptor > aEventList; 88 #endif 89 ::std::deque< AttachedObject_Impl > aObjList; 90 91 bool operator<( const AttacherIndex_Impl & ) const; 92 bool operator==( const AttacherIndex_Impl & ) const; 93 }; 94 95 #if 0 96 bool AttachedObject_Impl::operator<( const AttachedObject_Impl & r ) const 97 { 98 VOS_ENSHURE( FALSE, "not implemented" ); 99 return FALSE; 100 return this < &r; 101 } 102 103 bool AttachedObject_Impl::operator==( const AttachedObject_Impl & r ) const 104 { 105 VOS_ENSHURE( FALSE, "not implemented" ); 106 return this == &r; 107 } 108 109 bool AttacherIndex_Impl::operator<( const AttacherIndex_Impl & r ) const 110 { 111 VOS_ENSHURE( FALSE, "not implemented" ); 112 return this < &r; 113 } 114 bool AttacherIndex_Impl::operator==( const AttacherIndex_Impl & r ) const 115 { 116 VOS_ENSHURE( FALSE, "not implemented" ); 117 return this == &r; 118 } 119 #endif 120 121 //----------------------------------------------------------------------------- 122 class ImplEventAttacherManager 123 : public WeakImplHelper2< XEventAttacherManager, XPersistObject > 124 { 125 friend class AttacherAllListener_Impl; 126 ::std::deque< AttacherIndex_Impl > aIndex; 127 Mutex aLock; 128 // Container fuer die ScriptListener 129 OInterfaceContainerHelper aScriptListeners; 130 // EventAttacher-Instanz 131 Reference< XEventAttacher > xAttacher; 132 Reference< XMultiServiceFactory > mxSMgr; 133 Reference< XIdlReflection > mxCoreReflection; 134 Reference< XIntrospection > mxIntrospection; 135 Reference< XTypeConverter > xConverter; 136 sal_Int16 nVersion; 137 public: 138 ImplEventAttacherManager( const Reference< XIntrospection > & rIntrospection, 139 const Reference< XMultiServiceFactory > rSMgr ); 140 ~ImplEventAttacherManager(); 141 142 // Methoden von XEventAttacherManager 143 virtual void SAL_CALL registerScriptEvent(sal_Int32 Index, const ScriptEventDescriptor& ScriptEvent) 144 throw( IllegalArgumentException, RuntimeException ); 145 virtual void SAL_CALL registerScriptEvents(sal_Int32 Index, const Sequence< ScriptEventDescriptor >& ScriptEvents) 146 throw( IllegalArgumentException, RuntimeException ); 147 virtual void SAL_CALL revokeScriptEvent(sal_Int32 Index, const OUString& ListenerType, const OUString& EventMethod, const OUString& removeListenerParam) 148 throw( IllegalArgumentException, RuntimeException ); 149 virtual void SAL_CALL revokeScriptEvents(sal_Int32 Index) 150 throw( IllegalArgumentException, RuntimeException ); 151 virtual void SAL_CALL insertEntry(sal_Int32 Index) 152 throw( IllegalArgumentException, RuntimeException ); 153 virtual void SAL_CALL removeEntry(sal_Int32 Index) 154 throw( IllegalArgumentException, RuntimeException ); 155 virtual Sequence< ScriptEventDescriptor > SAL_CALL getScriptEvents(sal_Int32 Index) 156 throw( IllegalArgumentException, RuntimeException ); 157 virtual void SAL_CALL attach(sal_Int32 Index, const Reference< XInterface >& Object, const Any& Helper) 158 throw( IllegalArgumentException, ServiceNotRegisteredException, RuntimeException ); 159 virtual void SAL_CALL detach(sal_Int32 nIndex, const Reference< XInterface >& xObject) 160 throw( IllegalArgumentException, RuntimeException ); 161 virtual void SAL_CALL addScriptListener(const Reference< XScriptListener >& aListener) 162 throw( IllegalArgumentException, RuntimeException ); 163 virtual void SAL_CALL removeScriptListener(const Reference< XScriptListener >& Listener) 164 throw( IllegalArgumentException, RuntimeException ); 165 166 // Methoden von XPersistObject 167 virtual OUString SAL_CALL getServiceName(void) throw( RuntimeException ); 168 virtual void SAL_CALL write(const Reference< XObjectOutputStream >& OutStream) throw( IOException, RuntimeException ); 169 virtual void SAL_CALL read(const Reference< XObjectInputStream >& InStream) throw( IOException, RuntimeException ); 170 171 private: 172 Reference< XIdlReflection > getReflection() throw( Exception ); 173 174 /** checks if <arg>_nIndex</arg> is a valid index, throws an <type>IllegalArgumentException</type> if not 175 @param _nIndex 176 the index to check 177 @return 178 the iterator pointing to the position indicated by the index 179 */ 180 ::std::deque<AttacherIndex_Impl>::iterator implCheckIndex( sal_Int32 _nIndex ) SAL_THROW ( ( IllegalArgumentException ) ); 181 }; 182 183 //======================================================================== 184 //======================================================================== 185 //======================================================================== 186 187 // Implementation eines EventAttacher-bezogenen AllListeners, der 188 // nur einzelne Events an einen allgemeinen AllListener weiterleitet 189 class AttacherAllListener_Impl : public WeakImplHelper1< XAllListener > 190 { 191 ImplEventAttacherManager* mpManager; 192 Reference< XEventAttacherManager > xManager; 193 OUString aScriptType; 194 OUString aScriptCode; 195 sal_Int16 nVersion; 196 197 void convertToEventReturn( Any & rRet, const Type & rRetType ) 198 throw( CannotConvertException ); 199 public: 200 AttacherAllListener_Impl( ImplEventAttacherManager* pManager_, const OUString &rScriptType_, 201 const OUString & rScriptCode_ ); 202 203 // Methoden von XAllListener 204 virtual void SAL_CALL firing(const AllEventObject& Event) throw( RuntimeException ); 205 virtual Any SAL_CALL approveFiring(const AllEventObject& Event) throw( InvocationTargetException, RuntimeException ); 206 207 // Methoden von XEventListener 208 virtual void SAL_CALL disposing(const EventObject& Source) throw( RuntimeException ); 209 }; 210 211 //======================================================================== 212 AttacherAllListener_Impl::AttacherAllListener_Impl 213 ( 214 ImplEventAttacherManager* pManager_, 215 const OUString & rScriptType_, 216 const OUString & rScriptCode_ 217 ) 218 : mpManager( pManager_ ) 219 , xManager( pManager_ ) 220 , aScriptType( rScriptType_ ) 221 , aScriptCode( rScriptCode_ ) 222 , nVersion( 2 ) 223 { 224 } 225 226 227 //======================================================================== 228 // Methoden von XAllListener 229 void SAL_CALL AttacherAllListener_Impl::firing(const AllEventObject& Event) 230 throw( RuntimeException ) 231 { 232 ScriptEvent aScriptEvent; 233 aScriptEvent.Source = (OWeakObject *)mpManager; // get correct XInterface 234 aScriptEvent.ListenerType = Event.ListenerType; 235 aScriptEvent.MethodName = Event.MethodName; 236 aScriptEvent.Arguments = Event.Arguments; 237 aScriptEvent.Helper = Event.Helper; 238 aScriptEvent.ScriptType = aScriptType; 239 aScriptEvent.ScriptCode = aScriptCode; 240 241 // ueber alle Listener iterieren und Events senden 242 OInterfaceIteratorHelper aIt( mpManager->aScriptListeners ); 243 while( aIt.hasMoreElements() ) 244 ((XScriptListener *)aIt.next())->firing( aScriptEvent ); 245 } 246 247 //======================================================================== 248 // Convert to the standard event return 249 void AttacherAllListener_Impl::convertToEventReturn( Any & rRet, const Type & rRetType ) 250 throw( CannotConvertException ) 251 { 252 // no return value? Set to the specified values 253 if( rRet.getValueType().getTypeClass() == TypeClass_VOID ) 254 { 255 switch( rRetType.getTypeClass() ) 256 { 257 case TypeClass_INTERFACE: 258 { 259 rRet <<= Reference< XInterface >(); 260 } 261 break; 262 263 case TypeClass_BOOLEAN: 264 rRet <<= sal_True; 265 break; 266 267 case TypeClass_STRING: 268 rRet <<= OUString(); 269 break; 270 271 case TypeClass_FLOAT: rRet <<= float(0); break; 272 case TypeClass_DOUBLE: rRet <<= double(0.0); break; 273 case TypeClass_BYTE: rRet <<= sal_uInt8(0); break; 274 case TypeClass_SHORT: rRet <<= sal_Int16( 0 ); break; 275 case TypeClass_LONG: rRet <<= sal_Int32( 0 ); break; 276 case TypeClass_UNSIGNED_SHORT: rRet <<= sal_uInt16( 0 ); break; 277 case TypeClass_UNSIGNED_LONG: rRet <<= sal_uInt32( 0 ); break; 278 279 default: 280 OSL_ASSERT(false); 281 break; 282 } 283 } 284 else if( !rRet.getValueType().equals( rRetType ) ) 285 { 286 if( mpManager->xConverter.is() ) 287 rRet = mpManager->xConverter->convertTo( rRet, rRetType ); 288 else 289 throw CannotConvertException(); 290 } 291 } 292 293 //======================================================================== 294 // Methoden von XAllListener 295 Any SAL_CALL AttacherAllListener_Impl::approveFiring( const AllEventObject& Event ) 296 throw( InvocationTargetException, RuntimeException ) 297 { 298 ScriptEvent aScriptEvent; 299 aScriptEvent.Source = (OWeakObject *)mpManager; // get correct XInterface 300 aScriptEvent.ListenerType = Event.ListenerType; 301 aScriptEvent.MethodName = Event.MethodName; 302 aScriptEvent.Arguments = Event.Arguments; 303 aScriptEvent.Helper = Event.Helper; 304 aScriptEvent.ScriptType = aScriptType; 305 aScriptEvent.ScriptCode = aScriptCode; 306 307 Any aRet; 308 // ueber alle Listener iterieren und Events senden 309 OInterfaceIteratorHelper aIt( mpManager->aScriptListeners ); 310 while( aIt.hasMoreElements() ) 311 { 312 aRet = ((XScriptListener *)aIt.next())->approveFiring( aScriptEvent ); 313 try 314 { 315 Reference< XIdlClass > xListenerType = mpManager->getReflection()-> 316 forName( Event.ListenerType.getTypeName() ); 317 Reference< XIdlMethod > xMeth = xListenerType->getMethod( Event.MethodName ); 318 if( xMeth.is() ) 319 { 320 Reference< XIdlClass > xRetType = xMeth->getReturnType(); 321 Type aRetType(xRetType->getTypeClass(), xRetType->getName()); 322 convertToEventReturn( aRet, aRetType ); 323 } 324 325 switch( aRet.getValueType().getTypeClass() ) 326 { 327 case TypeClass_INTERFACE: 328 { 329 // Interface not null, return 330 Reference< XInterface > x; 331 aRet >>= x; 332 if( x.is() ) 333 return aRet; 334 } 335 break; 336 337 case TypeClass_BOOLEAN: 338 // FALSE -> Return 339 if( !(*(sal_Bool*)aRet.getValue()) ) 340 return aRet; 341 break; 342 343 case TypeClass_STRING: 344 // none empty string -> return 345 if( ((OUString*)aRet.getValue())->isEmpty() == false ) 346 return aRet; 347 break; 348 349 // none zero number -> return 350 case TypeClass_FLOAT: if( *((float*)aRet.getValue()) ) return aRet; break; 351 case TypeClass_DOUBLE: if( *((double*)aRet.getValue()) ) return aRet; break; 352 case TypeClass_BYTE: if( *((sal_uInt8*)aRet.getValue()) ) return aRet; break; 353 case TypeClass_SHORT: if( *((sal_Int16*)aRet.getValue()) ) return aRet; break; 354 case TypeClass_LONG: if( *((sal_Int32*)aRet.getValue()) ) return aRet; break; 355 case TypeClass_UNSIGNED_SHORT: if( *((sal_uInt16*)aRet.getValue()) ) return aRet; break; 356 case TypeClass_UNSIGNED_LONG: if( *((sal_uInt32*)aRet.getValue()) ) return aRet; break; 357 358 default: 359 OSL_ASSERT(false); 360 break; 361 } 362 } 363 catch( CannotConvertException& ) 364 { 365 // silent ignore conversions errors from a script call 366 Reference< XIdlClass > xListenerType = mpManager->getReflection()-> 367 forName( Event.ListenerType.getTypeName() ); 368 Reference< XIdlMethod > xMeth = xListenerType->getMethod( Event.MethodName ); 369 if( xMeth.is() ) 370 { 371 Reference< XIdlClass > xRetType = xMeth->getReturnType(); 372 Type aRetType(xRetType->getTypeClass(), xRetType->getName()); 373 aRet.clear(); 374 convertToEventReturn( aRet, aRetType ); 375 } 376 } 377 } 378 return aRet; 379 } 380 381 //======================================================================== 382 // Methoden von XEventListener 383 void SAL_CALL AttacherAllListener_Impl::disposing(const EventObject& ) 384 throw( RuntimeException ) 385 { 386 // It is up to the container to release the object 387 } 388 389 390 //======================================================================== 391 //======================================================================== 392 //======================================================================== 393 394 // Create-Methode fuer EventAttacherManager 395 Reference< XEventAttacherManager > createEventAttacherManager( const Reference< XIntrospection > & rIntrospection, 396 const Reference< XMultiServiceFactory > & rSMgr ) 397 throw( Exception ) 398 { 399 return new ImplEventAttacherManager( rIntrospection, rSMgr ); 400 } 401 402 // Create-Methode fuer EventAttacherManager 403 Reference< XEventAttacherManager > createEventAttacherManager( const Reference< XMultiServiceFactory > & rSMgr ) 404 throw( Exception ) 405 { 406 if ( rSMgr.is() ) 407 { 408 Reference< XInterface > xIFace( rSMgr->createInstance( OUString::createFromAscii("com.sun.star.beans.Introspection") ) ); 409 if ( xIFace.is() ) 410 { 411 Reference< XIntrospection > xIntrospection( xIFace, UNO_QUERY); 412 return new ImplEventAttacherManager( xIntrospection, rSMgr ); 413 } 414 } 415 416 return Reference< XEventAttacherManager >(); 417 } 418 419 //----------------------------------------------------------------------------- 420 ImplEventAttacherManager::ImplEventAttacherManager( const Reference< XIntrospection > & rIntrospection, 421 const Reference< XMultiServiceFactory > rSMgr ) 422 : aScriptListeners( aLock ) 423 , mxSMgr( rSMgr ) 424 , mxIntrospection( rIntrospection ) 425 { 426 if ( rSMgr.is() ) 427 { 428 Reference< XInterface > xIFace( rSMgr->createInstance( OUString::createFromAscii("com.sun.star.script.EventAttacher") ) ); 429 if ( xIFace.is() ) 430 { 431 xAttacher = Reference< XEventAttacher >::query( xIFace ); 432 } 433 xIFace = rSMgr->createInstance( OUString::createFromAscii("com.sun.star.script.Converter") ); 434 if ( xIFace.is() ) 435 { 436 xConverter = Reference< XTypeConverter >::query( xIFace ); 437 } 438 } 439 440 Reference< XInitialization > xInit( xAttacher, UNO_QUERY ); 441 if( xInit.is() ) 442 { 443 Sequence< Any > Arguments( 1 ); 444 Arguments[0] <<= rIntrospection; 445 xInit->initialize( Arguments ); 446 } 447 } 448 449 //----------------------------------------------------------------------------- 450 ImplEventAttacherManager::~ImplEventAttacherManager() 451 { 452 } 453 454 Reference< XIdlReflection > ImplEventAttacherManager::getReflection() throw( Exception ) 455 { 456 Guard< Mutex > aGuard( aLock ); 457 // Haben wir den Service schon? Sonst anlegen 458 if( !mxCoreReflection.is() ) 459 { 460 Reference< XInterface > xIFace( mxSMgr->createInstance( OUString::createFromAscii("com.sun.star.reflection.CoreReflection") ) ); 461 mxCoreReflection = Reference< XIdlReflection >( xIFace, UNO_QUERY); 462 } 463 return mxCoreReflection; 464 } 465 466 467 //----------------------------------------------------------------------------- 468 ::std::deque<AttacherIndex_Impl>::iterator ImplEventAttacherManager::implCheckIndex( sal_Int32 _nIndex ) SAL_THROW ( ( IllegalArgumentException ) ) 469 { 470 if (_nIndex < 0) 471 throw IllegalArgumentException(); 472 473 ::std::deque<AttacherIndex_Impl>::iterator aIt = aIndex.begin(); 474 for ( sal_Int32 i = 0; (i < _nIndex) && (aIt != aIndex.end()); ++i, ++aIt ) 475 ; 476 477 if( aIt == aIndex.end() ) 478 throw IllegalArgumentException(); 479 480 return aIt; 481 } 482 483 //----------------------------------------------------------------------------- 484 void detachAll_Impl 485 ( 486 ImplEventAttacherManager * pMgr, 487 sal_Int32 nIdx, 488 ::std::deque< AttachedObject_Impl > & rList 489 ) 490 { 491 ::std::deque< AttachedObject_Impl >::iterator aObjIt = rList.begin(); 492 ::std::deque< AttachedObject_Impl >::iterator aObjEnd = rList.end(); 493 while( aObjIt != aObjEnd ) 494 { 495 pMgr->detach( nIdx, (*aObjIt).xTarget ); 496 aObjIt++; 497 } 498 } 499 500 //----------------------------------------------------------------------------- 501 void attachAll_Impl 502 ( 503 ImplEventAttacherManager * pMgr, 504 sal_Int32 nIdx, 505 ::std::deque< AttachedObject_Impl > & rList 506 ) 507 { 508 ::std::deque< AttachedObject_Impl >::iterator aObjIt = rList.begin(); 509 ::std::deque< AttachedObject_Impl >::iterator aObjEnd = rList.end(); 510 while( aObjIt != aObjEnd ) 511 { 512 pMgr->attach( nIdx, (*aObjIt).xTarget, (*aObjIt).aHelper ); 513 aObjIt++; 514 } 515 } 516 517 //----------------------------------------------------------------------------- 518 //*** Methoden von XEventAttacherManager *** 519 void SAL_CALL ImplEventAttacherManager::registerScriptEvent 520 ( 521 sal_Int32 nIndex, 522 const ScriptEventDescriptor& ScriptEvent 523 ) 524 throw( IllegalArgumentException, RuntimeException ) 525 { 526 Guard< Mutex > aGuard( aLock ); 527 528 // Index pruefen und Array anpassen 529 ::std::deque<AttacherIndex_Impl>::iterator aIt = implCheckIndex( nIndex ); 530 531 ::std::deque< AttachedObject_Impl > aList = (*aIt).aObjList; 532 533 ScriptEventDescriptor aEvt = ScriptEvent; 534 const sal_Unicode* pLastDot = aEvt.ListenerType.getStr(); 535 pLastDot += rtl_ustr_lastIndexOfChar( pLastDot, '.' ); 536 if( pLastDot ) 537 aEvt.ListenerType = pLastDot +1; 538 #ifdef DEQUE_OK 539 (*aIt).aEventList.push_back( aEvt ); 540 #else 541 (*aIt).aEventList.realloc( (*aIt).aEventList.getLength() +1 ); 542 (*aIt).aEventList.getArray()[(*aIt).aEventList.getLength() -1] = aEvt; 543 #endif 544 545 // register new new Event 546 ::std::deque< AttachedObject_Impl >::iterator aObjIt = (*aIt).aObjList.begin(); 547 ::std::deque< AttachedObject_Impl >::iterator aObjEnd = (*aIt).aObjList.end(); 548 while( aObjIt != aObjEnd ) 549 { 550 // resize 551 sal_Int32 nPos = (*aObjIt).aAttachedListenerSeq.getLength(); 552 (*aObjIt).aAttachedListenerSeq.realloc( nPos + 1 ); 553 Reference< XEventListener > * pArray = (*aObjIt).aAttachedListenerSeq.getArray(); 554 555 Reference< XAllListener > xAll = 556 new AttacherAllListener_Impl( this, ScriptEvent.ScriptType, ScriptEvent.ScriptCode ); 557 try 558 { 559 pArray[nPos] = xAttacher->attachSingleEventListener( (*aObjIt).xTarget, xAll, 560 (*aObjIt).aHelper, ScriptEvent.ListenerType, 561 ScriptEvent.AddListenerParam, ScriptEvent.EventMethod ); 562 } 563 catch( Exception& ) 564 { 565 } 566 567 aObjIt++; 568 } 569 } 570 571 //----------------------------------------------------------------------------- 572 void SAL_CALL ImplEventAttacherManager::registerScriptEvents 573 ( 574 sal_Int32 nIndex, 575 const Sequence< ScriptEventDescriptor >& ScriptEvents 576 ) 577 throw( IllegalArgumentException, RuntimeException ) 578 { 579 Guard< Mutex > aGuard( aLock ); 580 581 // Index pruefen und Array anpassen 582 ::std::deque<AttacherIndex_Impl>::iterator aIt = implCheckIndex( nIndex ); 583 584 ::std::deque< AttachedObject_Impl > aList = (*aIt).aObjList; 585 detachAll_Impl( this, nIndex, aList ); 586 587 const ScriptEventDescriptor* pArray = ScriptEvents.getConstArray(); 588 sal_Int32 nLen = ScriptEvents.getLength(); 589 for( sal_Int32 i = 0 ; i < nLen ; i++ ) 590 registerScriptEvent( nIndex, pArray[ i ] ); 591 592 attachAll_Impl( this, nIndex, aList ); 593 } 594 595 //----------------------------------------------------------------------------- 596 void SAL_CALL ImplEventAttacherManager::revokeScriptEvent 597 ( 598 sal_Int32 nIndex, 599 const OUString& ListenerType, 600 const OUString& EventMethod, 601 const OUString& ToRemoveListenerParam 602 ) 603 throw( IllegalArgumentException, RuntimeException ) 604 { 605 Guard< Mutex > aGuard( aLock ); 606 607 ::std::deque<AttacherIndex_Impl>::iterator aIt = implCheckIndex( nIndex ); 608 609 ::std::deque< AttachedObject_Impl > aList = (*aIt).aObjList; 610 detachAll_Impl( this, nIndex, aList ); 611 612 OUString aLstType = ListenerType; 613 const sal_Unicode * pLastDot = aLstType.getStr(); 614 pLastDot += rtl_ustr_lastIndexOfChar( pLastDot, '.' ); 615 if( pLastDot ) 616 aLstType = pLastDot +1; 617 618 #ifdef DEQUE_OK 619 ::std::deque< ScriptEventDescriptor >::iterator aEvtIt = (*aIt).aEventList.begin(); 620 ::std::deque< ScriptEventDescriptor >::iterator aEvtEnd = (*aIt).aEventList.end(); 621 while( aEvtIt != aEvtEnd ) 622 { 623 if( aLstType == (*aEvtIt).ListenerType 624 && EventMethod == (*aEvtIt).EventMethod 625 && ToRemoveListenerParam == (*aEvtIt).AddListenerParam ) 626 { 627 (*aIt).aEventList.erase( aEvtIt ); 628 break; 629 } 630 631 aEvtIt++; 632 } 633 #else 634 Sequence< ScriptEventDescriptor >& rEventList = (*aIt).aEventList; 635 636 ScriptEventDescriptor* pEventList = rEventList.getArray(); 637 const ScriptEventDescriptor* pEventListEnd = pEventList + rEventList.getLength(); 638 for( ; pEventList < pEventListEnd; ++pEventList ) 639 { 640 if ( (aLstType == pEventList->ListenerType ) 641 && (EventMethod == pEventList->EventMethod ) 642 && (ToRemoveListenerParam == pEventList->AddListenerParam) 643 ) 644 { 645 ScriptEventDescriptor* pMoveTo = pEventList; 646 const ScriptEventDescriptor* pMoveFrom = pMoveTo + 1; 647 while (pMoveFrom < pEventListEnd) 648 { 649 *pMoveTo++ = *pMoveFrom++; 650 } 651 rEventList.realloc( rEventList.getLength() - 1 ); 652 break; 653 } 654 } 655 #endif 656 attachAll_Impl( this, nIndex, aList ); 657 } 658 659 //----------------------------------------------------------------------------- 660 void SAL_CALL ImplEventAttacherManager::revokeScriptEvents(sal_Int32 nIndex ) 661 throw( IllegalArgumentException, RuntimeException ) 662 { 663 Guard< Mutex > aGuard( aLock ); 664 ::std::deque<AttacherIndex_Impl>::iterator aIt = implCheckIndex( nIndex ); 665 666 ::std::deque< AttachedObject_Impl > aList = (*aIt).aObjList; 667 detachAll_Impl( this, nIndex, aList ); 668 #ifdef DEQUE_OK 669 (*aIt).aEventList = ::std::deque< ScriptEventDescriptor >(); 670 #else 671 (*aIt).aEventList.realloc( 0 ); 672 #endif 673 attachAll_Impl( this, nIndex, aList ); 674 } 675 676 //----------------------------------------------------------------------------- 677 void SAL_CALL ImplEventAttacherManager::insertEntry(sal_Int32 nIndex) 678 throw( IllegalArgumentException, RuntimeException ) 679 { 680 Guard< Mutex > aGuard( aLock ); 681 if( nIndex < 0 ) 682 throw IllegalArgumentException(); 683 684 // ::std::deque<AttacherIndex_Impl>::iterator aIt = aIndex.begin(); 685 // while( nIndex-- ) 686 // aIt++; 687 688 if ( static_cast< ::std::deque< AttacherIndex_Impl >::size_type>(nIndex) >= aIndex.size() ) 689 aIndex.resize(nIndex+1); 690 691 AttacherIndex_Impl aTmp; 692 aIndex.insert( aIndex.begin() + nIndex, aTmp ); 693 } 694 695 //----------------------------------------------------------------------------- 696 void SAL_CALL ImplEventAttacherManager::removeEntry(sal_Int32 nIndex) 697 throw( IllegalArgumentException, RuntimeException ) 698 { 699 Guard< Mutex > aGuard( aLock ); 700 ::std::deque<AttacherIndex_Impl>::iterator aIt = implCheckIndex( nIndex ); 701 702 ::std::deque< AttachedObject_Impl > aList = (*aIt).aObjList; 703 detachAll_Impl( this, nIndex, aList ); 704 aIndex.erase( aIt ); 705 } 706 707 //----------------------------------------------------------------------------- 708 Sequence< ScriptEventDescriptor > SAL_CALL ImplEventAttacherManager::getScriptEvents(sal_Int32 nIndex) 709 throw( IllegalArgumentException, RuntimeException ) 710 { 711 Guard< Mutex > aGuard( aLock ); 712 ::std::deque<AttacherIndex_Impl>::iterator aIt = implCheckIndex( nIndex ); 713 714 #ifdef DEQUE_OK 715 Sequence< ScriptEventDescriptor > aSeq( (*aIt).aEventList.size() ); 716 ScriptEventDescriptor * pArray = aSeq.getArray(); 717 718 ::std::deque< ScriptEventDescriptor >::iterator aEvtIt = (*aIt).aEventList.begin(); 719 ::std::deque< ScriptEventDescriptor >::iterator aEvtEnd = (*aIt).aEventList.end(); 720 sal_Int32 i = 0; 721 while( aEvtIt != aEvtEnd ) 722 { 723 pArray[i++] = *aEvtIt; 724 aEvtIt++; 725 } 726 return aSeq; 727 #else 728 return (*aIt).aEventList; 729 #endif 730 } 731 732 //----------------------------------------------------------------------------- 733 void SAL_CALL ImplEventAttacherManager::attach(sal_Int32 nIndex, const Reference< XInterface >& xObject, const Any & Helper) 734 throw( IllegalArgumentException, ServiceNotRegisteredException, RuntimeException ) 735 { 736 Guard< Mutex > aGuard( aLock ); 737 if( nIndex < 0 || !xObject.is() ) 738 throw IllegalArgumentException(); 739 740 if( static_cast< ::std::deque< AttacherIndex_Impl >::size_type>(nIndex) >= aIndex.size() ) 741 { 742 // alte Dateien lesen 743 if( nVersion == 1 ) 744 { 745 insertEntry( nIndex ); 746 attach( nIndex, xObject, Helper ); 747 return; 748 } 749 else 750 throw IllegalArgumentException(); 751 } 752 753 ::std::deque< AttacherIndex_Impl >::iterator aCurrentPosition = aIndex.begin() + nIndex; 754 755 AttachedObject_Impl aTmp; 756 aTmp.xTarget = xObject; 757 aTmp.aHelper = Helper; 758 aCurrentPosition->aObjList.push_back( aTmp ); 759 760 //::std::deque< AttachedObject_Impl >::iterator aObjIt = (*aIt).aObjList.back(); 761 AttachedObject_Impl & rCurObj = aCurrentPosition->aObjList.back(); 762 #ifdef DEQUE_OK 763 rCurObj.aAttachedListenerSeq = Sequence< Reference< XEventListener > >( aCurrentPosition->aEventList.size() ); 764 #else 765 rCurObj.aAttachedListenerSeq = Sequence< Reference< XEventListener > >( aCurrentPosition->aEventList.getLength() ); 766 #endif 767 Reference< XEventListener > * pArray = rCurObj.aAttachedListenerSeq.getArray(); 768 769 #ifdef DEQUE_OK 770 ::std::deque< ScriptEventDescriptor >::iterator aEvtIt = aCurrentPosition->aEventList.begin(); 771 ::std::deque< ScriptEventDescriptor >::iterator aEvtEnd = aCurrentPosition->aEventList.end(); 772 sal_Int32 i = 0; 773 while( aEvtIt != aEvtEnd ) 774 { 775 Reference< XAllListener > xAll = 776 new AttacherAllListener_Impl( this, (*aEvtIt).ScriptType, (*aEvtIt).ScriptCode ); 777 Reference< XEventListener > xAdapter; 778 try 779 { 780 xAdapter = xAttacher->attachSingleEventListener( rCurObj.xTarget, xAll, 781 rCurObj.aHelper, (*aEvtIt).ScriptType, 782 (*aEvtIt).AddListenerParam, (*aEvtIt).EventMethod ); 783 } 784 catch( Exception& ) 785 { 786 } 787 788 pArray[i++] = xAdapter; 789 aEvtIt++; 790 } 791 #else 792 sal_Int32 nLen = aCurrentPosition->aEventList.getLength(); 793 ScriptEventDescriptor * pEL = aCurrentPosition->aEventList.getArray(); 794 for(sal_Int32 i = 0; i < nLen; ++i ) 795 { 796 Reference< XAllListener > xAll = 797 new AttacherAllListener_Impl( this, pEL[i].ScriptType, pEL[i].ScriptCode ); 798 Reference< XEventListener > xAdapter; 799 try 800 { 801 xAdapter = xAttacher->attachSingleEventListener( rCurObj.xTarget, xAll, 802 rCurObj.aHelper, pEL[i].ListenerType, 803 pEL[i].AddListenerParam, pEL[i].EventMethod ); 804 } 805 catch( Exception& ) 806 { 807 } 808 809 pArray[i] = xAdapter; 810 } 811 #endif 812 } 813 814 //----------------------------------------------------------------------------- 815 void SAL_CALL ImplEventAttacherManager::detach(sal_Int32 nIndex, const Reference< XInterface >& xObject) 816 throw( IllegalArgumentException, RuntimeException ) 817 { 818 Guard< Mutex > aGuard( aLock ); 819 //return; 820 if( nIndex < 0 || static_cast< ::std::deque< AttacherIndex_Impl >::size_type>(nIndex) >= aIndex.size() || !xObject.is() ) 821 throw IllegalArgumentException(); 822 823 ::std::deque< AttacherIndex_Impl >::iterator aCurrentPosition = aIndex.begin() + nIndex; 824 ::std::deque< AttachedObject_Impl >::iterator aObjIt = aCurrentPosition->aObjList.begin(); 825 ::std::deque< AttachedObject_Impl >::iterator aObjEnd = aCurrentPosition->aObjList.end(); 826 while( aObjIt != aObjEnd ) 827 { 828 if( (*aObjIt).xTarget == xObject ) 829 { 830 Reference< XEventListener > * pArray = (*aObjIt).aAttachedListenerSeq.getArray(); 831 #ifdef DEQUE_OK 832 833 ::std::deque< ScriptEventDescriptor >::iterator aEvtIt = aCurrentPosition->aEventList.begin(); 834 ::std::deque< ScriptEventDescriptor >::iterator aEvtEnd = aCurrentPosition->aEventList.end(); 835 sal_Int32 i = 0; 836 while( aEvtIt != aEvtEnd ) 837 { 838 if( pArray[i].is() ) 839 { 840 try 841 { 842 xAttacher->removeListener( (*aObjIt).xTarget, (*aEvtIt).ListenerType, 843 (*aEvtIt).AddListenerParam, pArray[i] ); 844 } 845 catch( Exception& ) 846 { 847 } 848 } 849 i++; 850 aEvtIt++; 851 } 852 #else 853 sal_Int32 nLen = aCurrentPosition->aEventList.getLength(); 854 ScriptEventDescriptor * pEL = aCurrentPosition->aEventList.getArray(); 855 for( sal_Int32 i = 0; i < nLen; i++ ) 856 { 857 if( pArray[i].is() ) 858 { 859 try 860 { 861 xAttacher->removeListener( (*aObjIt).xTarget, pEL[i].ListenerType, 862 pEL[i].AddListenerParam, pArray[i] ); 863 } 864 catch( Exception& ) 865 { 866 } 867 } 868 } 869 #endif 870 aCurrentPosition->aObjList.erase( aObjIt ); 871 break; 872 } 873 aObjIt++; 874 } 875 } 876 877 void SAL_CALL ImplEventAttacherManager::addScriptListener(const Reference< XScriptListener >& aListener) 878 throw( IllegalArgumentException, RuntimeException ) 879 { 880 Guard< Mutex > aGuard( aLock ); 881 aScriptListeners.addInterface( aListener ); 882 } 883 884 void SAL_CALL ImplEventAttacherManager::removeScriptListener(const Reference< XScriptListener >& aListener) 885 throw( IllegalArgumentException, RuntimeException ) 886 { 887 Guard< Mutex > aGuard( aLock ); 888 aScriptListeners.removeInterface( aListener ); 889 } 890 891 892 // Methoden von XPersistObject 893 OUString SAL_CALL ImplEventAttacherManager::getServiceName(void) 894 throw( RuntimeException ) 895 { 896 return OUString( RTL_CONSTASCII_USTRINGPARAM("com.sun.star.uno.script.EventAttacherManager") ); 897 } 898 899 void SAL_CALL ImplEventAttacherManager::write(const Reference< XObjectOutputStream >& OutStream) 900 throw( IOException, RuntimeException ) 901 { 902 Guard< Mutex > aGuard( aLock ); 903 // Ohne XMarkableStream laeuft nichts 904 Reference< XMarkableStream > xMarkStream( OutStream, UNO_QUERY ); 905 if( !xMarkStream.is() ) 906 return; 907 908 // Version schreiben 909 OutStream->writeShort( 2 ); 910 911 // Position fuer Laenge merken 912 sal_Int32 nObjLenMark = xMarkStream->createMark(); 913 OutStream->writeLong( 0L ); 914 915 OutStream->writeLong( aIndex.size() ); 916 917 // Sequences schreiben 918 ::std::deque<AttacherIndex_Impl>::iterator aIt = aIndex.begin(); 919 ::std::deque<AttacherIndex_Impl>::iterator aEnd = aIndex.end(); 920 while( aIt != aEnd ) 921 { 922 #ifdef DEQUE_OK 923 // Laenge der Sequence und alle Descriptoren schreiben 924 OutStream->writeLong( (*aIt).aEventList.size() ); 925 ::std::deque< ScriptEventDescriptor >::iterator aEvtIt = (*aIt).aEventList.begin(); 926 ::std::deque< ScriptEventDescriptor >::iterator aEvtEnd = (*aIt).aEventList.end(); 927 while( aEvtIt != aEvtEnd ) 928 { 929 const ScriptEventDescriptor& rDesc = (*aEvtIt); 930 OutStream->writeUTF( rDesc.ListenerType ); 931 OutStream->writeUTF( rDesc.EventMethod ); 932 OutStream->writeUTF( rDesc.AddListenerParam ); 933 OutStream->writeUTF( rDesc.ScriptType ); 934 OutStream->writeUTF( rDesc.ScriptCode ); 935 936 aEvtIt++; 937 } 938 #else 939 sal_Int32 nLen = (*aIt).aEventList.getLength(); 940 // Laenge der Sequence und alle Descriptoren schreiben 941 OutStream->writeLong( nLen ); 942 ScriptEventDescriptor * pEL = (*aIt).aEventList.getArray(); 943 for( sal_Int32 i = 0; i < nLen; i++ ) 944 { 945 const ScriptEventDescriptor& rDesc = pEL[i]; 946 OutStream->writeUTF( rDesc.ListenerType ); 947 OutStream->writeUTF( rDesc.EventMethod ); 948 OutStream->writeUTF( rDesc.AddListenerParam ); 949 OutStream->writeUTF( rDesc.ScriptType ); 950 OutStream->writeUTF( rDesc.ScriptCode ); 951 } 952 #endif 953 aIt++; 954 } 955 956 // Die jetzt bekannte Laenge eintragen 957 sal_Int32 nObjLen = xMarkStream->offsetToMark( nObjLenMark ) -4; 958 xMarkStream->jumpToMark( nObjLenMark ); 959 OutStream->writeLong( nObjLen ); 960 xMarkStream->jumpToFurthest(); 961 xMarkStream->deleteMark( nObjLenMark ); 962 } 963 964 void SAL_CALL ImplEventAttacherManager::read(const Reference< XObjectInputStream >& InStream) 965 throw( IOException, RuntimeException ) 966 { 967 Guard< Mutex > aGuard( aLock ); 968 // Ohne XMarkableStream laeuft nichts 969 Reference< XMarkableStream > xMarkStream( InStream, UNO_QUERY ); 970 if( !xMarkStream.is() ) 971 return; 972 973 // Version lesen 974 nVersion = InStream->readShort(); 975 976 // Zunaechst kommen die Daten gemaess Version 1, 977 // muss auch bei hoeheren Versionen beibehalten werden 978 sal_Int32 nLen = InStream->readLong(); 979 980 // Position fuer Vergleichszwecke 981 sal_Int32 nObjLenMark = xMarkStream->createMark(); 982 983 // Anzahl der zu lesenden Sequences 984 sal_Int32 nItemCount = InStream->readLong(); 985 986 for( sal_Int32 i = 0 ; i < nItemCount ; i++ ) 987 { 988 insertEntry( i ); 989 // Laenge der Sequence lesen 990 sal_Int32 nSeqLen = InStream->readLong(); 991 992 // Sequence anlegen und Descriptoren lesen 993 Sequence< ScriptEventDescriptor > aSEDSeq( nSeqLen ); 994 ScriptEventDescriptor* pArray = aSEDSeq.getArray(); 995 for( sal_Int32 j = 0 ; j < nSeqLen ; j++ ) 996 { 997 ScriptEventDescriptor& rDesc = pArray[ j ]; 998 rDesc.ListenerType = InStream->readUTF(); 999 rDesc.EventMethod = InStream->readUTF(); 1000 rDesc.AddListenerParam = InStream->readUTF(); 1001 rDesc.ScriptType = InStream->readUTF(); 1002 rDesc.ScriptCode = InStream->readUTF(); 1003 } 1004 registerScriptEvents( i, aSEDSeq ); 1005 } 1006 1007 // Haben wir die angegebene Laenge gelesen? 1008 sal_Int32 nRealLen = xMarkStream->offsetToMark( nObjLenMark ); 1009 if( nRealLen != nLen ) 1010 { 1011 // Nur wenn die StreamVersion > 1 ist und noch Daten folgen, kann das 1012 // Ganze richtig sein. Sonst ist etwas voellig daneben gegangen. 1013 if( nRealLen > nLen || nVersion == 1 ) 1014 { 1015 VOS_ENSHURE( sal_False, "ImplEventAttacherManager::read(): Fatal Error, wrong object length" ); 1016 } 1017 else 1018 { 1019 // TODO: Pruefen, ob Zwischen-Speicherung der Daten sinnvoll sein koennte 1020 1021 // Vorerst einfach nur Skippen 1022 sal_Int32 nSkipCount = nLen - nRealLen; 1023 InStream->skipBytes( nSkipCount ); 1024 } 1025 } 1026 xMarkStream->jumpToFurthest(); 1027 xMarkStream->deleteMark( nObjLenMark ); 1028 } 1029 1030 } // namesapce comphelper 1031 1032 1033