Line data Source code
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 : */
19 :
20 : #include "definitioncontainer.hxx"
21 : #include "dbastrings.hrc"
22 : #include "apitools.hxx"
23 : #include "core_resource.hxx"
24 : #include "core_resource.hrc"
25 :
26 : #include <tools/debug.hxx>
27 : #include <tools/diagnose_ex.h>
28 : #include <osl/diagnose.h>
29 : #include <comphelper/sequence.hxx>
30 : #include <comphelper/enumhelper.hxx>
31 : #include <comphelper/extract.hxx>
32 : #include <cppuhelper/exc_hlp.hxx>
33 : #include <com/sun/star/lang/XComponent.hpp>
34 : #include <com/sun/star/ucb/CommandInfo.hpp>
35 : #include <com/sun/star/beans/XPropertySet.hpp>
36 : #include <com/sun/star/sdb/ErrorCondition.hpp>
37 : #include <comphelper/types.hxx>
38 : #include <ucbhelper/contentidentifier.hxx>
39 : #include <o3tl/compat_functional.hxx>
40 :
41 : using namespace ::com::sun::star::uno;
42 : using namespace ::com::sun::star::lang;
43 : using namespace ::com::sun::star::util;
44 : using namespace ::com::sun::star::beans;
45 : using namespace ::com::sun::star::container;
46 : using namespace ::com::sun::star::sdbcx;
47 : using namespace ::com::sun::star::sdb;
48 : using namespace ::osl;
49 : using namespace ::comphelper;
50 : using namespace ::cppu;
51 : using namespace ::com::sun::star::ucb;
52 :
53 : namespace dbaccess
54 : {
55 :
56 : // ODefinitionContainer_Impl
57 80 : void ODefinitionContainer_Impl::erase( TContentPtr _pDefinition )
58 : {
59 80 : NamedDefinitions::iterator aPos = find( _pDefinition );
60 80 : if ( aPos != end() )
61 0 : m_aDefinitions.erase( aPos );
62 80 : }
63 :
64 80 : ODefinitionContainer_Impl::const_iterator ODefinitionContainer_Impl::find( TContentPtr _pDefinition ) const
65 : {
66 : return ::std::find_if(
67 : m_aDefinitions.begin(),
68 : m_aDefinitions.end(),
69 : ::o3tl::compose1(
70 : ::std::bind2nd( ::std::equal_to< TContentPtr >(), _pDefinition ),
71 : ::o3tl::select2nd< NamedDefinitions::value_type >()
72 : )
73 80 : );
74 : }
75 :
76 80 : ODefinitionContainer_Impl::iterator ODefinitionContainer_Impl::find( TContentPtr _pDefinition )
77 : {
78 : return ::std::find_if(
79 : m_aDefinitions.begin(),
80 : m_aDefinitions.end(),
81 : ::o3tl::compose1(
82 : ::std::bind2nd( ::std::equal_to< TContentPtr >(), _pDefinition ),
83 : ::o3tl::select2nd< NamedDefinitions::value_type >()
84 : )
85 80 : );
86 : }
87 :
88 : // ODefinitionContainer
89 :
90 1518 : ODefinitionContainer::ODefinitionContainer( const Reference< XComponentContext >& _xORB
91 : , const Reference< XInterface >& _xParentContainer
92 : , const TContentPtr& _pImpl
93 : , bool _bCheckSlash
94 : )
95 : :OContentHelper(_xORB,_xParentContainer,_pImpl)
96 : ,m_aApproveListeners(m_aMutex)
97 : ,m_aContainerListeners(m_aMutex)
98 : ,m_bInPropertyChange(false)
99 1518 : ,m_bCheckSlash(_bCheckSlash)
100 : {
101 1518 : m_pImpl->m_aProps.bIsDocument = false;
102 1518 : m_pImpl->m_aProps.bIsFolder = true;
103 :
104 1518 : const ODefinitionContainer_Impl& rDefinitions( getDefinitions() );
105 1518 : ODefinitionContainer_Impl::const_iterator aEnd = rDefinitions.end();
106 1592 : for ( ODefinitionContainer_Impl::const_iterator aDefinition = rDefinitions.begin();
107 : aDefinition != aEnd;
108 : ++aDefinition
109 : )
110 : m_aDocuments.push_back(
111 : m_aDocumentMap.insert(
112 74 : Documents::value_type( aDefinition->first, Documents::mapped_type() ) ).first );
113 :
114 1518 : }
115 :
116 1512 : void SAL_CALL ODefinitionContainer::disposing()
117 : {
118 1512 : OContentHelper::disposing();
119 :
120 1512 : MutexGuard aGuard(m_aMutex);
121 :
122 : // say goodbye to our listeners
123 3024 : EventObject aEvt(*this);
124 1512 : m_aApproveListeners.disposeAndClear(aEvt);
125 1512 : m_aContainerListeners.disposeAndClear(aEvt);
126 :
127 : // dispose our elements
128 1512 : Documents::iterator aIter = m_aDocumentMap.begin();
129 1512 : Documents::iterator aEnd = m_aDocumentMap.end();
130 :
131 1664 : for (; aIter != aEnd; ++aIter)
132 : {
133 152 : Reference<XContent> xProp = aIter->second;
134 152 : if ( xProp.is() )
135 : {
136 4 : removeObjectListener(xProp);
137 4 : ::comphelper::disposeComponent(xProp);
138 : }
139 152 : }
140 :
141 : // remove our elements
142 1512 : m_aDocuments.clear();
143 : // !!! do this before clearing the map which the vector elements refer to !!!
144 3024 : m_aDocumentMap.clear();
145 1512 : }
146 :
147 1512 : ODefinitionContainer::~ODefinitionContainer()
148 : {
149 1512 : }
150 :
151 61582 : IMPLEMENT_FORWARD_XINTERFACE2( ODefinitionContainer,OContentHelper,ODefinitionContainer_Base)
152 4 : IMPLEMENT_GETTYPES2(ODefinitionContainer,OContentHelper,ODefinitionContainer_Base);
153 :
154 0 : css::uno::Sequence<sal_Int8> ODefinitionContainer::getImplementationId()
155 : throw (css::uno::RuntimeException, std::exception)
156 : {
157 0 : return css::uno::Sequence<sal_Int8>();
158 : }
159 :
160 : // XServiceInfo
161 0 : OUString SAL_CALL ODefinitionContainer::getImplementationName( ) throw(RuntimeException, std::exception)
162 : {
163 0 : return OUString("com.sun.star.sdb.ODefinitionContainer");
164 : }
165 :
166 0 : Sequence< OUString > SAL_CALL ODefinitionContainer::getSupportedServiceNames( ) throw(RuntimeException, std::exception)
167 : {
168 0 : Sequence< OUString > aReturn(2);
169 0 : aReturn.getArray()[0] = "com.sun.star.sdb.DefinitionContainer";
170 0 : aReturn.getArray()[1] = "com.sun.star.ucb.Content";
171 0 : return aReturn;
172 : }
173 :
174 : // XNameContainer
175 80 : void SAL_CALL ODefinitionContainer::insertByName( const OUString& _rName, const Any& aElement ) throw(IllegalArgumentException, ElementExistException, WrappedTargetException, RuntimeException, std::exception)
176 : {
177 80 : ResettableMutexGuard aGuard(m_aMutex);
178 :
179 : // approve the new object
180 160 : Reference< XContent > xNewElement(aElement,UNO_QUERY);
181 80 : approveNewObject( _rName, xNewElement ); // will throw if necessary
182 :
183 80 : notifyByName( aGuard, _rName, xNewElement, NULL, E_INSERTED, ApproveListeners );
184 80 : implAppend( _rName, xNewElement );
185 160 : notifyByName( aGuard, _rName, xNewElement, NULL, E_INSERTED, ContainerListemers );
186 80 : }
187 :
188 0 : void SAL_CALL ODefinitionContainer::removeByName( const OUString& _rName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException, std::exception)
189 : {
190 0 : ResettableMutexGuard aGuard(m_aMutex);
191 :
192 : // check the arguments
193 0 : if (_rName.isEmpty())
194 0 : throw IllegalArgumentException();
195 :
196 0 : if (!checkExistence(_rName))
197 0 : throw NoSuchElementException(_rName,*this);
198 :
199 : // the old element (for the notifications)
200 0 : Reference< XContent > xOldElement = implGetByName( _rName, impl_haveAnyListeners_nothrow() );
201 :
202 : // do the removal
203 0 : notifyByName( aGuard, _rName, NULL, xOldElement, E_REMOVED, ApproveListeners );
204 0 : implRemove( _rName );
205 0 : notifyByName( aGuard, _rName, NULL, xOldElement, E_REMOVED, ContainerListemers );
206 :
207 0 : removeObjectListener( xOldElement );
208 0 : disposeComponent(xOldElement);
209 0 : }
210 :
211 : // XNameReplace
212 0 : void SAL_CALL ODefinitionContainer::replaceByName( const OUString& _rName, const Any& aElement ) throw(IllegalArgumentException, NoSuchElementException, WrappedTargetException, RuntimeException, std::exception)
213 : {
214 0 : ResettableMutexGuard aGuard(m_aMutex);
215 :
216 : try
217 : {
218 : // let derived classes approve the new object
219 0 : Reference< XContent > xNewElement(aElement,UNO_QUERY);
220 0 : approveNewObject( _rName, xNewElement ); // will throw if necessary
221 :
222 : // the old element (for the notifications)
223 0 : Reference< XContent > xOldElement = implGetByName( _rName, impl_haveAnyListeners_nothrow() );
224 :
225 0 : notifyByName( aGuard, _rName, xNewElement, xOldElement, E_REPLACED, ApproveListeners );
226 0 : implReplace( _rName, xNewElement );
227 0 : notifyByName( aGuard, _rName, xNewElement, xOldElement, E_REPLACED, ContainerListemers );
228 :
229 : // and dispose it
230 0 : disposeComponent(xOldElement);
231 : }
232 0 : catch (const RuntimeException&)
233 : {
234 0 : throw;
235 : }
236 0 : catch (const NoSuchElementException&)
237 : {
238 0 : throw;
239 : }
240 0 : catch (const WrappedTargetException&)
241 : {
242 0 : throw;
243 : }
244 0 : catch (const Exception& e)
245 : {
246 0 : css::uno::Any a(cppu::getCaughtException());
247 : throw css::lang::WrappedTargetException(
248 0 : "wrapped Exception " + e.Message,
249 0 : css::uno::Reference<css::uno::XInterface>(), a);
250 0 : }
251 0 : }
252 :
253 : namespace
254 : {
255 : typedef Reference< XVeto > ( SAL_CALL XContainerApproveListener::*ContainerApprovalMethod )( const ContainerEvent& );
256 :
257 : struct RaiseExceptionFromVeto
258 : {
259 : private:
260 : ContainerApprovalMethod m_pMethod;
261 : const ContainerEvent& m_rEvent;
262 :
263 : public:
264 0 : RaiseExceptionFromVeto( ContainerApprovalMethod _pMethod, const ContainerEvent& _rEvent )
265 : :m_pMethod( _pMethod )
266 0 : ,m_rEvent( _rEvent )
267 : {
268 0 : }
269 :
270 0 : void operator()( const Reference< XContainerApproveListener >& _Listener ) const
271 : {
272 0 : Reference< XVeto > xVeto = (_Listener.get()->*m_pMethod)( m_rEvent );
273 0 : if ( !xVeto.is() )
274 0 : return;
275 :
276 0 : Any eVetoDetails = xVeto->getDetails();
277 :
278 0 : IllegalArgumentException aIllegalArgumentError;
279 0 : if ( eVetoDetails >>= aIllegalArgumentError )
280 0 : throw aIllegalArgumentError;
281 :
282 0 : WrappedTargetException aWrappedError;
283 0 : if ( eVetoDetails >>= aWrappedError )
284 0 : throw aWrappedError;
285 :
286 0 : throw WrappedTargetException( xVeto->getReason(), _Listener.get(), eVetoDetails );
287 : }
288 : };
289 : }
290 :
291 160 : void ODefinitionContainer::notifyByName( ResettableMutexGuard& _rGuard, const OUString& _rName,
292 : const Reference< XContent >& _xNewElement, const Reference< XContent >& _xOldElement,
293 : ContainerOperation _eOperation, ListenerType _eType )
294 : {
295 160 : bool bApprove = ( _eType == ApproveListeners );
296 :
297 160 : ::cppu::OInterfaceContainerHelper& rContainer( bApprove ? m_aApproveListeners : m_aContainerListeners );
298 160 : if ( !rContainer.getLength() )
299 320 : return;
300 :
301 0 : ContainerEvent aEvent( *this, makeAny( _rName ), makeAny( _xNewElement ), makeAny( _xOldElement ) );
302 :
303 0 : _rGuard.clear();
304 0 : switch ( _eOperation )
305 : {
306 : case E_INSERTED:
307 0 : if ( bApprove )
308 : rContainer.forEach< XContainerApproveListener, RaiseExceptionFromVeto >(
309 0 : RaiseExceptionFromVeto( &XContainerApproveListener::approveInsertElement, aEvent ) );
310 : else
311 0 : rContainer.notifyEach( &XContainerListener::elementInserted, aEvent );
312 0 : break;
313 : case E_REPLACED:
314 0 : if ( bApprove )
315 : rContainer.forEach< XContainerApproveListener, RaiseExceptionFromVeto >(
316 0 : RaiseExceptionFromVeto( &XContainerApproveListener::approveReplaceElement, aEvent ) );
317 : else
318 0 : rContainer.notifyEach( &XContainerListener::elementReplaced, aEvent );
319 0 : break;
320 : case E_REMOVED:
321 0 : if ( bApprove )
322 : rContainer.forEach< XContainerApproveListener, RaiseExceptionFromVeto >(
323 0 : RaiseExceptionFromVeto( &XContainerApproveListener::approveRemoveElement, aEvent ) );
324 : else
325 0 : rContainer.notifyEach( &XContainerListener::elementRemoved, aEvent );
326 0 : break;
327 : }
328 :
329 0 : if ( bApprove )
330 0 : _rGuard.reset();
331 : }
332 :
333 114 : void SAL_CALL ODefinitionContainer::addContainerListener( const Reference< XContainerListener >& _rxListener ) throw(RuntimeException, std::exception)
334 : {
335 114 : if (_rxListener.is())
336 114 : m_aContainerListeners.addInterface(_rxListener);
337 114 : }
338 :
339 110 : void SAL_CALL ODefinitionContainer::removeContainerListener( const Reference< XContainerListener >& _rxListener ) throw(RuntimeException, std::exception)
340 : {
341 110 : if (_rxListener.is())
342 110 : m_aContainerListeners.removeInterface(_rxListener);
343 110 : }
344 :
345 70 : void SAL_CALL ODefinitionContainer::addContainerApproveListener( const Reference< XContainerApproveListener >& _Listener ) throw (RuntimeException, std::exception)
346 : {
347 70 : if ( _Listener.is() )
348 70 : m_aApproveListeners.addInterface( _Listener );
349 70 : }
350 :
351 68 : void SAL_CALL ODefinitionContainer::removeContainerApproveListener( const Reference< XContainerApproveListener >& _Listener ) throw (RuntimeException, std::exception)
352 : {
353 68 : if ( _Listener.is() )
354 68 : m_aApproveListeners.removeInterface( _Listener );
355 68 : }
356 :
357 : // XElementAccess
358 0 : Type SAL_CALL ODefinitionContainer::getElementType( ) throw (RuntimeException, std::exception)
359 : {
360 0 : return cppu::UnoType<XContent>::get();
361 : }
362 :
363 1294 : sal_Bool SAL_CALL ODefinitionContainer::hasElements( ) throw (RuntimeException, std::exception)
364 : {
365 1294 : MutexGuard aGuard(m_aMutex);
366 1294 : return !m_aDocuments.empty();
367 : }
368 :
369 : // XEnumerationAccess
370 0 : Reference< XEnumeration > SAL_CALL ODefinitionContainer::createEnumeration( ) throw(RuntimeException, std::exception)
371 : {
372 0 : MutexGuard aGuard(m_aMutex);
373 0 : return new ::comphelper::OEnumerationByIndex(static_cast<XIndexAccess*>(this));
374 : }
375 :
376 : // XIndexAccess
377 0 : sal_Int32 SAL_CALL ODefinitionContainer::getCount( ) throw(RuntimeException, std::exception)
378 : {
379 0 : MutexGuard aGuard(m_aMutex);
380 0 : return m_aDocuments.size();
381 : }
382 :
383 0 : Any SAL_CALL ODefinitionContainer::getByIndex( sal_Int32 _nIndex ) throw(IndexOutOfBoundsException, WrappedTargetException, RuntimeException, std::exception)
384 : {
385 0 : MutexGuard aGuard(m_aMutex);
386 :
387 0 : if ((_nIndex < 0) || (_nIndex >= (sal_Int32)m_aDocuments.size()))
388 0 : throw IndexOutOfBoundsException();
389 :
390 0 : Documents::iterator aPos = m_aDocuments[_nIndex];
391 0 : Reference<XContent> xProp = aPos->second;
392 0 : if (!xProp.is())
393 : { // that's the first access to the object
394 : // -> create it
395 0 : xProp = createObject(aPos->first);
396 0 : aPos->second = Documents::mapped_type();
397 : // and update the name-access map
398 : }
399 :
400 0 : return makeAny(xProp);
401 : }
402 :
403 98 : Any SAL_CALL ODefinitionContainer::getByName( const OUString& _rName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException, std::exception)
404 : {
405 98 : MutexGuard aGuard(m_aMutex);
406 :
407 98 : return makeAny( implGetByName( _rName, true ) );
408 : }
409 :
410 98 : Reference< XContent > ODefinitionContainer::implGetByName(const OUString& _rName, bool _bReadIfNecessary) throw (NoSuchElementException)
411 : {
412 98 : Documents::iterator aMapPos = m_aDocumentMap.find(_rName);
413 98 : if (aMapPos == m_aDocumentMap.end())
414 0 : throw NoSuchElementException(_rName,*this);
415 :
416 98 : Reference< XContent > xProp = aMapPos->second;
417 :
418 98 : if (_bReadIfNecessary && !xProp.is())
419 : { // the object has never been accessed before, so we have to read it now
420 : // (that's the expensive part)
421 :
422 : // create the object and insert it into the map
423 60 : xProp = createObject(_rName);
424 60 : aMapPos->second = xProp;
425 60 : addObjectListener(xProp);
426 : }
427 :
428 98 : return xProp;
429 : }
430 :
431 118 : Sequence< OUString > SAL_CALL ODefinitionContainer::getElementNames( ) throw(RuntimeException, std::exception)
432 : {
433 118 : MutexGuard aGuard(m_aMutex);
434 :
435 118 : Sequence< OUString > aNames(m_aDocumentMap.size());
436 118 : OUString* pNames = aNames.getArray();
437 118 : Documents::iterator aEnd = m_aDocumentMap.end();
438 164 : for ( Documents::iterator aNameIter = m_aDocumentMap.begin();
439 : aNameIter != aEnd;
440 : ++pNames, ++aNameIter
441 : )
442 : {
443 46 : *pNames = aNameIter->first;
444 : }
445 :
446 118 : return aNames;
447 : }
448 :
449 124 : sal_Bool SAL_CALL ODefinitionContainer::hasByName( const OUString& _rName ) throw(RuntimeException, std::exception)
450 : {
451 124 : MutexGuard aGuard(m_aMutex);
452 :
453 124 : return checkExistence(_rName);
454 : }
455 :
456 28 : void SAL_CALL ODefinitionContainer::disposing( const EventObject& _rSource ) throw(RuntimeException, std::exception)
457 : {
458 28 : MutexGuard aGuard(m_aMutex);
459 56 : Reference< XContent > xSource(_rSource.Source, UNO_QUERY);
460 : // it's one of our documents ....
461 28 : Documents::iterator aIter = m_aDocumentMap.begin();
462 28 : Documents::iterator aEnd = m_aDocumentMap.end();
463 28 : for (;aIter != aEnd;++aIter )
464 : {
465 0 : if ( xSource == aIter->second.get() )
466 : {
467 0 : removeObjectListener(xSource);
468 : // and clear our document map/vector, so the object will be recreated on next access
469 0 : aIter->second = Documents::mapped_type();
470 : }
471 28 : }
472 28 : }
473 :
474 0 : void ODefinitionContainer::implRemove(const OUString& _rName)
475 : {
476 : // from the object maps
477 0 : Documents::iterator aFind = m_aDocumentMap.find(_rName);
478 0 : if ( aFind != m_aDocumentMap.end() )
479 : {
480 0 : m_aDocuments.erase( ::std::find(m_aDocuments.begin(),m_aDocuments.end(),aFind));
481 0 : m_aDocumentMap.erase(aFind);
482 :
483 0 : getDefinitions().erase( _rName );
484 :
485 0 : notifyDataSourceModified();
486 : }
487 0 : }
488 :
489 : namespace
490 : {
491 80 : bool lcl_ensureName( const Reference< XContent >& _rxContent, const OUString& _rName )
492 : {
493 80 : if ( !_rxContent.is() )
494 0 : return true;
495 :
496 : // obtain the current name. If it's the same as the new one,
497 : // don't do anything
498 : try
499 : {
500 80 : Reference< XPropertySet > xProps( _rxContent, UNO_QUERY );
501 80 : if ( xProps.is() )
502 : {
503 80 : OUString sCurrentName;
504 80 : OSL_VERIFY( xProps->getPropertyValue( PROPERTY_NAME ) >>= sCurrentName );
505 80 : if ( sCurrentName.equals( _rName ) )
506 80 : return true;
507 0 : }
508 : }
509 0 : catch( const Exception& )
510 : {
511 : OSL_FAIL( "lcl_ensureName: caught an exception while obtaining the current name!" );
512 : }
513 :
514 : // set the new name
515 0 : Reference< XRename > xRename( _rxContent, UNO_QUERY );
516 : OSL_ENSURE( xRename.is(), "lcl_ensureName: invalid content (not renameable)!" );
517 0 : if ( !xRename.is() )
518 0 : return false;
519 : try
520 : {
521 0 : xRename->rename( _rName );
522 0 : return true;
523 : }
524 0 : catch( const Exception& )
525 : {
526 : OSL_FAIL( "lcl_ensureName: caught an exception!" );
527 : }
528 0 : return false;
529 : }
530 : }
531 :
532 80 : void ODefinitionContainer::implAppend(const OUString& _rName, const Reference< XContent >& _rxNewObject)
533 : {
534 80 : MutexGuard aGuard(m_aMutex);
535 : try
536 : {
537 80 : Reference<XChild> xChild(_rxNewObject,UNO_QUERY);
538 80 : if ( xChild.is() )
539 80 : xChild->setParent(static_cast<OWeakObject*>(this));
540 :
541 80 : ODefinitionContainer_Impl& rDefinitions( getDefinitions() );
542 80 : ODefinitionContainer_Impl::const_iterator aFind = rDefinitions.find( _rName );
543 80 : if ( aFind == rDefinitions.end() )
544 : {
545 : // ensure that the new object has the proper name.
546 : // Somebody could create an object with name "foo", and insert it as "bar"
547 : // into a container. In this case, we need to ensure that the object name
548 : // is also "bar"
549 : // #i44786#
550 80 : lcl_ensureName( _rxNewObject, _rName );
551 :
552 80 : ::rtl::Reference< OContentHelper > pContent = OContentHelper::getImplementation( _rxNewObject );
553 80 : if ( pContent.is() )
554 : {
555 80 : TContentPtr pImpl = pContent->getImpl();
556 80 : rDefinitions.erase( pImpl );
557 80 : pImpl->m_aProps.aTitle = _rName;
558 80 : rDefinitions.insert( _rName, pImpl );
559 80 : }
560 : }
561 :
562 80 : m_aDocuments.push_back(m_aDocumentMap.insert(Documents::value_type(_rName,_rxNewObject)).first);
563 80 : notifyDataSourceModified();
564 : // now update our structures
565 80 : if ( _rxNewObject.is() )
566 80 : addObjectListener(_rxNewObject);
567 : }
568 0 : catch(Exception&)
569 : {
570 : OSL_FAIL("ODefinitionContainer::implAppend: caught something !");
571 80 : }
572 80 : }
573 :
574 0 : void ODefinitionContainer::implReplace(const OUString& _rName, const Reference< XContent >& _rxNewObject)
575 : {
576 : OSL_ENSURE(checkExistence(_rName), "ODefinitionContainer::implReplace : invalid name !");
577 :
578 0 : Documents::iterator aFind = m_aDocumentMap.find(_rName);
579 0 : removeObjectListener(aFind->second);
580 0 : aFind->second = _rxNewObject;
581 0 : addObjectListener(aFind->second);
582 0 : }
583 :
584 80 : void ODefinitionContainer::approveNewObject(const OUString& _sName,const Reference< XContent >& _rxObject) const
585 : {
586 : // check the arguments
587 80 : if ( _sName.isEmpty() )
588 : throw IllegalArgumentException(
589 : DBA_RES( RID_STR_NAME_MUST_NOT_BE_EMPTY ),
590 : *this,
591 0 : 0 );
592 :
593 80 : if ( m_bCheckSlash && _sName.indexOf( '/' ) != -1 )
594 : throw IllegalArgumentException(
595 : m_aErrorHelper.getErrorMessage( ErrorCondition::DB_OBJECT_NAME_WITH_SLASHES ),
596 : *this,
597 0 : 0 );
598 :
599 80 : if ( !_rxObject.is() )
600 : throw IllegalArgumentException(
601 : DBA_RES( RID_STR_NO_NULL_OBJECTS_IN_CONTAINER ),
602 : *this,
603 0 : 0 );
604 :
605 80 : const ODefinitionContainer_Impl& rDefinitions( getDefinitions() );
606 80 : if ( rDefinitions.find( _sName ) != rDefinitions.end() )
607 : throw ElementExistException(
608 : DBA_RES( RID_STR_NAME_ALREADY_USED ),
609 0 : *this );
610 :
611 80 : ::rtl::Reference< OContentHelper > pContent( OContentHelper::getImplementation( _rxObject ) );
612 80 : if ( !pContent.is() )
613 : throw IllegalArgumentException(
614 : DBA_RES( RID_STR_OBJECT_CONTAINER_MISMATCH ),
615 : *this,
616 0 : 1 );
617 :
618 80 : if ( rDefinitions.find( pContent->getImpl() ) != rDefinitions.end() )
619 : throw ElementExistException(
620 : DBA_RES( RID_STR_OBJECT_ALREADY_CONTAINED ),
621 0 : *this );
622 80 : }
623 :
624 : // XPropertyChangeListener
625 0 : void SAL_CALL ODefinitionContainer::propertyChange( const PropertyChangeEvent& evt ) throw (RuntimeException, std::exception)
626 : {
627 0 : ClearableMutexGuard aGuard(m_aMutex);
628 0 : if( evt.PropertyName == PROPERTY_NAME || evt.PropertyName == "Title" )
629 : {
630 0 : m_bInPropertyChange = true;
631 : try
632 : {
633 0 : OUString sNewName,sOldName;
634 0 : evt.OldValue >>= sOldName;
635 0 : evt.NewValue >>= sNewName;
636 0 : Reference<XContent> xContent( evt.Source, UNO_QUERY );
637 0 : removeObjectListener( xContent );
638 0 : implRemove( sOldName );
639 0 : implAppend( sNewName, xContent );
640 : }
641 0 : catch(const Exception&)
642 : {
643 : DBG_UNHANDLED_EXCEPTION();
644 0 : throw RuntimeException();
645 : }
646 0 : m_bInPropertyChange = false;
647 0 : }
648 0 : }
649 :
650 : // XVetoableChangeListener
651 0 : void SAL_CALL ODefinitionContainer::vetoableChange( const PropertyChangeEvent& aEvent ) throw (PropertyVetoException, RuntimeException, std::exception)
652 : {
653 0 : MutexGuard aGuard(m_aMutex);
654 :
655 0 : if( aEvent.PropertyName == PROPERTY_NAME || aEvent.PropertyName == "Title" )
656 : {
657 0 : OUString sNewName;
658 0 : aEvent.NewValue >>= sNewName;
659 0 : if(hasByName(sNewName))
660 0 : throw PropertyVetoException();
661 0 : }
662 0 : }
663 :
664 140 : void ODefinitionContainer::addObjectListener(const Reference< XContent >& _xNewObject)
665 : {
666 : OSL_ENSURE(_xNewObject.is(),"ODefinitionContainer::addObjectListener: Object is null!");
667 140 : Reference<XPropertySet> xProp(_xNewObject,UNO_QUERY);
668 140 : if ( xProp.is() )
669 : {
670 140 : xProp->addPropertyChangeListener(PROPERTY_NAME, this);
671 140 : xProp->addVetoableChangeListener(PROPERTY_NAME, this);
672 140 : }
673 140 : }
674 :
675 4 : void ODefinitionContainer::removeObjectListener(const Reference< XContent >& _xNewObject)
676 : {
677 4 : Reference<XPropertySet> xProp(_xNewObject,UNO_QUERY);
678 4 : if ( xProp.is() )
679 : {
680 4 : xProp->removePropertyChangeListener(PROPERTY_NAME, this);
681 4 : xProp->removeVetoableChangeListener(PROPERTY_NAME, this);
682 4 : }
683 4 : }
684 :
685 100 : bool ODefinitionContainer::checkExistence(const OUString& _rName)
686 : {
687 100 : return m_aDocumentMap.find(_rName) != m_aDocumentMap.end();
688 : }
689 :
690 : }
691 :
692 : // namespace dbaccess
693 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|