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_forms.hxx"
26
27 #include "componenttools.hxx"
28 #include "DatabaseForm.hxx"
29 #include "EventThread.hxx"
30 #include "frm_module.hxx"
31 #include "frm_resource.hrc"
32 #include "frm_resource.hxx"
33 #include "GroupManager.hxx"
34 #include "property.hrc"
35 #include "property.hxx"
36 #include "services.hxx"
37
38 #include <com/sun/star/awt/XControlContainer.hpp>
39 #include <com/sun/star/awt/XTextComponent.hpp>
40 #include <com/sun/star/form/DataSelectionType.hpp>
41 #include <com/sun/star/form/FormComponentType.hpp>
42 #include <com/sun/star/form/TabulatorCycle.hpp>
43 #include <com/sun/star/frame/FrameSearchFlag.hpp>
44 #include <com/sun/star/frame/XDispatch.hpp>
45 #include <com/sun/star/frame/XDispatchProvider.hpp>
46 #include <com/sun/star/frame/XModel.hpp>
47 #include <com/sun/star/io/XObjectInputStream.hpp>
48 #include <com/sun/star/io/XObjectOutputStream.hpp>
49 #include <com/sun/star/sdb/CommandType.hpp>
50 #include <com/sun/star/sdb/RowSetVetoException.hpp>
51 #include <com/sun/star/sdb/SQLContext.hpp>
52 #include <com/sun/star/sdb/XColumnUpdate.hpp>
53 #include <com/sun/star/sdbc/DataType.hpp>
54 #include <com/sun/star/sdbc/ResultSetConcurrency.hpp>
55 #include <com/sun/star/sdbc/ResultSetType.hpp>
56 #include <com/sun/star/sdbc/XRowSet.hpp>
57 #include <com/sun/star/sdbcx/Privilege.hpp>
58 #include <com/sun/star/sdbcx/XColumnsSupplier.hpp>
59 #include <com/sun/star/util/XCancellable.hpp>
60 #include <com/sun/star/util/XURLTransformer.hpp>
61 #include <com/sun/star/util/XModifiable2.hpp>
62
63 #include <comphelper/basicio.hxx>
64 #include <comphelper/container.hxx>
65 #include <comphelper/enumhelper.hxx>
66 #include <comphelper/extract.hxx>
67 #include <comphelper/seqstream.hxx>
68 #include <comphelper/sequence.hxx>
69 #include <comphelper/stl_types.hxx>
70 #include <comphelper/uno3.hxx>
71 #include <connectivity/dbtools.hxx>
72 #include <cppuhelper/exc_hlp.hxx>
73 #include <cppuhelper/implbase2.hxx>
74 #include <osl/mutex.hxx>
75 #include <rtl/math.hxx>
76 #include <rtl/tencinfo.h>
77 #include <svl/inetstrm.hxx>
78 #include <svl/inettype.hxx>
79 #include <tools/debug.hxx>
80 #include <tools/diagnose_ex.h>
81 #include <tools/fsys.hxx>
82 #include <tools/inetmsg.hxx>
83 #include <tools/urlobj.hxx>
84 #include <unotools/ucblockbytes.hxx>
85 #include <unotools/ucbstreamhelper.hxx>
86 #include <vcl/svapp.hxx>
87 #include <vcl/timer.hxx>
88 #include <vos/mutex.hxx>
89
90 #include <ctype.h>
91 #include <hash_map>
92
93 // compatibility: DatabaseCursorType is dead, but for compatibility reasons we still have to write it ...
94 namespace com {
95 namespace sun {
96 namespace star {
97 namespace data {
98
99 enum DatabaseCursorType
100 {
101 DatabaseCursorType_FORWARD = 0,
102 DatabaseCursorType_SNAPSHOT = 1,
103 DatabaseCursorType_KEYSET = 2,
104 DatabaseCursorType_DYNAMIC = 3,
105 DatabaseCursorType_MAKE_FIXED_SIZE = SAL_MAX_ENUM
106 };
107
108 } } } }
109
110 using namespace ::dbtools;
111 using namespace ::comphelper;
112 using namespace ::com::sun::star::uno;
113 using namespace ::com::sun::star::sdb;
114 using namespace ::com::sun::star::sdbc;
115 using namespace ::com::sun::star::sdbcx;
116 using namespace ::com::sun::star::beans;
117 using namespace ::com::sun::star::container;
118 using namespace ::com::sun::star::task;
119 using namespace ::com::sun::star::frame;
120 using namespace ::com::sun::star::form;
121 using namespace ::com::sun::star::awt;
122 using namespace ::com::sun::star::io;
123 using namespace ::com::sun::star::lang;
124 using namespace ::com::sun::star::data;
125 using namespace ::com::sun::star::util;
126
127 //--------------------------------------------------------------------------
createRegistryInfo_ODatabaseForm()128 extern "C" void SAL_CALL createRegistryInfo_ODatabaseForm()
129 {
130 static ::frm::OMultiInstanceAutoRegistration< ::frm::ODatabaseForm > aAutoRegistration;
131 }
132
133 //.........................................................................
134 namespace frm
135 {
136 //.........................................................................
137
138 //==================================================================
139 //= DocumentModifyGuard
140 //==================================================================
141 class DocumentModifyGuard
142 {
143 public:
DocumentModifyGuard(const Reference<XInterface> & _rxFormComponent)144 DocumentModifyGuard( const Reference< XInterface >& _rxFormComponent )
145 :m_xDocumentModify( getXModel( _rxFormComponent ), UNO_QUERY )
146 {
147 impl_changeModifiableFlag_nothrow( false );
148 }
~DocumentModifyGuard()149 ~DocumentModifyGuard()
150 {
151 impl_changeModifiableFlag_nothrow( true );
152 }
153
154 private:
impl_changeModifiableFlag_nothrow(const bool _enable)155 void impl_changeModifiableFlag_nothrow( const bool _enable )
156 {
157 try
158 {
159 if ( m_xDocumentModify.is() )
160 _enable ? m_xDocumentModify->enableSetModified() : m_xDocumentModify->disableSetModified();
161 }
162 catch( const Exception& )
163 {
164 DBG_UNHANDLED_EXCEPTION();
165 }
166 }
167
168 private:
169 Reference< XModifiable2 > m_xDocumentModify;
170 };
171
172 //==================================================================
173 //= OFormSubmitResetThread
174 //=-----------------------------------------------------------------
175 //= submitting and resetting html-forms asynchronously
176 //==================================================================
177
178 //------------------------------------------------------------------
179 class OFormSubmitResetThread: public OComponentEventThread
180 {
181 protected:
182
183 // duplicate an event with respect to it's type
184 virtual EventObject *cloneEvent( const EventObject *pEvt ) const;
185
186 // process an event. while processing the mutex isn't locked, and pCompImpl
187 // is made sure to remain valid
188 virtual void processEvent( ::cppu::OComponentHelper* _pCompImpl,
189 const EventObject* _pEvt,
190 const Reference<XControl>& _rControl,
191 sal_Bool _bSubmit);
192
193 public:
194
OFormSubmitResetThread(ODatabaseForm * pControl)195 OFormSubmitResetThread(ODatabaseForm* pControl) : OComponentEventThread(pControl) { }
196 };
197
198 //------------------------------------------------------------------
cloneEvent(const EventObject * pEvt) const199 EventObject* OFormSubmitResetThread::cloneEvent(
200 const EventObject *pEvt ) const
201 {
202 return new ::com::sun::star::awt::MouseEvent( *(::com::sun::star::awt::MouseEvent *)pEvt );
203 }
204
205 //------------------------------------------------------------------
processEvent(::cppu::OComponentHelper * pCompImpl,const EventObject * _pEvt,const Reference<XControl> & _rControl,sal_Bool _bSubmit)206 void OFormSubmitResetThread::processEvent(
207 ::cppu::OComponentHelper* pCompImpl,
208 const EventObject *_pEvt,
209 const Reference<XControl>& _rControl,
210 sal_Bool _bSubmit)
211 {
212 if (_bSubmit)
213 ((ODatabaseForm *)pCompImpl)->submit_impl(_rControl, *static_cast<const ::com::sun::star::awt::MouseEvent*>(_pEvt), true);
214 else
215 ((ODatabaseForm *)pCompImpl)->reset_impl(true);
216 }
217
218 //==================================================================
219 //= ODatabaseForm
220 //==================================================================
221
222 //------------------------------------------------------------------
Create(const Reference<XMultiServiceFactory> & _rxFactory)223 Reference< XInterface > SAL_CALL ODatabaseForm::Create( const Reference< XMultiServiceFactory >& _rxFactory )
224 {
225 return *( new ODatabaseForm( _rxFactory ) );
226 }
227
228 //------------------------------------------------------------------------------
getImplementationId()229 Sequence<sal_Int8> SAL_CALL ODatabaseForm::getImplementationId() throw(RuntimeException)
230 {
231 return OImplementationIds::getImplementationId(getTypes());
232 }
233
234 //------------------------------------------------------------------
getTypes()235 Sequence<Type> SAL_CALL ODatabaseForm::getTypes() throw(RuntimeException)
236 {
237 // ask the aggregate
238 Sequence<Type> aAggregateTypes;
239 Reference<XTypeProvider> xAggregateTypes;
240 if (query_aggregation(m_xAggregate, xAggregateTypes))
241 aAggregateTypes = xAggregateTypes->getTypes();
242
243 Sequence< Type > aRet = concatSequences(
244 aAggregateTypes, ODatabaseForm_BASE1::getTypes(), OFormComponents::getTypes()
245 );
246 aRet = concatSequences( aRet, ODatabaseForm_BASE2::getTypes(), ODatabaseForm_BASE3::getTypes() );
247 return concatSequences( aRet, OPropertySetAggregationHelper::getTypes() );
248 }
249
250 //------------------------------------------------------------------
queryAggregation(const Type & _rType)251 Any SAL_CALL ODatabaseForm::queryAggregation(const Type& _rType) throw(RuntimeException)
252 {
253 Any aReturn = ODatabaseForm_BASE1::queryInterface(_rType);
254 // our own interfaces
255 if (!aReturn.hasValue())
256 {
257 aReturn = ODatabaseForm_BASE2::queryInterface(_rType);
258 // property set related interfaces
259 if (!aReturn.hasValue())
260 {
261 aReturn = OPropertySetAggregationHelper::queryInterface(_rType);
262
263 // form component collection related interfaces
264 if (!aReturn.hasValue())
265 {
266 aReturn = OFormComponents::queryAggregation(_rType);
267
268 // interfaces already present in the aggregate which we want to reroute
269 // only available if we could create the aggregate
270 if (!aReturn.hasValue() && m_xAggregateAsRowSet.is())
271 aReturn = ODatabaseForm_BASE3::queryInterface(_rType);
272
273 // aggregate interfaces
274 // (ask the aggregated object _after_ the OComponentHelper (base of OFormComponents),
275 // so calls to the XComponent interface reach us and not the aggreagtion)
276 if (!aReturn.hasValue() && m_xAggregate.is())
277 aReturn = m_xAggregate->queryAggregation(_rType);
278 }
279 }
280 }
281
282 return aReturn;
283 }
284
285 DBG_NAME(ODatabaseForm);
286 //------------------------------------------------------------------
ODatabaseForm(const Reference<XMultiServiceFactory> & _rxFactory)287 ODatabaseForm::ODatabaseForm(const Reference<XMultiServiceFactory>& _rxFactory)
288 :OFormComponents(_rxFactory)
289 ,OPropertySetAggregationHelper(OComponentHelper::rBHelper)
290 ,OPropertyChangeListener(m_aMutex)
291 ,m_aLoadListeners(m_aMutex)
292 ,m_aRowSetApproveListeners(m_aMutex)
293 ,m_aRowSetListeners(m_aMutex)
294 ,m_aSubmitListeners(m_aMutex)
295 ,m_aErrorListeners(m_aMutex)
296 ,m_aResetListeners( *this, m_aMutex )
297 ,m_aPropertyBagHelper( *this )
298 ,m_pAggregatePropertyMultiplexer(NULL)
299 ,m_pGroupManager( NULL )
300 ,m_aParameterManager( m_aMutex, _rxFactory )
301 ,m_aFilterManager( _rxFactory )
302 ,m_pLoadTimer(NULL)
303 ,m_pThread(NULL)
304 ,m_nResetsPending(0)
305 ,m_nPrivileges(0)
306 ,m_bInsertOnly( sal_False )
307 ,m_eSubmitMethod(FormSubmitMethod_GET)
308 ,m_eSubmitEncoding(FormSubmitEncoding_URL)
309 ,m_eNavigation(NavigationBarMode_CURRENT)
310 ,m_bAllowInsert(sal_True)
311 ,m_bAllowUpdate(sal_True)
312 ,m_bAllowDelete(sal_True)
313 ,m_bLoaded(sal_False)
314 ,m_bSubForm(sal_False)
315 ,m_bForwardingConnection(sal_False)
316 ,m_bSharingConnection( sal_False )
317 {
318 DBG_CTOR( ODatabaseForm, NULL );
319 impl_construct();
320 }
321
322 //------------------------------------------------------------------
ODatabaseForm(const ODatabaseForm & _cloneSource)323 ODatabaseForm::ODatabaseForm( const ODatabaseForm& _cloneSource )
324 :OFormComponents( _cloneSource )
325 ,OPropertySetAggregationHelper( OComponentHelper::rBHelper )
326 ,OPropertyChangeListener( m_aMutex )
327 ,ODatabaseForm_BASE1()
328 ,ODatabaseForm_BASE2()
329 ,ODatabaseForm_BASE3()
330 ,IPropertyBagHelperContext()
331 ,m_aLoadListeners( m_aMutex )
332 ,m_aRowSetApproveListeners( m_aMutex )
333 ,m_aRowSetListeners( m_aMutex )
334 ,m_aSubmitListeners( m_aMutex )
335 ,m_aErrorListeners( m_aMutex )
336 ,m_aResetListeners( *this, m_aMutex )
337 ,m_aPropertyBagHelper( *this )
338 ,m_pAggregatePropertyMultiplexer( NULL )
339 ,m_pGroupManager( NULL )
340 ,m_aParameterManager( m_aMutex, _cloneSource.m_xServiceFactory )
341 ,m_aFilterManager( _cloneSource.m_xServiceFactory )
342 ,m_pLoadTimer( NULL )
343 ,m_pThread( NULL )
344 ,m_nResetsPending( 0 )
345 ,m_nPrivileges( 0 )
346 ,m_bInsertOnly( _cloneSource.m_bInsertOnly )
347 ,m_aControlBorderColorFocus( _cloneSource.m_aControlBorderColorFocus )
348 ,m_aControlBorderColorMouse( _cloneSource.m_aControlBorderColorMouse )
349 ,m_aControlBorderColorInvalid( _cloneSource.m_aControlBorderColorInvalid )
350 ,m_aDynamicControlBorder( _cloneSource.m_aDynamicControlBorder )
351 ,m_sName( _cloneSource.m_sName )
352 ,m_aTargetURL( _cloneSource.m_aTargetURL )
353 ,m_aTargetFrame( _cloneSource.m_aTargetFrame )
354 ,m_eSubmitMethod( _cloneSource.m_eSubmitMethod )
355 ,m_eSubmitEncoding( _cloneSource.m_eSubmitEncoding )
356 ,m_eNavigation( _cloneSource.m_eNavigation )
357 ,m_bAllowInsert( _cloneSource.m_bAllowInsert )
358 ,m_bAllowUpdate( _cloneSource.m_bAllowUpdate )
359 ,m_bAllowDelete( _cloneSource.m_bAllowDelete )
360 ,m_bLoaded( sal_False )
361 ,m_bSubForm( sal_False )
362 ,m_bForwardingConnection( sal_False )
363 ,m_bSharingConnection( sal_False )
364 {
365 DBG_CTOR( ODatabaseForm, NULL );
366
367 impl_construct();
368
369 osl_incrementInterlockedCount( &m_refCount );
370 {
371 // our aggregated rowset itself is not cloneable, so simply copy the properties
372 ::comphelper::copyProperties( _cloneSource.m_xAggregateSet, m_xAggregateSet );
373
374 // also care for the dynamic properties: If the clone source has properties which we do not have,
375 // then add them
376 try
377 {
378 Reference< XPropertySet > xSourceProps( const_cast< ODatabaseForm& >( _cloneSource ).queryAggregation(
379 XPropertySet::static_type() ), UNO_QUERY_THROW );
380 Reference< XPropertySetInfo > xSourcePSI( xSourceProps->getPropertySetInfo(), UNO_SET_THROW );
381 Reference< XPropertyState > xSourcePropState( xSourceProps, UNO_QUERY );
382
383 Reference< XPropertySetInfo > xDestPSI( getPropertySetInfo(), UNO_QUERY_THROW );
384
385 Sequence< Property > aSourceProperties( xSourcePSI->getProperties() );
386 for ( const Property* pSourceProperty = aSourceProperties.getConstArray();
387 pSourceProperty != aSourceProperties.getConstArray() + aSourceProperties.getLength();
388 ++pSourceProperty
389 )
390 {
391 if ( xDestPSI->hasPropertyByName( pSourceProperty->Name ) )
392 continue;
393
394 // the initial value passed to XPropertyContainer is also used as default, usually. So, try
395 // to retrieve the default of the source property
396 Any aInitialValue;
397 if ( xSourcePropState.is() )
398 {
399 aInitialValue = xSourcePropState->getPropertyDefault( pSourceProperty->Name );
400 }
401 else
402 {
403 aInitialValue = xSourceProps->getPropertyValue( pSourceProperty->Name );
404 }
405 addProperty( pSourceProperty->Name, pSourceProperty->Attributes, aInitialValue );
406 setPropertyValue( pSourceProperty->Name, xSourceProps->getPropertyValue( pSourceProperty->Name ) );
407 }
408 }
409 catch( const Exception& )
410 {
411 throw WrappedTargetException(
412 ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Could not clone the given database form." ) ),
413 *const_cast< ODatabaseForm* >( &_cloneSource ),
414 ::cppu::getCaughtException()
415 );
416 }
417 }
418 osl_decrementInterlockedCount( &m_refCount );
419 }
420
421 //------------------------------------------------------------------
impl_construct()422 void ODatabaseForm::impl_construct()
423 {
424 // aggregate a row set
425 increment(m_refCount);
426 {
427 m_xAggregate = Reference< XAggregation >( m_xServiceFactory->createInstance( SRV_SDB_ROWSET ), UNO_QUERY_THROW );
428 m_xAggregateAsRowSet.set( m_xAggregate, UNO_QUERY_THROW );
429 setAggregation( m_xAggregate );
430 }
431
432 // listen for the properties, important for Parameters
433 if ( m_xAggregateSet.is() )
434 {
435 m_pAggregatePropertyMultiplexer = new OPropertyChangeMultiplexer(this, m_xAggregateSet, sal_False);
436 m_pAggregatePropertyMultiplexer->acquire();
437 m_pAggregatePropertyMultiplexer->addProperty(PROPERTY_COMMAND);
438 m_pAggregatePropertyMultiplexer->addProperty(PROPERTY_ACTIVE_CONNECTION);
439 }
440
441 {
442 Reference< XWarningsSupplier > xRowSetWarnings( m_xAggregate, UNO_QUERY );
443 m_aWarnings.setExternalWarnings( xRowSetWarnings );
444 }
445
446 if ( m_xAggregate.is() )
447 {
448 m_xAggregate->setDelegator( static_cast< XWeak* >( this ) );
449 }
450
451 {
452 m_aFilterManager.initialize( m_xAggregateSet );
453 m_aParameterManager.initialize( this, m_xAggregate );
454
455 declareForwardedProperty( PROPERTY_ID_ACTIVE_CONNECTION );
456 }
457 decrement( m_refCount );
458
459 m_pGroupManager = new OGroupManager( this );
460 m_pGroupManager->acquire();
461 }
462
463 //------------------------------------------------------------------
~ODatabaseForm()464 ODatabaseForm::~ODatabaseForm()
465 {
466 DBG_DTOR(ODatabaseForm,NULL);
467
468 m_pGroupManager->release();
469 m_pGroupManager = NULL;
470
471 if (m_xAggregate.is())
472 m_xAggregate->setDelegator( NULL );
473
474 m_aWarnings.setExternalWarnings( NULL );
475
476 if (m_pAggregatePropertyMultiplexer)
477 {
478 m_pAggregatePropertyMultiplexer->dispose();
479 m_pAggregatePropertyMultiplexer->release();
480 m_pAggregatePropertyMultiplexer = NULL;
481 }
482 }
483
484 //==============================================================================
485 // html tools
486 //------------------------------------------------------------------------
GetDataURLEncoded(const Reference<XControl> & SubmitButton,const::com::sun::star::awt::MouseEvent & MouseEvt)487 ::rtl::OUString ODatabaseForm::GetDataURLEncoded(const Reference<XControl>& SubmitButton, const ::com::sun::star::awt::MouseEvent& MouseEvt)
488 {
489 return GetDataEncoded(true,SubmitButton,MouseEvt);
490 }
491 // -----------------------------------------------------------------------------
GetDataEncoded(bool _bURLEncoded,const Reference<XControl> & SubmitButton,const::com::sun::star::awt::MouseEvent & MouseEvt)492 ::rtl::OUString ODatabaseForm::GetDataEncoded(bool _bURLEncoded,const Reference<XControl>& SubmitButton, const ::com::sun::star::awt::MouseEvent& MouseEvt)
493 {
494 // Liste von successful Controls fuellen
495 HtmlSuccessfulObjList aSuccObjList;
496 FillSuccessfulList( aSuccObjList, SubmitButton, MouseEvt );
497
498
499 // Liste zu ::rtl::OUString zusammensetzen
500 ::rtl::OUStringBuffer aResult;
501 ::rtl::OUString aName;
502 ::rtl::OUString aValue;
503
504 for ( HtmlSuccessfulObjListIterator pSuccObj = aSuccObjList.begin();
505 pSuccObj < aSuccObjList.end();
506 ++pSuccObj
507 )
508 {
509 aName = pSuccObj->aName;
510 aValue = pSuccObj->aValue;
511 if( pSuccObj->nRepresentation == SUCCESSFUL_REPRESENT_FILE && aValue.getLength() )
512 {
513 // Bei File-URLs wird der Dateiname und keine URL uebertragen,
514 // weil Netscape dies so macht.
515 INetURLObject aURL;
516 aURL.SetSmartProtocol(INET_PROT_FILE);
517 aURL.SetSmartURL(aValue);
518 if( INET_PROT_FILE == aURL.GetProtocol() )
519 aValue = INetURLObject::decode(aURL.PathToFileName(), '%', INetURLObject::DECODE_UNAMBIGUOUS);
520 }
521 Encode( aName );
522 Encode( aValue );
523
524 aResult.append(aName);
525 aResult.append(sal_Unicode('='));
526 aResult.append(aValue);
527
528 if (pSuccObj < aSuccObjList.end() - 1)
529 {
530 if ( _bURLEncoded )
531 aResult.append(sal_Unicode('&'));
532 else
533 aResult.appendAscii("\r\n");
534 }
535 }
536
537
538 aSuccObjList.clear();
539
540 return aResult.makeStringAndClear();
541 }
542
543 //==============================================================================
544 // html tools
545 //------------------------------------------------------------------------
GetDataTextEncoded(const Reference<XControl> & SubmitButton,const::com::sun::star::awt::MouseEvent & MouseEvt)546 ::rtl::OUString ODatabaseForm::GetDataTextEncoded(const Reference<XControl>& SubmitButton, const ::com::sun::star::awt::MouseEvent& MouseEvt)
547 {
548 return GetDataEncoded(false,SubmitButton,MouseEvt);
549 }
550
551 //------------------------------------------------------------------------
GetDataMultiPartEncoded(const Reference<XControl> & SubmitButton,const::com::sun::star::awt::MouseEvent & MouseEvt,::rtl::OUString & rContentType)552 Sequence<sal_Int8> ODatabaseForm::GetDataMultiPartEncoded(const Reference<XControl>& SubmitButton, const ::com::sun::star::awt::MouseEvent& MouseEvt, ::rtl::OUString& rContentType)
553 {
554
555 // Parent erzeugen
556 INetMIMEMessage aParent;
557 aParent.EnableAttachChild( INETMSG_MULTIPART_FORM_DATA );
558
559
560 // Liste von successful Controls fuellen
561 HtmlSuccessfulObjList aSuccObjList;
562 FillSuccessfulList( aSuccObjList, SubmitButton, MouseEvt );
563
564
565 // Liste zu ::rtl::OUString zusammensetzen
566 ::rtl::OUString aResult;
567 for ( HtmlSuccessfulObjListIterator pSuccObj = aSuccObjList.begin();
568 pSuccObj < aSuccObjList.end();
569 ++pSuccObj
570 )
571 {
572 if( pSuccObj->nRepresentation == SUCCESSFUL_REPRESENT_TEXT )
573 InsertTextPart( aParent, pSuccObj->aName, pSuccObj->aValue );
574 else if( pSuccObj->nRepresentation == SUCCESSFUL_REPRESENT_FILE )
575 InsertFilePart( aParent, pSuccObj->aName, pSuccObj->aValue );
576 }
577
578
579 // Liste loeschen
580 aSuccObjList.clear();
581
582 // Fuer Parent MessageStream erzeugen
583 INetMIMEMessageStream aMessStream;
584 aMessStream.SetSourceMessage( &aParent );
585 aMessStream.GenerateHeader( sal_False );
586
587 // MessageStream in SvStream kopieren
588 SvMemoryStream aMemStream;
589 char* pBuf = new char[1025];
590 int nRead;
591 while( (nRead = aMessStream.Read(pBuf, 1024)) > 0 )
592 aMemStream.Write( pBuf, nRead );
593 delete[] pBuf;
594
595 aMemStream.Flush();
596 aMemStream.Seek( 0 );
597 void* pData = (void*)aMemStream.GetData();
598 sal_Int32 nLen = aMemStream.Seek(STREAM_SEEK_TO_END);
599
600 rContentType = UniString(aParent.GetContentType());
601 return Sequence<sal_Int8>((sal_Int8*)pData, nLen);
602 }
603
604 //------------------------------------------------------------------------
605 namespace
606 {
appendDigits(sal_Int32 _nNumber,sal_Int8 nDigits,::rtl::OUStringBuffer & _rOut)607 static void appendDigits( sal_Int32 _nNumber, sal_Int8 nDigits, ::rtl::OUStringBuffer& _rOut )
608 {
609 sal_Int32 nCurLen = _rOut.getLength();
610 _rOut.append( _nNumber );
611 while ( _rOut.getLength() - nCurLen < nDigits )
612 _rOut.insert( nCurLen, (sal_Unicode)'0' );
613 }
614 }
615
616 //------------------------------------------------------------------------
AppendComponent(HtmlSuccessfulObjList & rList,const Reference<XPropertySet> & xComponentSet,const::rtl::OUString & rNamePrefix,const Reference<XControl> & rxSubmitButton,const::com::sun::star::awt::MouseEvent & MouseEvt)617 void ODatabaseForm::AppendComponent(HtmlSuccessfulObjList& rList, const Reference<XPropertySet>& xComponentSet, const ::rtl::OUString& rNamePrefix,
618 const Reference<XControl>& rxSubmitButton, const ::com::sun::star::awt::MouseEvent& MouseEvt)
619 {
620 if (!xComponentSet.is())
621 return;
622
623 // MIB 25.6.98: Geschachtelte Formulare abfangen ... oder muesste
624 // man sie submitten?
625 if (!hasProperty(PROPERTY_CLASSID, xComponentSet))
626 return;
627
628 // Namen ermitteln
629 if (!hasProperty(PROPERTY_NAME, xComponentSet))
630 return;
631
632 sal_Int16 nClassId = 0;
633 xComponentSet->getPropertyValue(PROPERTY_CLASSID) >>= nClassId;
634 ::rtl::OUString aName;
635 xComponentSet->getPropertyValue( PROPERTY_NAME ) >>= aName;
636 if( !aName.getLength() && nClassId != FormComponentType::IMAGEBUTTON)
637 return;
638 else // Name um den Prefix erweitern
639 aName = rNamePrefix + aName;
640
641 switch( nClassId )
642 {
643 // Buttons
644 case FormComponentType::COMMANDBUTTON:
645 {
646 // Es wird nur der gedrueckte Submit-Button ausgewertet
647 // MIB: Sofern ueberhaupt einer uebergeben wurde
648 if( rxSubmitButton.is() )
649 {
650 Reference<XPropertySet> xSubmitButtonComponent(rxSubmitButton->getModel(), UNO_QUERY);
651 if (xSubmitButtonComponent == xComponentSet && hasProperty(PROPERTY_LABEL, xComponentSet))
652 {
653 // <name>=<label>
654 ::rtl::OUString aLabel;
655 xComponentSet->getPropertyValue( PROPERTY_LABEL ) >>= aLabel;
656 rList.push_back( HtmlSuccessfulObj(aName, aLabel) );
657 }
658 }
659 } break;
660
661 // ImageButtons
662 case FormComponentType::IMAGEBUTTON:
663 {
664 // Es wird nur der gedrueckte Submit-Button ausgewertet
665 // MIB: Sofern ueberhaupt einer uebergeben wurde
666 if( rxSubmitButton.is() )
667 {
668 Reference<XPropertySet> xSubmitButtonComponent(rxSubmitButton->getModel(), UNO_QUERY);
669 if (xSubmitButtonComponent == xComponentSet)
670 {
671 // <name>.x=<pos.X>&<name>.y=<pos.Y>
672 ::rtl::OUString aLhs = aName;
673 ::rtl::OUString aRhs = ::rtl::OUString::valueOf( MouseEvt.X );
674
675 // nur wenn ein Name vorhanden ist, kann ein name.x
676 aLhs += aName.getLength() ? UniString::CreateFromAscii(".x") : UniString::CreateFromAscii("x");
677 rList.push_back( HtmlSuccessfulObj(aLhs, aRhs) );
678
679 aLhs = aName;
680 aRhs = ::rtl::OUString::valueOf( MouseEvt.Y );
681 aLhs += aName.getLength() ? UniString::CreateFromAscii(".y") : UniString::CreateFromAscii("y");
682 rList.push_back( HtmlSuccessfulObj(aLhs, aRhs) );
683
684 }
685 }
686 } break;
687
688 // CheckBoxen / RadioButtons
689 case FormComponentType::CHECKBOX:
690 case FormComponentType::RADIOBUTTON:
691 {
692 // <name>=<refValue>
693 if( !hasProperty(PROPERTY_STATE, xComponentSet) )
694 break;
695 sal_Int16 nChecked = 0;
696 xComponentSet->getPropertyValue( PROPERTY_STATE ) >>= nChecked;
697 if( nChecked != 1 )
698 break;
699
700 ::rtl::OUString aStrValue;
701 if( hasProperty(PROPERTY_REFVALUE, xComponentSet) )
702 xComponentSet->getPropertyValue( PROPERTY_REFVALUE ) >>= aStrValue;
703
704 rList.push_back( HtmlSuccessfulObj(aName, aStrValue) );
705 } break;
706
707 // Edit
708 case FormComponentType::TEXTFIELD:
709 {
710 // <name>=<text>
711 if( !hasProperty(PROPERTY_TEXT, xComponentSet) )
712 break;
713
714 // MIB: Spezial-Behandlung fuer Multiline-Edit nur dann, wenn
715 // es auch ein Control dazu gibt.
716 Any aTmp = xComponentSet->getPropertyValue( PROPERTY_MULTILINE );
717 sal_Bool bMulti = rxSubmitButton.is()
718 && (aTmp.getValueType().getTypeClass() == TypeClass_BOOLEAN)
719 && getBOOL(aTmp);
720 ::rtl::OUString sText;
721 if ( bMulti ) // Bei MultiLineEdit Text am Control abholen
722 {
723
724 Reference<XControlContainer> xControlContainer(rxSubmitButton->getContext(), UNO_QUERY);
725 if( !xControlContainer.is() ) break;
726
727 Sequence<Reference<XControl> > aControlSeq = xControlContainer->getControls();
728 Reference<XControl> xControl;
729 Reference<XFormComponent> xControlComponent;
730
731 // Richtiges Control suchen
732 sal_Int32 i;
733 for( i=0; i<aControlSeq.getLength(); i++ )
734 {
735 xControl = aControlSeq.getConstArray()[i];
736 Reference<XPropertySet> xModel(xControl->getModel(), UNO_QUERY);
737 if (xModel == xComponentSet)
738 {
739 Reference<XTextComponent> xTextComponent(xControl, UNO_QUERY);
740 if( xTextComponent.is() )
741 sText = xTextComponent->getText();
742 break;
743 }
744 }
745 // Control nicht gefunden oder nicht existent, (Edit im Grid)
746 if (i == aControlSeq.getLength())
747 xComponentSet->getPropertyValue( PROPERTY_TEXT ) >>= sText;
748 }
749 else
750 xComponentSet->getPropertyValue( PROPERTY_TEXT ) >>= sText;
751
752 rList.push_back( HtmlSuccessfulObj(aName, sText) );
753 } break;
754
755 // ComboBox, Patternfield
756 case FormComponentType::COMBOBOX:
757 case FormComponentType::PATTERNFIELD:
758 {
759 // <name>=<text>
760 if( hasProperty(PROPERTY_TEXT, xComponentSet) )
761 {
762 ::rtl::OUString aText;
763 xComponentSet->getPropertyValue( PROPERTY_TEXT ) >>= aText;
764 rList.push_back( HtmlSuccessfulObj(aName, aText) );
765 }
766 } break;
767 case FormComponentType::CURRENCYFIELD:
768 case FormComponentType::NUMERICFIELD:
769 {
770 // <name>=<wert> // wert wird als double mit Punkt als Decimaltrenner
771 // kein Wert angegeben (NULL) -> wert leer
772 if( hasProperty(PROPERTY_VALUE, xComponentSet) )
773 {
774 ::rtl::OUString aText;
775 Any aVal = xComponentSet->getPropertyValue( PROPERTY_VALUE );
776
777 double aDoubleVal = 0;
778 if (aVal >>= aDoubleVal)
779 {
780 sal_Int16 nScale = 0;
781 xComponentSet->getPropertyValue( PROPERTY_DECIMAL_ACCURACY ) >>= nScale;
782 aText = ::rtl::math::doubleToUString(aDoubleVal, rtl_math_StringFormat_F, nScale, '.', sal_True);
783 }
784 rList.push_back( HtmlSuccessfulObj(aName, aText) );
785 }
786 } break;
787 case FormComponentType::DATEFIELD:
788 {
789 // <name>=<wert> // Wert wird als Datum im Format (MM-DD-YYYY)
790 // kein Wert angegeben (NULL) -> wert leer
791 if( hasProperty(PROPERTY_DATE, xComponentSet) )
792 {
793 ::rtl::OUString aText;
794 Any aVal = xComponentSet->getPropertyValue( PROPERTY_DATE );
795 sal_Int32 nInt32Val = 0;
796 if (aVal >>= nInt32Val)
797 {
798 ::Date aDate( nInt32Val );
799 ::rtl::OUStringBuffer aBuffer;
800 appendDigits( aDate.GetMonth(), 2, aBuffer );
801 aBuffer.append( (sal_Unicode)'-' );
802 appendDigits( aDate.GetDay(), 2, aBuffer );
803 aBuffer.append( (sal_Unicode)'-' );
804 appendDigits( aDate.GetYear(), 4, aBuffer );
805 aText = aBuffer.makeStringAndClear();
806 }
807 rList.push_back( HtmlSuccessfulObj(aName, aText) );
808 }
809 } break;
810 case FormComponentType::TIMEFIELD:
811 {
812 // <name>=<wert> // Wert wird als Zeit im Format (HH:MM:SS) angegeben
813 // kein Wert angegeben (NULL) -> wert leer
814 if( hasProperty(PROPERTY_TIME, xComponentSet) )
815 {
816 ::rtl::OUString aText;
817 Any aVal = xComponentSet->getPropertyValue( PROPERTY_TIME );
818 sal_Int32 nInt32Val = 0;
819 if (aVal >>= nInt32Val)
820 {
821 ::Time aTime(nInt32Val);
822 ::rtl::OUStringBuffer aBuffer;
823 appendDigits( aTime.GetHour(), 2, aBuffer );
824 aBuffer.append( (sal_Unicode)'-' );
825 appendDigits( aTime.GetMin(), 2, aBuffer );
826 aBuffer.append( (sal_Unicode)'-' );
827 appendDigits( aTime.GetSec(), 2, aBuffer );
828 aText = aBuffer.makeStringAndClear();
829 }
830 rList.push_back( HtmlSuccessfulObj(aName, aText) );
831 }
832 } break;
833
834 // starform
835 case FormComponentType::HIDDENCONTROL:
836 {
837
838 // <name>=<value>
839 if( hasProperty(PROPERTY_HIDDEN_VALUE, xComponentSet) )
840 {
841 ::rtl::OUString aText;
842 xComponentSet->getPropertyValue( PROPERTY_HIDDEN_VALUE ) >>= aText;
843 rList.push_back( HtmlSuccessfulObj(aName, aText) );
844 }
845 } break;
846
847 // starform
848 case FormComponentType::FILECONTROL:
849 {
850 // <name>=<text>
851 if( hasProperty(PROPERTY_TEXT, xComponentSet) )
852 {
853
854 ::rtl::OUString aText;
855 xComponentSet->getPropertyValue( PROPERTY_TEXT ) >>= aText;
856 rList.push_back( HtmlSuccessfulObj(aName, aText, SUCCESSFUL_REPRESENT_FILE) );
857 }
858 } break;
859
860 // starform
861 case FormComponentType::LISTBOX:
862 {
863
864 // <name>=<Token0>&<name>=<Token1>&...&<name>=<TokenN> (Mehrfachselektion)
865 if (!hasProperty(PROPERTY_SELECT_SEQ, xComponentSet) ||
866 !hasProperty(PROPERTY_STRINGITEMLIST, xComponentSet))
867 break;
868
869 // angezeigte Werte
870 Sequence< ::rtl::OUString > aVisibleList;
871 xComponentSet->getPropertyValue( PROPERTY_STRINGITEMLIST ) >>= aVisibleList;
872 sal_Int32 nStringCnt = aVisibleList.getLength();
873 const ::rtl::OUString* pStrings = aVisibleList.getConstArray();
874
875 // Werte-Liste
876 Sequence< ::rtl::OUString > aValueList;
877 xComponentSet->getPropertyValue( PROPERTY_VALUE_SEQ ) >>= aValueList;
878 sal_Int32 nValCnt = aValueList.getLength();
879 const ::rtl::OUString* pVals = aValueList.getConstArray();
880
881 // Selektion
882 Sequence<sal_Int16> aSelectList;
883 xComponentSet->getPropertyValue( PROPERTY_SELECT_SEQ ) >>= aSelectList;
884 sal_Int32 nSelCount = aSelectList.getLength();
885 const sal_Int16* pSels = aSelectList.getConstArray();
886
887 // Einfach- oder Mehrfach-Selektion
888 // Bei Einfach-Selektionen beruecksichtigt MT nur den ersten Eintrag
889 // in der Liste.
890 if (nSelCount > 1 && !getBOOL(xComponentSet->getPropertyValue(PROPERTY_MULTISELECTION)))
891 nSelCount = 1;
892
893 // Die Indizes in der Selektions-Liste koennen auch ungueltig sein,
894 // also muss man die gueltigen erstmal raussuchen um die Laenge
895 // der neuen Liste zu bestimmen.
896 sal_Int32 nCurCnt = 0;
897 sal_Int32 i;
898 for( i=0; i<nSelCount; ++i )
899 {
900 if( pSels[i] < nStringCnt )
901 ++nCurCnt;
902 }
903
904 ::rtl::OUString aSubValue;
905 for(i=0; i<nCurCnt; ++i )
906 {
907 sal_Int16 nSelPos = pSels[i];
908 if (nSelPos < nValCnt && pVals[nSelPos].getLength())
909 {
910 aSubValue = pVals[nSelPos];
911 }
912 else
913 {
914 aSubValue = pStrings[nSelPos];
915 }
916 rList.push_back( HtmlSuccessfulObj(aName, aSubValue) );
917 }
918 } break;
919 case FormComponentType::GRIDCONTROL:
920 {
921 // Die einzelnen Spaltenwerte werden verschickt,
922 // der Name wird mit dem Prefix des Names des Grids erweitert
923 Reference<XIndexAccess> xContainer(xComponentSet, UNO_QUERY);
924 if (!xContainer.is())
925 break;
926
927 aName += UniString('.');
928
929 Reference<XPropertySet> xSet;
930 sal_Int32 nCount = xContainer->getCount();
931 // we know already how many objects should be appended,
932 // so why not allocate the space for them
933 rList.reserve( nCount + rList.capacity() ); // not size()
934 for (sal_Int32 i = 0; i < nCount; ++i)
935 {
936 xContainer->getByIndex(i) >>= xSet;
937 if (xSet.is())
938 AppendComponent(rList, xSet, aName, rxSubmitButton, MouseEvt);
939 }
940 }
941 }
942 }
943
944 //------------------------------------------------------------------------
FillSuccessfulList(HtmlSuccessfulObjList & rList,const Reference<XControl> & rxSubmitButton,const::com::sun::star::awt::MouseEvent & MouseEvt)945 void ODatabaseForm::FillSuccessfulList( HtmlSuccessfulObjList& rList,
946 const Reference<XControl>& rxSubmitButton, const ::com::sun::star::awt::MouseEvent& MouseEvt )
947 {
948 // Liste loeschen
949 rList.clear();
950 // Ueber Components iterieren
951 Reference<XPropertySet> xComponentSet;
952 ::rtl::OUString aPrefix;
953
954 // we know already how many objects should be appended,
955 // so why not allocate the space for them
956 rList.reserve( getCount() );
957 for( sal_Int32 nIndex=0; nIndex < getCount(); nIndex++ )
958 {
959 getByIndex( nIndex ) >>= xComponentSet;
960 AppendComponent(rList, xComponentSet, aPrefix, rxSubmitButton, MouseEvt);
961 }
962 }
963
964 //------------------------------------------------------------------------
Encode(::rtl::OUString & rString) const965 void ODatabaseForm::Encode( ::rtl::OUString& rString ) const
966 {
967 ::rtl::OUString aResult;
968
969 // Immer ANSI #58641
970 // rString.Convert(CHARSET_SYSTEM, CHARSET_ANSI);
971
972
973 // Zeilenendezeichen werden als CR dargestellt
974 UniString sConverter = rString;
975 sConverter.ConvertLineEnd( LINEEND_CR );
976 rString = sConverter;
977
978
979 // Jeden einzelnen Character ueberpruefen
980 sal_Int32 nStrLen = rString.getLength();
981 sal_Unicode nCharCode;
982 for( sal_Int32 nCurPos=0; nCurPos < nStrLen; ++nCurPos )
983 {
984 nCharCode = rString[nCurPos];
985
986 // Behandlung fuer chars, die kein alphanumerisches Zeichen sind
987 // und CharacterCodes > 127
988 if( (!isalnum(nCharCode) && nCharCode != (sal_Unicode)' ') || nCharCode > 127 )
989 {
990 switch( nCharCode )
991 {
992 case 13: // CR
993 aResult += ::rtl::OUString::createFromAscii("%0D%0A"); // Hex-Darstellung CR LF
994 break;
995
996
997 // Netscape Sonderbehandlung
998 case 42: // '*'
999 case 45: // '-'
1000 case 46: // '.'
1001 case 64: // '@'
1002 case 95: // '_'
1003 aResult += UniString(nCharCode);
1004 break;
1005
1006 default:
1007 {
1008 // In Hex umrechnen
1009 short nHi = ((sal_Int16)nCharCode) / 16;
1010 short nLo = ((sal_Int16)nCharCode) - (nHi*16);
1011 if( nHi > 9 ) nHi += (int)'A'-10; else nHi += (int)'0';
1012 if( nLo > 9 ) nLo += (int)'A'-10; else nLo += (int)'0';
1013 aResult += UniString('%');
1014 aResult += UniString((sal_Unicode)nHi);
1015 aResult += UniString((sal_Unicode)nLo);
1016 }
1017 }
1018 }
1019 else
1020 aResult += UniString(nCharCode);
1021 }
1022
1023
1024 // Spaces durch '+' ersetzen
1025 aResult = aResult.replace(' ', '+');
1026
1027 rString = aResult;
1028 }
1029
1030 //------------------------------------------------------------------------
InsertTextPart(INetMIMEMessage & rParent,const::rtl::OUString & rName,const::rtl::OUString & rData)1031 void ODatabaseForm::InsertTextPart( INetMIMEMessage& rParent, const ::rtl::OUString& rName,
1032 const ::rtl::OUString& rData )
1033 {
1034
1035 // Part als Message-Child erzeugen
1036 INetMIMEMessage* pChild = new INetMIMEMessage();
1037
1038
1039 // Header
1040 ::rtl::OUString aContentDisp = ::rtl::OUString::createFromAscii("form-data; name=\"");
1041 aContentDisp += rName;
1042 aContentDisp += UniString('\"');
1043 pChild->SetContentDisposition( aContentDisp );
1044 pChild->SetContentType( UniString::CreateFromAscii("text/plain") );
1045
1046 rtl_TextEncoding eSystemEncoding = gsl_getSystemTextEncoding();
1047 const sal_Char* pBestMatchingEncoding = rtl_getBestMimeCharsetFromTextEncoding( eSystemEncoding );
1048 UniString aBestMatchingEncoding = UniString::CreateFromAscii( pBestMatchingEncoding );
1049 pChild->SetContentTransferEncoding(aBestMatchingEncoding);
1050
1051 // Body
1052 SvMemoryStream* pStream = new SvMemoryStream;
1053 pStream->WriteLine( ByteString( UniString(rData), rtl_getTextEncodingFromMimeCharset(pBestMatchingEncoding) ) );
1054 pStream->Flush();
1055 pStream->Seek( 0 );
1056 pChild->SetDocumentLB( new SvLockBytes(pStream, sal_True) );
1057 rParent.AttachChild( *pChild );
1058 }
1059
1060 //------------------------------------------------------------------------
InsertFilePart(INetMIMEMessage & rParent,const::rtl::OUString & rName,const::rtl::OUString & rFileName)1061 sal_Bool ODatabaseForm::InsertFilePart( INetMIMEMessage& rParent, const ::rtl::OUString& rName,
1062 const ::rtl::OUString& rFileName )
1063 {
1064 UniString aFileName( rFileName );
1065 UniString aContentType(UniString::CreateFromAscii(CONTENT_TYPE_STR_TEXT_PLAIN));
1066 SvStream *pStream = 0;
1067
1068 if( aFileName.Len() )
1069 {
1070 // Bisher koennen wir nur File-URLs verarbeiten
1071 INetURLObject aURL;
1072 aURL.SetSmartProtocol(INET_PROT_FILE);
1073 aURL.SetSmartURL(rFileName);
1074 if( INET_PROT_FILE == aURL.GetProtocol() )
1075 {
1076 aFileName = INetURLObject::decode(aURL.PathToFileName(), '%', INetURLObject::DECODE_UNAMBIGUOUS);
1077 DirEntry aDirEntry( aFileName );
1078 if( aDirEntry.Exists() )
1079 {
1080 pStream = ::utl::UcbStreamHelper::CreateStream(aFileName, STREAM_READ);
1081 if (!pStream || (pStream->GetError() != ERRCODE_NONE))
1082 {
1083 delete pStream;
1084 pStream = 0;
1085 }
1086 }
1087 INetContentType eContentType = INetContentTypes::GetContentType4Extension(
1088 aDirEntry.GetExtension() );
1089 if (eContentType != CONTENT_TYPE_UNKNOWN)
1090 aContentType = INetContentTypes::GetContentType(eContentType);
1091 }
1092 }
1093
1094 // Wenn irgendetwas nicht geklappt hat, legen wir einen leeren
1095 // MemoryStream an
1096 if( !pStream )
1097 pStream = new SvMemoryStream;
1098
1099
1100 // Part als Message-Child erzeugen
1101 INetMIMEMessage* pChild = new INetMIMEMessage;
1102
1103
1104 // Header
1105 ::rtl::OUString aContentDisp = ::rtl::OUString::createFromAscii( "form-data; name=\"" );
1106 aContentDisp += rName;
1107 aContentDisp += UniString('\"');
1108 aContentDisp += ::rtl::OUString::createFromAscii("; filename=\"");
1109 aContentDisp += aFileName;
1110 aContentDisp += UniString('\"');
1111 pChild->SetContentDisposition( aContentDisp );
1112 pChild->SetContentType( aContentType );
1113 pChild->SetContentTransferEncoding( UniString(::rtl::OUString::createFromAscii("8bit")) );
1114
1115
1116 // Body
1117 pChild->SetDocumentLB( new SvLockBytes(pStream, sal_True) );
1118 rParent.AttachChild( *pChild );
1119
1120 return sal_True;
1121 }
1122
1123 //==============================================================================
1124 // internals
1125 //------------------------------------------------------------------------------
onError(const SQLErrorEvent & _rEvent)1126 void ODatabaseForm::onError( const SQLErrorEvent& _rEvent )
1127 {
1128 m_aErrorListeners.notifyEach( &XSQLErrorListener::errorOccured, _rEvent );
1129 }
1130
1131 //------------------------------------------------------------------------------
onError(const SQLException & _rException,const::rtl::OUString & _rContextDescription)1132 void ODatabaseForm::onError( const SQLException& _rException, const ::rtl::OUString& _rContextDescription )
1133 {
1134 if ( !m_aErrorListeners.getLength() )
1135 return;
1136
1137 SQLErrorEvent aEvent( *this, makeAny( prependErrorInfo( _rException, *this, _rContextDescription ) ) );
1138 onError( aEvent );
1139 }
1140
1141 //------------------------------------------------------------------------------
updateParameterInfo()1142 void ODatabaseForm::updateParameterInfo()
1143 {
1144 m_aParameterManager.updateParameterInfo( m_aFilterManager );
1145 }
1146
1147 //------------------------------------------------------------------------------
hasValidParent() const1148 bool ODatabaseForm::hasValidParent() const
1149 {
1150 // do we have to fill the parameters again?
1151 if (m_bSubForm)
1152 {
1153 Reference<XResultSet> xResultSet(m_xParent, UNO_QUERY);
1154 if (!xResultSet.is())
1155 {
1156 DBG_ERROR("ODatabaseForm::hasValidParent() : no parent resultset !");
1157 return false;
1158 }
1159 try
1160 {
1161 Reference< XPropertySet > xSet( m_xParent, UNO_QUERY );
1162 Reference< XLoadable > xLoad( m_xParent, UNO_QUERY );
1163 if ( xLoad->isLoaded()
1164 && ( xResultSet->isBeforeFirst()
1165 || xResultSet->isAfterLast()
1166 || getBOOL( xSet->getPropertyValue( PROPERTY_ISNEW ) )
1167 )
1168 )
1169 // the parent form is loaded and on a "virtual" row -> not valid
1170 return false;
1171 }
1172 catch(Exception&)
1173 {
1174 // parent could be forwardonly?
1175 return false;
1176 }
1177 }
1178 return true;
1179 }
1180
1181 //------------------------------------------------------------------------------
fillParameters(::osl::ResettableMutexGuard & _rClearForNotifies,const Reference<XInteractionHandler> & _rxCompletionHandler)1182 bool ODatabaseForm::fillParameters( ::osl::ResettableMutexGuard& _rClearForNotifies, const Reference< XInteractionHandler >& _rxCompletionHandler )
1183 {
1184 // do we have to fill the parameters again?
1185 if ( !m_aParameterManager.isUpToDate() )
1186 updateParameterInfo();
1187
1188 // is there a valid parent?
1189 if ( m_bSubForm && !hasValidParent() )
1190 return true;
1191
1192 // ensure we're connected
1193 if ( !implEnsureConnection() )
1194 return false;
1195
1196 if ( m_aParameterManager.isUpToDate() )
1197 return m_aParameterManager.fillParameterValues( _rxCompletionHandler, _rClearForNotifies );
1198
1199 return true;
1200 }
1201
1202 //------------------------------------------------------------------------------
saveInsertOnlyState()1203 void ODatabaseForm::saveInsertOnlyState( )
1204 {
1205 OSL_ENSURE( !m_aIgnoreResult.hasValue(), "ODatabaseForm::saveInsertOnlyState: overriding old value!" );
1206 m_aIgnoreResult = m_xAggregateSet->getPropertyValue( PROPERTY_INSERTONLY );
1207 }
1208
1209 //------------------------------------------------------------------------------
restoreInsertOnlyState()1210 void ODatabaseForm::restoreInsertOnlyState( )
1211 {
1212 if ( m_aIgnoreResult.hasValue() )
1213 {
1214 m_xAggregateSet->setPropertyValue( PROPERTY_INSERTONLY, m_aIgnoreResult );
1215 m_aIgnoreResult = Any();
1216 }
1217 }
1218
1219 //------------------------------------------------------------------------------
executeRowSet(::osl::ResettableMutexGuard & _rClearForNotifies,sal_Bool bMoveToFirst,const Reference<XInteractionHandler> & _rxCompletionHandler)1220 sal_Bool ODatabaseForm::executeRowSet(::osl::ResettableMutexGuard& _rClearForNotifies, sal_Bool bMoveToFirst, const Reference< XInteractionHandler >& _rxCompletionHandler)
1221 {
1222 if (!m_xAggregateAsRowSet.is())
1223 return sal_False;
1224
1225 if (!fillParameters(_rClearForNotifies, _rxCompletionHandler))
1226 return sal_False;
1227
1228 restoreInsertOnlyState( );
1229
1230 // ensure the aggregated row set has the correct properties
1231 sal_Int32 nConcurrency = ResultSetConcurrency::READ_ONLY;
1232
1233 // if we have a parent, who is not positioned on a valid row
1234 // we can't be updatable!
1235 if (m_bSubForm && !hasValidParent())
1236 {
1237 nConcurrency = ResultSetConcurrency::READ_ONLY;
1238
1239 // don't use any parameters if we don't have a valid parent
1240 m_aParameterManager.setAllParametersNull();
1241
1242 // switch to "insert only" mode
1243 saveInsertOnlyState( );
1244 m_xAggregateSet->setPropertyValue( PROPERTY_INSERTONLY, makeAny( sal_True ) );
1245 }
1246 else if (m_bAllowInsert || m_bAllowUpdate || m_bAllowDelete)
1247 nConcurrency = ResultSetConcurrency::UPDATABLE;
1248 else
1249 nConcurrency = ResultSetConcurrency::READ_ONLY;
1250
1251 m_xAggregateSet->setPropertyValue( PROPERTY_RESULTSET_CONCURRENCY, makeAny( (sal_Int32)nConcurrency ) );
1252 m_xAggregateSet->setPropertyValue( PROPERTY_RESULTSET_TYPE, makeAny( (sal_Int32)ResultSetType::SCROLL_SENSITIVE ) );
1253
1254 sal_Bool bSuccess = sal_False;
1255 try
1256 {
1257 m_xAggregateAsRowSet->execute();
1258 bSuccess = sal_True;
1259 }
1260 catch( const RowSetVetoException& eVeto )
1261 {
1262 (void)eVeto;
1263 }
1264 catch(SQLException& eDb)
1265 {
1266 _rClearForNotifies.clear();
1267 if (m_sCurrentErrorContext.getLength())
1268 onError(eDb, m_sCurrentErrorContext);
1269 else
1270 onError(eDb, FRM_RES_STRING(RID_STR_READERROR));
1271 _rClearForNotifies.reset();
1272
1273 restoreInsertOnlyState( );
1274 }
1275
1276 if (bSuccess)
1277 {
1278 // adjust the privilege property
1279 // m_nPrivileges;
1280 m_xAggregateSet->getPropertyValue(PROPERTY_PRIVILEGES) >>= m_nPrivileges;
1281 if (!m_bAllowInsert)
1282 m_nPrivileges &= ~Privilege::INSERT;
1283 if (!m_bAllowUpdate)
1284 m_nPrivileges &= ~Privilege::UPDATE;
1285 if (!m_bAllowDelete)
1286 m_nPrivileges &= ~Privilege::DELETE;
1287
1288 if (bMoveToFirst)
1289 {
1290 // the row set is positioned _before_ the first row (per definitionem), so move the set ...
1291 try
1292 {
1293 // if we have an insert only rowset we move to the insert row
1294 next();
1295 if (((m_nPrivileges & Privilege::INSERT) == Privilege::INSERT)
1296 && isAfterLast())
1297 {
1298 // move on the insert row of set
1299 // resetting must be done later, after the load events have been posted
1300 // see :moveToInsertRow and load , reload
1301 Reference<XResultSetUpdate> xUpdate;
1302 if (query_aggregation( m_xAggregate, xUpdate))
1303 xUpdate->moveToInsertRow();
1304 }
1305 }
1306 catch(SQLException& eDB)
1307 {
1308 _rClearForNotifies.clear();
1309 if (m_sCurrentErrorContext.getLength())
1310 onError(eDB, m_sCurrentErrorContext);
1311 else
1312 onError(eDB, FRM_RES_STRING(RID_STR_READERROR));
1313 _rClearForNotifies.reset();
1314 bSuccess = sal_False;
1315 }
1316 }
1317 }
1318 return bSuccess;
1319 }
1320
1321 //------------------------------------------------------------------
disposing()1322 void ODatabaseForm::disposing()
1323 {
1324 if (m_pAggregatePropertyMultiplexer)
1325 m_pAggregatePropertyMultiplexer->dispose();
1326
1327 if (m_bLoaded)
1328 unload();
1329
1330 // cancel the submit/reset-thread
1331 {
1332 ::osl::MutexGuard aGuard( m_aMutex );
1333 if (m_pThread)
1334 {
1335 m_pThread->release();
1336 m_pThread = NULL;
1337 }
1338 }
1339
1340 EventObject aEvt(static_cast<XWeak*>(this));
1341 m_aLoadListeners.disposeAndClear(aEvt);
1342 m_aRowSetApproveListeners.disposeAndClear(aEvt);
1343 m_aParameterManager.disposing( aEvt );
1344 m_aResetListeners.disposing();
1345 m_aSubmitListeners.disposeAndClear(aEvt);
1346 m_aErrorListeners.disposeAndClear(aEvt);
1347
1348 m_aParameterManager.dispose(); // (to free any references it may have to me)
1349 m_aFilterManager.dispose(); // (dito)
1350
1351 OFormComponents::disposing();
1352 OPropertySetAggregationHelper::disposing();
1353
1354 // stop listening on the aggregate
1355 if (m_xAggregateAsRowSet.is())
1356 m_xAggregateAsRowSet->removeRowSetListener(this);
1357
1358 // dispose the active connection
1359 Reference<XComponent> xAggregationComponent;
1360 if (query_aggregation(m_xAggregate, xAggregationComponent))
1361 xAggregationComponent->dispose();
1362
1363 m_aPropertyBagHelper.dispose();
1364 }
1365
1366 //------------------------------------------------------------------------------
getConnection()1367 Reference< XConnection > ODatabaseForm::getConnection()
1368 {
1369 Reference< XConnection > xConn;
1370 m_xAggregateSet->getPropertyValue( PROPERTY_ACTIVE_CONNECTION ) >>= xConn;
1371 return xConn;
1372 }
1373
1374 //------------------------------------------------------------------------------
getMutex()1375 ::osl::Mutex& ODatabaseForm::getMutex()
1376 {
1377 return m_aMutex;
1378 }
1379
1380 //==============================================================================
1381 // property handling
1382 //------------------------------------------------------------------------------
describeFixedAndAggregateProperties(Sequence<Property> & _rProps,Sequence<Property> & _rAggregateProps) const1383 void ODatabaseForm::describeFixedAndAggregateProperties(
1384 Sequence< Property >& _rProps,
1385 Sequence< Property >& _rAggregateProps ) const
1386 {
1387 BEGIN_DESCRIBE_AGGREGATION_PROPERTIES(22, m_xAggregateSet)
1388 // we want to "override" the privileges, since we have additional "AllowInsert" etc. properties
1389 RemoveProperty( _rAggregateProps, PROPERTY_PRIVILEGES );
1390
1391 // InsertOnly is also to be overridden, since we sometimes change it ourself
1392 RemoveProperty( _rAggregateProps, PROPERTY_INSERTONLY );
1393
1394 // we remove and re-declare the DataSourceName property, 'cause we want it to be constrained, and the
1395 // original property of our aggregate isn't
1396 RemoveProperty( _rAggregateProps, PROPERTY_DATASOURCE );
1397
1398 // for connection sharing, we need to override the ActiveConnection property, too
1399 RemoveProperty( _rAggregateProps, PROPERTY_ACTIVE_CONNECTION );
1400
1401 // the Filter property is also overwritten, since we have some implicit filters
1402 // (e.g. the ones which result from linking master fields to detail fields
1403 // via column names instead of parameters)
1404 RemoveProperty( _rAggregateProps, PROPERTY_FILTER );
1405 RemoveProperty( _rAggregateProps, PROPERTY_APPLYFILTER );
1406
1407 DECL_IFACE_PROP4(ACTIVE_CONNECTION, XConnection, BOUND, TRANSIENT, MAYBEVOID, CONSTRAINED);
1408 DECL_BOOL_PROP2 ( APPLYFILTER, BOUND, MAYBEDEFAULT );
1409 DECL_PROP1 ( NAME, ::rtl::OUString, BOUND );
1410 DECL_PROP1 ( MASTERFIELDS, Sequence< ::rtl::OUString >, BOUND );
1411 DECL_PROP1 ( DETAILFIELDS, Sequence< ::rtl::OUString >, BOUND );
1412 DECL_PROP2 ( DATASOURCE, ::rtl::OUString, BOUND, CONSTRAINED );
1413 DECL_PROP3 ( CYCLE, TabulatorCycle, BOUND, MAYBEVOID, MAYBEDEFAULT );
1414 DECL_PROP2 ( FILTER, ::rtl::OUString, BOUND, MAYBEDEFAULT );
1415 DECL_BOOL_PROP2 ( INSERTONLY, BOUND, MAYBEDEFAULT );
1416 DECL_PROP1 ( NAVIGATION, NavigationBarMode, BOUND );
1417 DECL_BOOL_PROP1 ( ALLOWADDITIONS, BOUND );
1418 DECL_BOOL_PROP1 ( ALLOWEDITS, BOUND );
1419 DECL_BOOL_PROP1 ( ALLOWDELETIONS, BOUND );
1420 DECL_PROP2 ( PRIVILEGES, sal_Int32, TRANSIENT, READONLY );
1421 DECL_PROP1 ( TARGET_URL, ::rtl::OUString, BOUND );
1422 DECL_PROP1 ( TARGET_FRAME, ::rtl::OUString, BOUND );
1423 DECL_PROP1 ( SUBMIT_METHOD, FormSubmitMethod, BOUND );
1424 DECL_PROP1 ( SUBMIT_ENCODING, FormSubmitEncoding, BOUND );
1425 DECL_BOOL_PROP3 ( DYNAMIC_CONTROL_BORDER, BOUND, MAYBEVOID, MAYBEDEFAULT );
1426 DECL_PROP3 ( CONTROL_BORDER_COLOR_FOCUS, sal_Int32, BOUND, MAYBEVOID, MAYBEDEFAULT );
1427 DECL_PROP3 ( CONTROL_BORDER_COLOR_MOUSE, sal_Int32, BOUND, MAYBEVOID, MAYBEDEFAULT );
1428 DECL_PROP3 ( CONTROL_BORDER_COLOR_INVALID, sal_Int32, BOUND, MAYBEVOID, MAYBEDEFAULT );
1429 END_DESCRIBE_PROPERTIES();
1430 }
1431
1432 //------------------------------------------------------------------------------
getPropertiesInterface()1433 Reference< XMultiPropertySet > ODatabaseForm::getPropertiesInterface()
1434 {
1435 return Reference< XMultiPropertySet >( *this, UNO_QUERY );
1436 }
1437
1438 //------------------------------------------------------------------------------
getInfoHelper()1439 ::cppu::IPropertyArrayHelper& ODatabaseForm::getInfoHelper()
1440 {
1441 return m_aPropertyBagHelper.getInfoHelper();
1442 }
1443
1444 //------------------------------------------------------------------------------
getPropertySetInfo()1445 Reference< XPropertySetInfo > ODatabaseForm::getPropertySetInfo() throw( RuntimeException )
1446 {
1447 return createPropertySetInfo( getInfoHelper() );
1448 }
1449
1450 //--------------------------------------------------------------------
addProperty(const::rtl::OUString & _rName,::sal_Int16 _nAttributes,const Any & _rInitialValue)1451 void SAL_CALL ODatabaseForm::addProperty( const ::rtl::OUString& _rName, ::sal_Int16 _nAttributes, const Any& _rInitialValue ) throw (PropertyExistException, IllegalTypeException, IllegalArgumentException, RuntimeException)
1452 {
1453 m_aPropertyBagHelper.addProperty( _rName, _nAttributes, _rInitialValue );
1454 }
1455
1456 //--------------------------------------------------------------------
removeProperty(const::rtl::OUString & _rName)1457 void SAL_CALL ODatabaseForm::removeProperty( const ::rtl::OUString& _rName ) throw (UnknownPropertyException, NotRemoveableException, RuntimeException)
1458 {
1459 m_aPropertyBagHelper.removeProperty( _rName );
1460 }
1461
1462 //--------------------------------------------------------------------
getPropertyValues()1463 Sequence< PropertyValue > SAL_CALL ODatabaseForm::getPropertyValues() throw (RuntimeException)
1464 {
1465 return m_aPropertyBagHelper.getPropertyValues();
1466 }
1467
1468 //--------------------------------------------------------------------
setPropertyValues(const Sequence<PropertyValue> & _rProps)1469 void SAL_CALL ODatabaseForm::setPropertyValues( const Sequence< PropertyValue >& _rProps ) throw (UnknownPropertyException, PropertyVetoException, IllegalArgumentException, WrappedTargetException, RuntimeException)
1470 {
1471 m_aPropertyBagHelper.setPropertyValues( _rProps );
1472 }
1473
1474 //------------------------------------------------------------------------------
getWarnings()1475 Any SAL_CALL ODatabaseForm::getWarnings( ) throw (SQLException, RuntimeException)
1476 {
1477 return m_aWarnings.getWarnings();
1478 }
1479
1480 //------------------------------------------------------------------------------
clearWarnings()1481 void SAL_CALL ODatabaseForm::clearWarnings( ) throw (SQLException, RuntimeException)
1482 {
1483 m_aWarnings.clearWarnings();
1484 }
1485
1486 //------------------------------------------------------------------------------
createClone()1487 Reference< XCloneable > SAL_CALL ODatabaseForm::createClone( ) throw (RuntimeException)
1488 {
1489 ODatabaseForm* pClone = new ODatabaseForm( *this );
1490 osl_incrementInterlockedCount( &pClone->m_refCount );
1491 pClone->clonedFrom( *this );
1492 osl_decrementInterlockedCount( &pClone->m_refCount );
1493 return pClone;
1494 }
1495
1496 //------------------------------------------------------------------------------
fire(sal_Int32 * pnHandles,const Any * pNewValues,const Any * pOldValues,sal_Int32 nCount,sal_Bool bVetoable)1497 void ODatabaseForm::fire( sal_Int32* pnHandles, const Any* pNewValues, const Any* pOldValues, sal_Int32 nCount, sal_Bool bVetoable )
1498 {
1499 // same as in getFastPropertyValue(sal_Int32) : if we're resetting currently don't fire any changes of the
1500 // IsModified property from sal_False to sal_True, as this is only temporary 'til the reset is done
1501 if (m_nResetsPending > 0)
1502 {
1503 // look for the PROPERTY_ID_ISMODIFIED
1504 sal_Int32 nPos = 0;
1505 for (nPos=0; nPos<nCount; ++nPos)
1506 if (pnHandles[nPos] == PROPERTY_ID_ISMODIFIED)
1507 break;
1508
1509 if ((nPos < nCount) && (pNewValues[nPos].getValueType().getTypeClass() == TypeClass_BOOLEAN) && getBOOL(pNewValues[nPos]))
1510 { // yeah, we found it, and it changed to TRUE
1511 if (nPos == 0)
1512 { // just cut the first element
1513 ++pnHandles;
1514 ++pNewValues;
1515 ++pOldValues;
1516 --nCount;
1517 }
1518 else if (nPos == nCount - 1)
1519 // just cut the last element
1520 --nCount;
1521 else
1522 { // split into two base class calls
1523 OPropertySetAggregationHelper::fire(pnHandles, pNewValues, pOldValues, nPos, bVetoable);
1524 ++nPos;
1525 OPropertySetAggregationHelper::fire(pnHandles + nPos, pNewValues + nPos, pOldValues + nPos, nCount - nPos, bVetoable);
1526 return;
1527 }
1528 }
1529 }
1530
1531 OPropertySetAggregationHelper::fire(pnHandles, pNewValues, pOldValues, nCount, bVetoable);
1532 }
1533
1534 //------------------------------------------------------------------------------
getFastPropertyValue(sal_Int32 nHandle)1535 Any SAL_CALL ODatabaseForm::getFastPropertyValue( sal_Int32 nHandle )
1536 throw(UnknownPropertyException, WrappedTargetException, RuntimeException)
1537 {
1538 if ((nHandle == PROPERTY_ID_ISMODIFIED) && (m_nResetsPending > 0))
1539 return ::cppu::bool2any((sal_False));
1540 // don't allow the aggregate which is currently being reset to return a (temporary) "yes"
1541 else
1542 return OPropertySetAggregationHelper::getFastPropertyValue(nHandle);
1543 }
1544
1545 //------------------------------------------------------------------------------
getFastPropertyValue(Any & rValue,sal_Int32 nHandle) const1546 void ODatabaseForm::getFastPropertyValue( Any& rValue, sal_Int32 nHandle ) const
1547 {
1548 switch (nHandle)
1549 {
1550 case PROPERTY_ID_INSERTONLY:
1551 rValue <<= m_bInsertOnly;
1552 break;
1553
1554 case PROPERTY_ID_FILTER:
1555 rValue <<= m_aFilterManager.getFilterComponent( FilterManager::fcPublicFilter );
1556 break;
1557
1558 case PROPERTY_ID_APPLYFILTER:
1559 rValue <<= m_aFilterManager.isApplyPublicFilter();
1560 break;
1561
1562 case PROPERTY_ID_DATASOURCE:
1563 rValue = m_xAggregateSet->getPropertyValue( PROPERTY_DATASOURCE );
1564 break;
1565
1566 case PROPERTY_ID_TARGET_URL:
1567 rValue <<= m_aTargetURL;
1568 break;
1569 case PROPERTY_ID_TARGET_FRAME:
1570 rValue <<= m_aTargetFrame;
1571 break;
1572 case PROPERTY_ID_SUBMIT_METHOD:
1573 rValue <<= m_eSubmitMethod;
1574 break;
1575 case PROPERTY_ID_SUBMIT_ENCODING:
1576 rValue <<= m_eSubmitEncoding;
1577 break;
1578 case PROPERTY_ID_NAME:
1579 rValue <<= m_sName;
1580 break;
1581 case PROPERTY_ID_MASTERFIELDS:
1582 rValue <<= m_aMasterFields;
1583 break;
1584 case PROPERTY_ID_DETAILFIELDS:
1585 rValue <<= m_aDetailFields;
1586 break;
1587 case PROPERTY_ID_CYCLE:
1588 rValue = m_aCycle;
1589 break;
1590 case PROPERTY_ID_NAVIGATION:
1591 rValue <<= m_eNavigation;
1592 break;
1593 case PROPERTY_ID_ALLOWADDITIONS:
1594 rValue <<= (sal_Bool)m_bAllowInsert;
1595 break;
1596 case PROPERTY_ID_ALLOWEDITS:
1597 rValue <<= (sal_Bool)m_bAllowUpdate;
1598 break;
1599 case PROPERTY_ID_ALLOWDELETIONS:
1600 rValue <<= (sal_Bool)m_bAllowDelete;
1601 break;
1602 case PROPERTY_ID_PRIVILEGES:
1603 rValue <<= (sal_Int32)m_nPrivileges;
1604 break;
1605 case PROPERTY_ID_DYNAMIC_CONTROL_BORDER:
1606 rValue = m_aDynamicControlBorder;
1607 break;
1608 case PROPERTY_ID_CONTROL_BORDER_COLOR_FOCUS:
1609 rValue = m_aControlBorderColorFocus;
1610 break;
1611 case PROPERTY_ID_CONTROL_BORDER_COLOR_MOUSE:
1612 rValue = m_aControlBorderColorMouse;
1613 break;
1614 case PROPERTY_ID_CONTROL_BORDER_COLOR_INVALID:
1615 rValue = m_aControlBorderColorInvalid;
1616 break;
1617 default:
1618 if ( m_aPropertyBagHelper.hasDynamicPropertyByHandle( nHandle ) )
1619 m_aPropertyBagHelper.getDynamicFastPropertyValue( nHandle, rValue );
1620 else
1621 OPropertySetAggregationHelper::getFastPropertyValue( rValue, nHandle );
1622 break;
1623 }
1624 }
1625
1626 //------------------------------------------------------------------------------
convertFastPropertyValue(Any & rConvertedValue,Any & rOldValue,sal_Int32 nHandle,const Any & rValue)1627 sal_Bool ODatabaseForm::convertFastPropertyValue( Any& rConvertedValue, Any& rOldValue,
1628 sal_Int32 nHandle, const Any& rValue ) throw( IllegalArgumentException )
1629 {
1630 sal_Bool bModified(sal_False);
1631 switch (nHandle)
1632 {
1633 case PROPERTY_ID_INSERTONLY:
1634 bModified = tryPropertyValue( rConvertedValue, rOldValue, rValue, m_bInsertOnly );
1635 break;
1636
1637 case PROPERTY_ID_FILTER:
1638 bModified = tryPropertyValue( rConvertedValue, rOldValue, rValue, m_aFilterManager.getFilterComponent( FilterManager::fcPublicFilter ) );
1639 break;
1640
1641 case PROPERTY_ID_APPLYFILTER:
1642 bModified = tryPropertyValue( rConvertedValue, rOldValue, rValue, m_aFilterManager.isApplyPublicFilter() );
1643 break;
1644
1645 case PROPERTY_ID_DATASOURCE:
1646 {
1647 Any aAggregateProperty;
1648 getFastPropertyValue(aAggregateProperty, PROPERTY_ID_DATASOURCE);
1649 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, aAggregateProperty, ::getCppuType(static_cast<const ::rtl::OUString*>(NULL)));
1650 }
1651 break;
1652 case PROPERTY_ID_TARGET_URL:
1653 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_aTargetURL);
1654 break;
1655 case PROPERTY_ID_TARGET_FRAME:
1656 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_aTargetFrame);
1657 break;
1658 case PROPERTY_ID_SUBMIT_METHOD:
1659 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_eSubmitMethod);
1660 break;
1661 case PROPERTY_ID_SUBMIT_ENCODING:
1662 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_eSubmitEncoding);
1663 break;
1664 case PROPERTY_ID_NAME:
1665 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_sName);
1666 break;
1667 case PROPERTY_ID_MASTERFIELDS:
1668 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_aMasterFields);
1669 break;
1670 case PROPERTY_ID_DETAILFIELDS:
1671 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_aDetailFields);
1672 break;
1673 case PROPERTY_ID_CYCLE:
1674 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_aCycle, ::getCppuType(static_cast<const TabulatorCycle*>(NULL)));
1675 break;
1676 case PROPERTY_ID_NAVIGATION:
1677 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_eNavigation);
1678 break;
1679 case PROPERTY_ID_ALLOWADDITIONS:
1680 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_bAllowInsert);
1681 break;
1682 case PROPERTY_ID_ALLOWEDITS:
1683 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_bAllowUpdate);
1684 break;
1685 case PROPERTY_ID_ALLOWDELETIONS:
1686 bModified = tryPropertyValue(rConvertedValue, rOldValue, rValue, m_bAllowDelete);
1687 break;
1688 case PROPERTY_ID_DYNAMIC_CONTROL_BORDER:
1689 bModified = tryPropertyValue( rConvertedValue, rOldValue, rValue, m_aDynamicControlBorder, ::getBooleanCppuType() );
1690 break;
1691 case PROPERTY_ID_CONTROL_BORDER_COLOR_FOCUS:
1692 bModified = tryPropertyValue( rConvertedValue, rOldValue, rValue, m_aControlBorderColorFocus, getCppuType( static_cast< sal_Int32* >( NULL ) ) );
1693 break;
1694 case PROPERTY_ID_CONTROL_BORDER_COLOR_MOUSE:
1695 bModified = tryPropertyValue( rConvertedValue, rOldValue, rValue, m_aControlBorderColorMouse, getCppuType( static_cast< sal_Int32* >( NULL ) ) );
1696 break;
1697 case PROPERTY_ID_CONTROL_BORDER_COLOR_INVALID:
1698 bModified = tryPropertyValue( rConvertedValue, rOldValue, rValue, m_aControlBorderColorInvalid, getCppuType( static_cast< sal_Int32* >( NULL ) ) );
1699 break;
1700 default:
1701 if ( m_aPropertyBagHelper.hasDynamicPropertyByHandle ( nHandle ) )
1702 bModified = m_aPropertyBagHelper.convertDynamicFastPropertyValue( nHandle, rValue, rConvertedValue, rOldValue );
1703 else
1704 bModified = OPropertySetAggregationHelper::convertFastPropertyValue( rConvertedValue, rOldValue, nHandle, rValue );
1705 break;
1706 }
1707 return bModified;
1708 }
1709
1710 //------------------------------------------------------------------------------
setFastPropertyValue_NoBroadcast(sal_Int32 nHandle,const Any & rValue)1711 void ODatabaseForm::setFastPropertyValue_NoBroadcast( sal_Int32 nHandle, const Any& rValue ) throw( Exception )
1712 {
1713 switch (nHandle)
1714 {
1715 case PROPERTY_ID_INSERTONLY:
1716 rValue >>= m_bInsertOnly;
1717 if ( m_aIgnoreResult.hasValue() )
1718 m_aIgnoreResult <<= m_bInsertOnly;
1719 else
1720 m_xAggregateSet->setPropertyValue( PROPERTY_INSERTONLY, makeAny( m_bInsertOnly ) );
1721 break;
1722
1723 case PROPERTY_ID_FILTER:
1724 {
1725 ::rtl::OUString sNewFilter;
1726 rValue >>= sNewFilter;
1727 m_aFilterManager.setFilterComponent( FilterManager::fcPublicFilter, sNewFilter );
1728 }
1729 break;
1730
1731 case PROPERTY_ID_APPLYFILTER:
1732 {
1733 sal_Bool bApply = sal_True;
1734 rValue >>= bApply;
1735 m_aFilterManager.setApplyPublicFilter( bApply );
1736 }
1737 break;
1738
1739 case PROPERTY_ID_DATASOURCE:
1740 {
1741 Reference< XConnection > xSomeConnection;
1742 if ( ::dbtools::isEmbeddedInDatabase( getParent(), xSomeConnection ) )
1743 throw PropertyVetoException();
1744
1745 try
1746 {
1747 m_xAggregateSet->setPropertyValue(PROPERTY_DATASOURCE, rValue);
1748 }
1749 catch(Exception&) { }
1750 }
1751 break;
1752 case PROPERTY_ID_TARGET_URL:
1753 rValue >>= m_aTargetURL;
1754 break;
1755 case PROPERTY_ID_TARGET_FRAME:
1756 rValue >>= m_aTargetFrame;
1757 break;
1758 case PROPERTY_ID_SUBMIT_METHOD:
1759 rValue >>= m_eSubmitMethod;
1760 break;
1761 case PROPERTY_ID_SUBMIT_ENCODING:
1762 rValue >>= m_eSubmitEncoding;
1763 break;
1764 case PROPERTY_ID_NAME:
1765 rValue >>= m_sName;
1766 break;
1767 case PROPERTY_ID_MASTERFIELDS:
1768 rValue >>= m_aMasterFields;
1769 invlidateParameters();
1770 break;
1771 case PROPERTY_ID_DETAILFIELDS:
1772 rValue >>= m_aDetailFields;
1773 invlidateParameters();
1774 break;
1775 case PROPERTY_ID_CYCLE:
1776 m_aCycle = rValue;
1777 break;
1778 case PROPERTY_ID_NAVIGATION:
1779 rValue >>= m_eNavigation;
1780 break;
1781 case PROPERTY_ID_ALLOWADDITIONS:
1782 m_bAllowInsert = getBOOL(rValue);
1783 break;
1784 case PROPERTY_ID_ALLOWEDITS:
1785 m_bAllowUpdate = getBOOL(rValue);
1786 break;
1787 case PROPERTY_ID_ALLOWDELETIONS:
1788 m_bAllowDelete = getBOOL(rValue);
1789 break;
1790 case PROPERTY_ID_DYNAMIC_CONTROL_BORDER:
1791 m_aDynamicControlBorder = rValue;
1792 break;
1793 case PROPERTY_ID_CONTROL_BORDER_COLOR_FOCUS:
1794 m_aControlBorderColorFocus = rValue;
1795 break;
1796 case PROPERTY_ID_CONTROL_BORDER_COLOR_MOUSE:
1797 m_aControlBorderColorMouse = rValue;
1798 break;
1799 case PROPERTY_ID_CONTROL_BORDER_COLOR_INVALID:
1800 m_aControlBorderColorInvalid = rValue;
1801 break;
1802
1803 case PROPERTY_ID_ACTIVE_CONNECTION:
1804 {
1805 Reference< XConnection > xOuterConnection;
1806 if ( ::dbtools::isEmbeddedInDatabase( getParent(), xOuterConnection ) )
1807 {
1808 if ( xOuterConnection != Reference< XConnection >( rValue, UNO_QUERY ) )
1809 // somebody's trying to set a connection which is not equal the connection
1810 // implied by the database we're embedded in
1811 throw PropertyVetoException();
1812 }
1813 OPropertySetAggregationHelper::setFastPropertyValue_NoBroadcast( nHandle, rValue );
1814 break;
1815 }
1816
1817 default:
1818 if ( m_aPropertyBagHelper.hasDynamicPropertyByHandle( nHandle ) )
1819 m_aPropertyBagHelper.setDynamicFastPropertyValue( nHandle, rValue );
1820 else
1821 OPropertySetAggregationHelper::setFastPropertyValue_NoBroadcast( nHandle, rValue );
1822 break;
1823 }
1824 }
1825
1826 //------------------------------------------------------------------
forwardingPropertyValue(sal_Int32 _nHandle)1827 void SAL_CALL ODatabaseForm::forwardingPropertyValue( sal_Int32 _nHandle )
1828 {
1829 OSL_ENSURE( _nHandle == PROPERTY_ID_ACTIVE_CONNECTION, "ODatabaseForm::forwardingPropertyValue: unexpected property!" );
1830 if ( _nHandle == PROPERTY_ID_ACTIVE_CONNECTION )
1831 {
1832 if ( m_bSharingConnection )
1833 stopSharingConnection( );
1834 m_bForwardingConnection = sal_True;
1835 }
1836 }
1837
1838 //------------------------------------------------------------------
forwardedPropertyValue(sal_Int32 _nHandle,bool)1839 void SAL_CALL ODatabaseForm::forwardedPropertyValue( sal_Int32 _nHandle, bool /*_bSuccess*/ )
1840 {
1841 OSL_ENSURE( _nHandle == PROPERTY_ID_ACTIVE_CONNECTION, "ODatabaseForm::forwardedPropertyValue: unexpected property!" );
1842 if ( _nHandle == PROPERTY_ID_ACTIVE_CONNECTION )
1843 {
1844 m_bForwardingConnection = sal_False;
1845 }
1846 }
1847
1848 //==============================================================================
1849 // com::sun::star::beans::XPropertyState
1850 //------------------------------------------------------------------
getPropertyStateByHandle(sal_Int32 nHandle)1851 PropertyState ODatabaseForm::getPropertyStateByHandle(sal_Int32 nHandle)
1852 {
1853 PropertyState eState;
1854 switch (nHandle)
1855 {
1856 case PROPERTY_ID_NAVIGATION:
1857 return (NavigationBarMode_CURRENT == m_eNavigation) ? PropertyState_DEFAULT_VALUE : PropertyState_DIRECT_VALUE;
1858
1859 case PROPERTY_ID_CYCLE:
1860 eState = m_aCycle.hasValue() ? PropertyState_DIRECT_VALUE : PropertyState_DEFAULT_VALUE;
1861 break;
1862
1863 case PROPERTY_ID_INSERTONLY:
1864 eState = m_bInsertOnly ? PropertyState_DIRECT_VALUE : PropertyState_DEFAULT_VALUE;
1865 break;
1866
1867 case PROPERTY_ID_FILTER:
1868 if ( !m_aFilterManager.getFilterComponent( FilterManager::fcPublicFilter ).getLength() )
1869 eState = PropertyState_DEFAULT_VALUE;
1870 else
1871 eState = PropertyState_DIRECT_VALUE;
1872 break;
1873
1874 case PROPERTY_ID_APPLYFILTER:
1875 eState = m_aFilterManager.isApplyPublicFilter() ? PropertyState_DEFAULT_VALUE : PropertyState_DIRECT_VALUE;
1876 break;
1877
1878 case PROPERTY_ID_DYNAMIC_CONTROL_BORDER:
1879 eState = m_aDynamicControlBorder.hasValue() ? PropertyState_DIRECT_VALUE : PropertyState_DEFAULT_VALUE;
1880 break;
1881
1882 case PROPERTY_ID_CONTROL_BORDER_COLOR_FOCUS:
1883 eState = m_aControlBorderColorFocus.hasValue() ? PropertyState_DIRECT_VALUE : PropertyState_DEFAULT_VALUE;
1884 break;
1885
1886 case PROPERTY_ID_CONTROL_BORDER_COLOR_MOUSE:
1887 eState = m_aControlBorderColorMouse.hasValue() ? PropertyState_DIRECT_VALUE : PropertyState_DEFAULT_VALUE;
1888 break;
1889
1890 case PROPERTY_ID_CONTROL_BORDER_COLOR_INVALID:
1891 eState = m_aControlBorderColorInvalid.hasValue() ? PropertyState_DIRECT_VALUE : PropertyState_DEFAULT_VALUE;
1892 break;
1893
1894 default:
1895 eState = OPropertySetAggregationHelper::getPropertyStateByHandle(nHandle);
1896 }
1897 return eState;
1898 }
1899
1900 //------------------------------------------------------------------
setPropertyToDefaultByHandle(sal_Int32 nHandle)1901 void ODatabaseForm::setPropertyToDefaultByHandle(sal_Int32 nHandle)
1902 {
1903 switch (nHandle)
1904 {
1905 case PROPERTY_ID_INSERTONLY:
1906 case PROPERTY_ID_FILTER:
1907 case PROPERTY_ID_APPLYFILTER:
1908 case PROPERTY_ID_NAVIGATION:
1909 case PROPERTY_ID_CYCLE:
1910 case PROPERTY_ID_DYNAMIC_CONTROL_BORDER:
1911 case PROPERTY_ID_CONTROL_BORDER_COLOR_FOCUS:
1912 case PROPERTY_ID_CONTROL_BORDER_COLOR_MOUSE:
1913 case PROPERTY_ID_CONTROL_BORDER_COLOR_INVALID:
1914 setFastPropertyValue( nHandle, getPropertyDefaultByHandle( nHandle ) );
1915 break;
1916
1917 default:
1918 OPropertySetAggregationHelper::setPropertyToDefaultByHandle(nHandle);
1919 }
1920 }
1921
1922 //------------------------------------------------------------------
getPropertyDefaultByHandle(sal_Int32 nHandle) const1923 Any ODatabaseForm::getPropertyDefaultByHandle( sal_Int32 nHandle ) const
1924 {
1925 Any aReturn;
1926 switch (nHandle)
1927 {
1928 case PROPERTY_ID_INSERTONLY:
1929 case PROPERTY_ID_DYNAMIC_CONTROL_BORDER:
1930 aReturn <<= sal_False;
1931 break;
1932
1933 case PROPERTY_ID_FILTER:
1934 aReturn <<= ::rtl::OUString();
1935 break;
1936
1937 case PROPERTY_ID_APPLYFILTER:
1938 aReturn <<= sal_True;
1939 break;
1940
1941 case PROPERTY_ID_NAVIGATION:
1942 aReturn = makeAny(NavigationBarMode_CURRENT);
1943 break;
1944
1945 case PROPERTY_ID_CYCLE:
1946 case PROPERTY_ID_CONTROL_BORDER_COLOR_FOCUS:
1947 case PROPERTY_ID_CONTROL_BORDER_COLOR_MOUSE:
1948 case PROPERTY_ID_CONTROL_BORDER_COLOR_INVALID:
1949 break;
1950
1951 default:
1952 if ( m_aPropertyBagHelper.hasDynamicPropertyByHandle( nHandle ) )
1953 m_aPropertyBagHelper.getDynamicPropertyDefaultByHandle( nHandle, aReturn );
1954 else
1955 aReturn = OPropertySetAggregationHelper::getPropertyDefaultByHandle( nHandle );
1956 break;
1957 }
1958 return aReturn;
1959 }
1960
1961 //==============================================================================
1962 // com::sun::star::form::XReset
1963 //------------------------------------------------------------------------------
reset()1964 void SAL_CALL ODatabaseForm::reset() throw( RuntimeException )
1965 {
1966 ::osl::ResettableMutexGuard aGuard(m_aMutex);
1967
1968 if (isLoaded())
1969 {
1970 ::osl::MutexGuard aResetGuard(m_aResetSafety);
1971 ++m_nResetsPending;
1972 reset_impl(true);
1973 return;
1974 }
1975
1976 if ( !m_aResetListeners.empty() )
1977 {
1978 ::osl::MutexGuard aResetGuard(m_aResetSafety);
1979 ++m_nResetsPending;
1980 // create an own thread if we have (approve-)reset-listeners (so the listeners can't do that much damage
1981 // to this thread which is probably the main one)
1982 if (!m_pThread)
1983 {
1984 m_pThread = new OFormSubmitResetThread(this);
1985 m_pThread->acquire();
1986 m_pThread->create();
1987 }
1988 EventObject aEvt;
1989 m_pThread->addEvent(&aEvt, sal_False);
1990 }
1991 else
1992 {
1993 // direct call without any approving by the listeners
1994 aGuard.clear();
1995
1996 ::osl::MutexGuard aResetGuard(m_aResetSafety);
1997 ++m_nResetsPending;
1998 reset_impl(false);
1999 }
2000 }
2001
2002 //-----------------------------------------------------------------------------
reset_impl(bool _bAproveByListeners)2003 void ODatabaseForm::reset_impl(bool _bAproveByListeners)
2004 {
2005 if ( _bAproveByListeners )
2006 if ( !m_aResetListeners.approveReset() )
2007 return;
2008
2009 ::osl::ResettableMutexGuard aResetGuard(m_aResetSafety);
2010 // do we have a database connected form and stay on the insert row
2011 sal_Bool bInsertRow = sal_False;
2012 if (m_xAggregateSet.is())
2013 bInsertRow = getBOOL(m_xAggregateSet->getPropertyValue(PROPERTY_ISNEW));
2014 if (bInsertRow)
2015 {
2016 try
2017 {
2018 // Iterate through all columns and set the default value
2019 Reference< XColumnsSupplier > xColsSuppl( m_xAggregateSet, UNO_QUERY );
2020 Reference< XIndexAccess > xIndexCols( xColsSuppl->getColumns(), UNO_QUERY );
2021 for (sal_Int32 i = 0; i < xIndexCols->getCount(); ++i)
2022 {
2023 Reference< XPropertySet > xColProps;
2024 xIndexCols->getByIndex(i) >>= xColProps;
2025
2026 Reference< XColumnUpdate > xColUpdate( xColProps, UNO_QUERY );
2027 if ( !xColUpdate.is() )
2028 continue;
2029
2030 Reference< XPropertySetInfo > xPSI;
2031 if ( xColProps.is() )
2032 xPSI = xColProps->getPropertySetInfo( );
2033
2034 static const ::rtl::OUString PROPERTY_CONTROLDEFAULT( RTL_CONSTASCII_USTRINGPARAM( "ControlDefault" ) );
2035 if ( xPSI.is() && xPSI->hasPropertyByName( PROPERTY_CONTROLDEFAULT ) )
2036 {
2037 Any aDefault = xColProps->getPropertyValue( PROPERTY_CONTROLDEFAULT );
2038
2039 sal_Bool bReadOnly = sal_False;
2040 if ( xPSI->hasPropertyByName( PROPERTY_ISREADONLY ) )
2041 xColProps->getPropertyValue( PROPERTY_ISREADONLY ) >>= bReadOnly;
2042
2043 if ( !bReadOnly )
2044 {
2045 try
2046 {
2047 if ( aDefault.hasValue() )
2048 xColUpdate->updateObject( aDefault );
2049 }
2050 catch(Exception&)
2051 {
2052 DBG_UNHANDLED_EXCEPTION();
2053 }
2054 }
2055 }
2056 }
2057 }
2058 catch(Exception&)
2059 {
2060 }
2061
2062 if (m_bSubForm)
2063 {
2064 Reference< XColumnsSupplier > xParentColSupp( m_xParent, UNO_QUERY );
2065 Reference< XNameAccess > xParentCols;
2066 if ( xParentColSupp.is() )
2067 xParentCols = xParentColSupp->getColumns();
2068
2069 if ( xParentCols.is() && xParentCols->hasElements() && m_aMasterFields.getLength() )
2070 {
2071 try
2072 {
2073 // analyze our parameters
2074 if ( !m_aParameterManager.isUpToDate() )
2075 updateParameterInfo();
2076
2077 m_aParameterManager.resetParameterValues( );
2078 }
2079 catch(const Exception&)
2080 {
2081 OSL_ENSURE(sal_False, "ODatabaseForm::reset_impl: could not initialize the master-detail-driven parameters!");
2082 }
2083 }
2084 }
2085 }
2086
2087 aResetGuard.clear();
2088 // iterate through all components. don't use an XIndexAccess as this will cause massive
2089 // problems with the count.
2090 Reference<XEnumeration> xIter = createEnumeration();
2091 while (xIter->hasMoreElements())
2092 {
2093 Reference<XReset> xReset;
2094 xIter->nextElement() >>= xReset;
2095 if (xReset.is())
2096 {
2097 // TODO : all reset-methods have to be thread-safe
2098 xReset->reset();
2099 }
2100 }
2101
2102 aResetGuard.reset();
2103 // ensure that the row isn't modified
2104 // (do this _before_ the listeners are notified ! their reaction (maybe asynchronous) may depend
2105 // on the modified state of the row
2106 // 21.02.00 - 73265 - FS)
2107 if (bInsertRow)
2108 m_xAggregateSet->setPropertyValue(PROPERTY_ISMODIFIED, ::cppu::bool2any(sal_Bool(sal_False)));
2109
2110 aResetGuard.clear();
2111 {
2112 m_aResetListeners.resetted();
2113 }
2114
2115 aResetGuard.reset();
2116 // and again : ensure the row isn't modified
2117 // we already did this after we (and maybe our dependents) resetted the values, but the listeners may have changed the row, too
2118 if (bInsertRow)
2119 m_xAggregateSet->setPropertyValue(PROPERTY_ISMODIFIED, ::cppu::bool2any((sal_False)));
2120
2121 --m_nResetsPending;
2122 }
2123
2124 //-----------------------------------------------------------------------------
addResetListener(const Reference<XResetListener> & _rListener)2125 void SAL_CALL ODatabaseForm::addResetListener(const Reference<XResetListener>& _rListener) throw( RuntimeException )
2126 {
2127 m_aResetListeners.addTypedListener( _rListener );
2128 }
2129
2130 //-----------------------------------------------------------------------------
removeResetListener(const Reference<XResetListener> & _rListener)2131 void SAL_CALL ODatabaseForm::removeResetListener(const Reference<XResetListener>& _rListener) throw( RuntimeException )
2132 {
2133 m_aResetListeners.removeTypedListener( _rListener );
2134 }
2135
2136 //==============================================================================
2137 // com::sun::star::form::XSubmit
2138 //------------------------------------------------------------------------------
submit(const Reference<XControl> & Control,const::com::sun::star::awt::MouseEvent & MouseEvt)2139 void SAL_CALL ODatabaseForm::submit( const Reference<XControl>& Control,
2140 const ::com::sun::star::awt::MouseEvent& MouseEvt ) throw( RuntimeException )
2141 {
2142 {
2143 ::osl::MutexGuard aGuard(m_aMutex);
2144 // Sind Controls und eine Submit-URL vorhanden?
2145 if( !getCount() || !m_aTargetURL.getLength() )
2146 return;
2147 }
2148
2149 ::osl::ClearableMutexGuard aGuard(m_aMutex);
2150 if (m_aSubmitListeners.getLength())
2151 {
2152 // create an own thread if we have (approve-)submit-listeners (so the listeners can't do that much damage
2153 // to this thread which is probably the main one)
2154 if (!m_pThread)
2155 {
2156 m_pThread = new OFormSubmitResetThread(this);
2157 m_pThread->acquire();
2158 m_pThread->create();
2159 }
2160 m_pThread->addEvent(&MouseEvt, Control, sal_True);
2161 }
2162 else
2163 {
2164 // direct call without any approving by the listeners
2165 aGuard.clear();
2166 submit_impl( Control, MouseEvt, true );
2167 }
2168 }
2169 // -----------------------------------------------------------------------------
lcl_dispatch(const Reference<XFrame> & xFrame,const Reference<XURLTransformer> & xTransformer,const::rtl::OUString & aURLStr,const::rtl::OUString & aReferer,const::rtl::OUString & aTargetName,const::rtl::OUString & aData,rtl_TextEncoding _eEncoding)2170 void lcl_dispatch(const Reference< XFrame >& xFrame,const Reference<XURLTransformer>& xTransformer,const ::rtl::OUString& aURLStr,const ::rtl::OUString& aReferer,const ::rtl::OUString& aTargetName
2171 ,const ::rtl::OUString& aData,rtl_TextEncoding _eEncoding)
2172 {
2173 URL aURL;
2174 aURL.Complete = aURLStr;
2175 xTransformer->parseStrict(aURL);
2176
2177 Reference< XDispatch > xDisp = Reference< XDispatchProvider > (xFrame,UNO_QUERY)->queryDispatch(aURL, aTargetName,
2178 FrameSearchFlag::SELF | FrameSearchFlag::PARENT | FrameSearchFlag::CHILDREN |
2179 FrameSearchFlag::SIBLINGS | FrameSearchFlag::CREATE | FrameSearchFlag::TASKS);
2180
2181 if (xDisp.is())
2182 {
2183 Sequence<PropertyValue> aArgs(2);
2184 aArgs.getArray()[0].Name = ::rtl::OUString::createFromAscii("Referer");
2185 aArgs.getArray()[0].Value <<= aReferer;
2186
2187 // build a sequence from the to-be-submitted string
2188 ByteString a8BitData(aData.getStr(), (sal_uInt16)aData.getLength(), _eEncoding);
2189 // always ANSI #58641
2190 Sequence< sal_Int8 > aPostData((sal_Int8*)a8BitData.GetBuffer(), a8BitData.Len());
2191 Reference< XInputStream > xPostData = new SequenceInputStream(aPostData);
2192
2193 aArgs.getArray()[1].Name = ::rtl::OUString::createFromAscii("PostData");
2194 aArgs.getArray()[1].Value <<= xPostData;
2195
2196 xDisp->dispatch(aURL, aArgs);
2197 } // if (xDisp.is())
2198 }
2199 //------------------------------------------------------------------------------
submit_impl(const Reference<XControl> & Control,const::com::sun::star::awt::MouseEvent & MouseEvt,bool _bAproveByListeners)2200 void ODatabaseForm::submit_impl(const Reference<XControl>& Control, const ::com::sun::star::awt::MouseEvent& MouseEvt, bool _bAproveByListeners)
2201 {
2202
2203 if (_bAproveByListeners)
2204 {
2205 ::cppu::OInterfaceIteratorHelper aIter(m_aSubmitListeners);
2206 EventObject aEvt(static_cast<XWeak*>(this));
2207 sal_Bool bCanceled = sal_False;
2208 while (aIter.hasMoreElements() && !bCanceled)
2209 {
2210 if (!((XSubmitListener*)aIter.next())->approveSubmit(aEvt))
2211 bCanceled = sal_True;
2212 }
2213
2214 if (bCanceled)
2215 return;
2216 }
2217
2218 FormSubmitEncoding eSubmitEncoding;
2219 FormSubmitMethod eSubmitMethod;
2220 ::rtl::OUString aURLStr;
2221 ::rtl::OUString aReferer;
2222 ::rtl::OUString aTargetName;
2223 Reference< XModel > xModel;
2224 {
2225 ::vos::OGuard aGuard( Application::GetSolarMutex() );
2226 // starform->Forms
2227
2228 Reference<XChild> xParent(m_xParent, UNO_QUERY);
2229
2230 if (xParent.is())
2231 xModel = getXModel(xParent->getParent());
2232
2233 if (xModel.is())
2234 aReferer = xModel->getURL();
2235
2236 // TargetItem
2237 aTargetName = m_aTargetFrame;
2238
2239 eSubmitEncoding = m_eSubmitEncoding;
2240 eSubmitMethod = m_eSubmitMethod;
2241 aURLStr = m_aTargetURL;
2242 }
2243
2244 if (!xModel.is())
2245 return;
2246 Reference< XFrame > xFrame = xModel->getCurrentController()->getFrame();
2247 if (!xFrame.is())
2248 return;
2249
2250 Reference<XURLTransformer>
2251 xTransformer(m_xServiceFactory->createInstance(
2252 ::rtl::OUString::createFromAscii("com.sun.star.util.URLTransformer")), UNO_QUERY);
2253 DBG_ASSERT(xTransformer.is(), "ODatabaseForm::submit_impl : could not create an URL transformer !");
2254
2255 // URL-Encoding
2256 if( eSubmitEncoding == FormSubmitEncoding_URL )
2257 {
2258 ::rtl::OUString aData;
2259 {
2260 ::vos::OGuard aGuard( Application::GetSolarMutex() );
2261 aData = GetDataURLEncoded( Control, MouseEvt );
2262 }
2263
2264 URL aURL;
2265 // FormMethod GET
2266 if( eSubmitMethod == FormSubmitMethod_GET )
2267 {
2268 INetURLObject aUrlObj( aURLStr, INetURLObject::WAS_ENCODED );
2269 aUrlObj.SetParam( aData, INetURLObject::ENCODE_ALL );
2270 aURL.Complete = aUrlObj.GetMainURL( INetURLObject::DECODE_UNAMBIGUOUS );
2271 if (xTransformer.is())
2272 xTransformer->parseStrict(aURL);
2273
2274 Reference< XDispatch > xDisp = Reference< XDispatchProvider > (xFrame,UNO_QUERY)->queryDispatch(aURL, aTargetName,
2275 FrameSearchFlag::SELF | FrameSearchFlag::PARENT | FrameSearchFlag::CHILDREN |
2276 FrameSearchFlag::SIBLINGS | FrameSearchFlag::CREATE | FrameSearchFlag::TASKS);
2277
2278 if (xDisp.is())
2279 {
2280 Sequence<PropertyValue> aArgs(1);
2281 aArgs.getArray()->Name = ::rtl::OUString::createFromAscii("Referer");
2282 aArgs.getArray()->Value <<= aReferer;
2283 xDisp->dispatch(aURL, aArgs);
2284 }
2285 }
2286 // FormMethod POST
2287 else if( eSubmitMethod == FormSubmitMethod_POST )
2288 {
2289 lcl_dispatch(xFrame,xTransformer,aURLStr,aReferer,aTargetName,aData,RTL_TEXTENCODING_MS_1252);
2290 }
2291 }
2292 else if( eSubmitEncoding == FormSubmitEncoding_MULTIPART )
2293 {
2294 URL aURL;
2295 aURL.Complete = aURLStr;
2296 xTransformer->parseStrict(aURL);
2297
2298 Reference< XDispatch > xDisp = Reference< XDispatchProvider > (xFrame,UNO_QUERY)->queryDispatch(aURL, aTargetName,
2299 FrameSearchFlag::SELF | FrameSearchFlag::PARENT | FrameSearchFlag::CHILDREN |
2300 FrameSearchFlag::SIBLINGS | FrameSearchFlag::CREATE | FrameSearchFlag::TASKS);
2301
2302 if (xDisp.is())
2303 {
2304 ::rtl::OUString aContentType;
2305 Sequence<sal_Int8> aData;
2306 {
2307 ::vos::OGuard aGuard( Application::GetSolarMutex() );
2308 aData = GetDataMultiPartEncoded(Control, MouseEvt, aContentType);
2309 }
2310 if (!aData.getLength())
2311 return;
2312
2313 Sequence<PropertyValue> aArgs(3);
2314 aArgs.getArray()[0].Name = ::rtl::OUString::createFromAscii("Referer");
2315 aArgs.getArray()[0].Value <<= aReferer;
2316 aArgs.getArray()[1].Name = ::rtl::OUString::createFromAscii("ContentType");
2317 aArgs.getArray()[1].Value <<= aContentType;
2318
2319 // build a sequence from the to-be-submitted string
2320 Reference< XInputStream > xPostData = new SequenceInputStream(aData);
2321
2322 aArgs.getArray()[2].Name = ::rtl::OUString::createFromAscii("PostData");
2323 aArgs.getArray()[2].Value <<= xPostData;
2324
2325 xDisp->dispatch(aURL, aArgs);
2326 }
2327 }
2328 else if( eSubmitEncoding == FormSubmitEncoding_TEXT )
2329 {
2330 ::rtl::OUString aData;
2331 {
2332 ::vos::OGuard aGuard( Application::GetSolarMutex() );
2333 aData = GetDataTextEncoded( Reference<XControl> (), MouseEvt );
2334 }
2335
2336 lcl_dispatch(xFrame,xTransformer,aURLStr,aReferer,aTargetName,aData,osl_getThreadTextEncoding());
2337 }
2338 else {
2339 DBG_ERROR("ODatabaseForm::submit_Impl : wrong encoding !");
2340 }
2341
2342 }
2343
2344 // XSubmit
2345 //------------------------------------------------------------------------------
addSubmitListener(const Reference<XSubmitListener> & _rListener)2346 void SAL_CALL ODatabaseForm::addSubmitListener(const Reference<XSubmitListener>& _rListener) throw( RuntimeException )
2347 {
2348 m_aSubmitListeners.addInterface(_rListener);
2349 }
2350
2351 //------------------------------------------------------------------------------
removeSubmitListener(const Reference<XSubmitListener> & _rListener)2352 void SAL_CALL ODatabaseForm::removeSubmitListener(const Reference<XSubmitListener>& _rListener) throw( RuntimeException )
2353 {
2354 m_aSubmitListeners.removeInterface(_rListener);
2355 }
2356
2357 //==============================================================================
2358 // com::sun::star::sdbc::XSQLErrorBroadcaster
2359 //------------------------------------------------------------------------------
addSQLErrorListener(const Reference<XSQLErrorListener> & _rListener)2360 void SAL_CALL ODatabaseForm::addSQLErrorListener(const Reference<XSQLErrorListener>& _rListener) throw( RuntimeException )
2361 {
2362 m_aErrorListeners.addInterface(_rListener);
2363 }
2364
2365 //------------------------------------------------------------------------------
removeSQLErrorListener(const Reference<XSQLErrorListener> & _rListener)2366 void SAL_CALL ODatabaseForm::removeSQLErrorListener(const Reference<XSQLErrorListener>& _rListener) throw( RuntimeException )
2367 {
2368 m_aErrorListeners.removeInterface(_rListener);
2369 }
2370
2371 //------------------------------------------------------------------------------
invlidateParameters()2372 void ODatabaseForm::invlidateParameters()
2373 {
2374 ::osl::MutexGuard aGuard(m_aMutex);
2375 m_aParameterManager.clearAllParameterInformation();
2376 }
2377
2378 //==============================================================================
2379 // OChangeListener
2380 //------------------------------------------------------------------------------
_propertyChanged(const PropertyChangeEvent & evt)2381 void ODatabaseForm::_propertyChanged(const PropertyChangeEvent& evt) throw( RuntimeException )
2382 {
2383 if ((0 == evt.PropertyName.compareToAscii(PROPERTY_ACTIVE_CONNECTION)) && !m_bForwardingConnection)
2384 {
2385 // the rowset changed its active connection itself (without interaction from our side), so
2386 // we need to fire this event, too
2387 sal_Int32 nHandle = PROPERTY_ID_ACTIVE_CONNECTION;
2388 fire(&nHandle, &evt.NewValue, &evt.OldValue, 1, sal_False);
2389 }
2390 else // it was one of the statement relevant props
2391 {
2392 // if the statement has changed we have to delete the parameter info
2393 invlidateParameters();
2394 }
2395 }
2396
2397 //==============================================================================
2398 // smartXChild
2399 //------------------------------------------------------------------------------
setParent(const InterfaceRef & Parent)2400 void SAL_CALL ODatabaseForm::setParent(const InterfaceRef& Parent) throw ( ::com::sun::star::lang::NoSupportException, ::com::sun::star::uno::RuntimeException)
2401 {
2402 // SYNCHRONIZED ----->
2403 ::osl::ResettableMutexGuard aGuard(m_aMutex);
2404
2405 Reference<XForm> xParentForm(getParent(), UNO_QUERY);
2406 if (xParentForm.is())
2407 {
2408 try
2409 {
2410 Reference< XRowSetApproveBroadcaster > xParentApprBroadcast( xParentForm, UNO_QUERY_THROW );
2411 xParentApprBroadcast->removeRowSetApproveListener( this );
2412
2413 Reference< XLoadable > xParentLoadable( xParentForm, UNO_QUERY_THROW );
2414 xParentLoadable->removeLoadListener( this );
2415
2416 Reference< XPropertySet > xParentProperties( xParentForm, UNO_QUERY_THROW );
2417 xParentProperties->removePropertyChangeListener( PROPERTY_ISNEW, this );
2418 }
2419 catch( const Exception& )
2420 {
2421 DBG_UNHANDLED_EXCEPTION();
2422 }
2423 }
2424
2425 OFormComponents::setParent(Parent);
2426
2427 xParentForm.set(getParent(), UNO_QUERY);
2428 if ( xParentForm.is() )
2429 {
2430 try
2431 {
2432 Reference< XRowSetApproveBroadcaster > xParentApprBroadcast( xParentForm, UNO_QUERY_THROW );
2433 xParentApprBroadcast->addRowSetApproveListener( this );
2434
2435 Reference< XLoadable > xParentLoadable( xParentForm, UNO_QUERY_THROW );
2436 xParentLoadable->addLoadListener( this );
2437
2438 Reference< XPropertySet > xParentProperties( xParentForm, UNO_QUERY_THROW );
2439 xParentProperties->addPropertyChangeListener( PROPERTY_ISNEW, this );
2440 }
2441 catch( const Exception& )
2442 {
2443 DBG_UNHANDLED_EXCEPTION();
2444 }
2445 }
2446
2447 Reference< XPropertySet > xAggregateProperties( m_xAggregateSet );
2448 aGuard.clear();
2449 // <----- SYNCHRONIZED
2450
2451 Reference< XConnection > xOuterConnection;
2452 sal_Bool bIsEmbedded = ::dbtools::isEmbeddedInDatabase( Parent, xOuterConnection );
2453
2454 if ( bIsEmbedded )
2455 xAggregateProperties->setPropertyValue( PROPERTY_DATASOURCE, makeAny( ::rtl::OUString() ) );
2456 }
2457
2458 //==============================================================================
2459 // smartXTabControllerModel
2460 //------------------------------------------------------------------------------
getGroupControl()2461 sal_Bool SAL_CALL ODatabaseForm::getGroupControl() throw(com::sun::star::uno::RuntimeException)
2462 {
2463 ::osl::ResettableMutexGuard aGuard(m_aMutex);
2464
2465 // Sollen Controls in einer TabOrder gruppe zusammengefasst werden?
2466 if (m_aCycle.hasValue())
2467 {
2468 sal_Int32 nCycle = 0;
2469 ::cppu::enum2int(nCycle, m_aCycle);
2470 return nCycle != TabulatorCycle_PAGE;
2471 }
2472
2473 if (isLoaded() && getConnection().is())
2474 return sal_True;
2475
2476 return sal_False;
2477 }
2478
2479 //------------------------------------------------------------------------------
setControlModels(const Sequence<Reference<XControlModel>> & rControls)2480 void SAL_CALL ODatabaseForm::setControlModels(const Sequence<Reference<XControlModel> >& rControls) throw( RuntimeException )
2481 {
2482 ::osl::ResettableMutexGuard aGuard(m_aMutex);
2483
2484 // TabIndex in der Reihenfolge der Sequence setzen
2485 const Reference<XControlModel>* pControls = rControls.getConstArray();
2486 sal_Int16 nTabIndex = 1;
2487 sal_Int32 nCount = getCount();
2488 sal_Int32 nNewCount = rControls.getLength();
2489
2490 // HiddenControls und Formulare werden nicht aufgefuehrt
2491 if (nNewCount <= nCount)
2492 {
2493 Any aElement;
2494 for (sal_Int32 i=0; i < nNewCount; ++i, ++pControls)
2495 {
2496 Reference<XFormComponent> xComp(*pControls, UNO_QUERY);
2497 if (xComp.is())
2498 {
2499 // suchen der Componente in der Liste
2500 for (sal_Int32 j = 0; j < nCount; ++j)
2501 {
2502 Reference<XFormComponent> xElement;
2503 ::cppu::extractInterface(xElement, getByIndex(j));
2504 if (xComp == xElement)
2505 {
2506 Reference<XPropertySet> xSet(xComp, UNO_QUERY);
2507 if (xSet.is() && hasProperty(PROPERTY_TABINDEX, xSet))
2508 xSet->setPropertyValue( PROPERTY_TABINDEX, makeAny(nTabIndex++) );
2509 break;
2510 }
2511 }
2512 }
2513 }
2514 }
2515 }
2516
2517 //------------------------------------------------------------------------------
getControlModels()2518 Sequence<Reference<XControlModel> > SAL_CALL ODatabaseForm::getControlModels() throw( RuntimeException )
2519 {
2520 ::osl::MutexGuard aGuard(m_aMutex);
2521 return m_pGroupManager->getControlModels();
2522 }
2523
2524 //------------------------------------------------------------------------------
setGroup(const Sequence<Reference<XControlModel>> & _rGroup,const::rtl::OUString & Name)2525 void SAL_CALL ODatabaseForm::setGroup( const Sequence<Reference<XControlModel> >& _rGroup, const ::rtl::OUString& Name ) throw( RuntimeException )
2526 {
2527 ::osl::MutexGuard aGuard(m_aMutex);
2528
2529 // Die Controls werden gruppiert, indem ihr Name dem Namen des ersten
2530 // Controls der Sequenz angepasst wird
2531 const Reference<XControlModel>* pControls = _rGroup.getConstArray();
2532 Reference< XPropertySet > xSet;
2533 ::rtl::OUString sGroupName( Name );
2534
2535 for( sal_Int32 i=0; i<_rGroup.getLength(); ++i, ++pControls )
2536 {
2537 xSet = xSet.query( *pControls );
2538 if ( !xSet.is() )
2539 {
2540 // can't throw an exception other than a RuntimeException (which would not be appropriate),
2541 // so we ignore (and only assert) this
2542 OSL_ENSURE( sal_False, "ODatabaseForm::setGroup: invalid arguments!" );
2543 continue;
2544 }
2545
2546 if (!sGroupName.getLength())
2547 xSet->getPropertyValue(PROPERTY_NAME) >>= sGroupName;
2548 else
2549 xSet->setPropertyValue(PROPERTY_NAME, makeAny(sGroupName));
2550 }
2551 }
2552
2553 //------------------------------------------------------------------------------
getGroupCount()2554 sal_Int32 SAL_CALL ODatabaseForm::getGroupCount() throw( RuntimeException )
2555 {
2556 ::osl::MutexGuard aGuard(m_aMutex);
2557 return m_pGroupManager->getGroupCount();
2558 }
2559
2560 //------------------------------------------------------------------------------
getGroup(sal_Int32 nGroup,Sequence<Reference<XControlModel>> & _rGroup,::rtl::OUString & _rName)2561 void SAL_CALL ODatabaseForm::getGroup( sal_Int32 nGroup, Sequence<Reference<XControlModel> >& _rGroup, ::rtl::OUString& _rName ) throw( RuntimeException )
2562 {
2563 ::osl::MutexGuard aGuard(m_aMutex);
2564 _rGroup.realloc(0);
2565 _rName = ::rtl::OUString();
2566
2567 if ((nGroup < 0) || (nGroup >= m_pGroupManager->getGroupCount()))
2568 return;
2569 m_pGroupManager->getGroup( nGroup, _rGroup, _rName );
2570 }
2571
2572 //------------------------------------------------------------------------------
getGroupByName(const::rtl::OUString & Name,Sequence<Reference<XControlModel>> & _rGroup)2573 void SAL_CALL ODatabaseForm::getGroupByName(const ::rtl::OUString& Name, Sequence< Reference<XControlModel> >& _rGroup) throw( RuntimeException )
2574 {
2575 ::osl::MutexGuard aGuard(m_aMutex);
2576 _rGroup.realloc(0);
2577 m_pGroupManager->getGroupByName( Name, _rGroup );
2578 }
2579
2580 //==============================================================================
2581 // com::sun::star::lang::XEventListener
2582 //------------------------------------------------------------------------------
disposing(const EventObject & Source)2583 void SAL_CALL ODatabaseForm::disposing(const EventObject& Source) throw( RuntimeException )
2584 {
2585 // does the call come from the connection which we are sharing with our parent?
2586 if ( isSharingConnection() )
2587 {
2588 Reference< XConnection > xConnSource( Source.Source, UNO_QUERY );
2589 if ( xConnSource.is() )
2590 {
2591 #if OSL_DEBUG_LEVEL > 0
2592 Reference< XConnection > xActiveConn;
2593 m_xAggregateSet->getPropertyValue( PROPERTY_ACTIVE_CONNECTION ) >>= xActiveConn;
2594 OSL_ENSURE( xActiveConn.get() == xConnSource.get(), "ODatabaseForm::disposing: where did this come from?" );
2595 // there should be exactly one XConnection object we're listening at - our aggregate connection
2596 #endif
2597 disposingSharedConnection( xConnSource );
2598 }
2599 }
2600
2601 OInterfaceContainer::disposing(Source);
2602
2603 // does the disposing come from the aggregate ?
2604 if (m_xAggregate.is())
2605 { // no -> forward it
2606 com::sun::star::uno::Reference<com::sun::star::lang::XEventListener> xListener;
2607 if (query_aggregation(m_xAggregate, xListener))
2608 xListener->disposing(Source);
2609 }
2610 }
2611
2612 //------------------------------------------------------------------------------
impl_createLoadTimer()2613 void ODatabaseForm::impl_createLoadTimer()
2614 {
2615 OSL_PRECOND( m_pLoadTimer == NULL, "ODatabaseForm::impl_createLoadTimer: timer already exists!" );
2616 m_pLoadTimer = new Timer();
2617 m_pLoadTimer->SetTimeout(100);
2618 m_pLoadTimer->SetTimeoutHdl(LINK(this,ODatabaseForm,OnTimeout));
2619 }
2620
2621 //==============================================================================
2622 // com::sun::star::form::XLoadListener
2623 //------------------------------------------------------------------------------
loaded(const EventObject &)2624 void SAL_CALL ODatabaseForm::loaded(const EventObject& /*aEvent*/) throw( RuntimeException )
2625 {
2626 {
2627 ::osl::MutexGuard aGuard( m_aMutex );
2628 Reference< XRowSet > xParentRowSet( m_xParent, UNO_QUERY_THROW );
2629 xParentRowSet->addRowSetListener( this );
2630
2631 impl_createLoadTimer();
2632 }
2633
2634 load_impl( sal_True );
2635 }
2636
2637 //------------------------------------------------------------------------------
unloading(const EventObject &)2638 void SAL_CALL ODatabaseForm::unloading(const EventObject& /*aEvent*/) throw( RuntimeException )
2639 {
2640 {
2641 // now stop the rowset listening if we are a subform
2642 ::osl::MutexGuard aGuard( m_aMutex );
2643
2644 if ( m_pLoadTimer && m_pLoadTimer->IsActive() )
2645 m_pLoadTimer->Stop();
2646 DELETEZ( m_pLoadTimer );
2647
2648 Reference< XRowSet > xParentRowSet( m_xParent, UNO_QUERY_THROW );
2649 xParentRowSet->removeRowSetListener( this );
2650 }
2651
2652 unload();
2653 }
2654
2655 //------------------------------------------------------------------------------
unloaded(const EventObject &)2656 void SAL_CALL ODatabaseForm::unloaded(const EventObject& /*aEvent*/) throw( RuntimeException )
2657 {
2658 // nothing to do
2659 }
2660
2661 //------------------------------------------------------------------------------
reloading(const EventObject &)2662 void SAL_CALL ODatabaseForm::reloading(const EventObject& /*aEvent*/) throw( RuntimeException )
2663 {
2664 // now stop the rowset listening if we are a subform
2665 ::osl::MutexGuard aGuard(m_aMutex);
2666 Reference<XRowSet> xParentRowSet(m_xParent, UNO_QUERY);
2667 if (xParentRowSet.is())
2668 xParentRowSet->removeRowSetListener(this);
2669
2670 if (m_pLoadTimer && m_pLoadTimer->IsActive())
2671 m_pLoadTimer->Stop();
2672 }
2673
2674 //------------------------------------------------------------------------------
reloaded(const EventObject &)2675 void SAL_CALL ODatabaseForm::reloaded(const EventObject& /*aEvent*/) throw( RuntimeException )
2676 {
2677 reload_impl(sal_True);
2678 {
2679 ::osl::MutexGuard aGuard(m_aMutex);
2680 Reference<XRowSet> xParentRowSet(m_xParent, UNO_QUERY);
2681 if (xParentRowSet.is())
2682 xParentRowSet->addRowSetListener(this);
2683 }
2684 }
2685
2686 //------------------------------------------------------------------------------
IMPL_LINK(ODatabaseForm,OnTimeout,void *,EMPTYARG)2687 IMPL_LINK( ODatabaseForm, OnTimeout, void*, EMPTYARG )
2688 {
2689 reload_impl(sal_True);
2690 return 1;
2691 }
2692
2693 //==============================================================================
2694 // com::sun::star::form::XLoadable
2695 //------------------------------------------------------------------------------
load()2696 void SAL_CALL ODatabaseForm::load() throw( RuntimeException )
2697 {
2698 load_impl(sal_False);
2699 }
2700
2701 //------------------------------------------------------------------------------
canShareConnection(const Reference<XPropertySet> & _rxParentProps)2702 sal_Bool ODatabaseForm::canShareConnection( const Reference< XPropertySet >& _rxParentProps )
2703 {
2704 // our own data source
2705 ::rtl::OUString sOwnDatasource;
2706 m_xAggregateSet->getPropertyValue( PROPERTY_DATASOURCE ) >>= sOwnDatasource;
2707
2708 // our parents data source
2709 ::rtl::OUString sParentDataSource;
2710 OSL_ENSURE( _rxParentProps.is() && _rxParentProps->getPropertySetInfo().is() && _rxParentProps->getPropertySetInfo()->hasPropertyByName( PROPERTY_DATASOURCE ),
2711 "ODatabaseForm::doShareConnection: invalid parent form!" );
2712 if ( _rxParentProps.is() )
2713 _rxParentProps->getPropertyValue( PROPERTY_DATASOURCE ) >>= sParentDataSource;
2714
2715 sal_Bool bCanShareConnection = sal_False;
2716
2717 // both rowsets share are connected to the same data source
2718 if ( sParentDataSource == sOwnDatasource )
2719 {
2720 if ( 0 != sParentDataSource.getLength() )
2721 // and it's really a data source name (not empty)
2722 bCanShareConnection = sal_True;
2723 else
2724 { // the data source name is empty
2725 // -> ook for the URL
2726 ::rtl::OUString sParentURL;
2727 ::rtl::OUString sMyURL;
2728 _rxParentProps->getPropertyValue( PROPERTY_URL ) >>= sParentURL;
2729 m_xAggregateSet->getPropertyValue( PROPERTY_URL ) >>= sMyURL;
2730
2731 bCanShareConnection = (sParentURL == sMyURL);
2732 }
2733 }
2734
2735 if ( bCanShareConnection )
2736 {
2737 // check for the user/password
2738
2739 // take the user property on the rowset (if any) into account
2740 ::rtl::OUString sParentUser, sParentPwd;
2741 _rxParentProps->getPropertyValue( PROPERTY_USER ) >>= sParentUser;
2742 _rxParentProps->getPropertyValue( PROPERTY_PASSWORD ) >>= sParentPwd;
2743
2744 ::rtl::OUString sMyUser, sMyPwd;
2745 m_xAggregateSet->getPropertyValue( PROPERTY_USER ) >>= sMyUser;
2746 m_xAggregateSet->getPropertyValue( PROPERTY_PASSWORD ) >>= sMyPwd;
2747
2748 bCanShareConnection =
2749 ( sParentUser == sMyUser )
2750 && ( sParentPwd == sMyPwd );
2751 }
2752
2753 return bCanShareConnection;
2754 }
2755
2756 //------------------------------------------------------------------------------
doShareConnection(const Reference<XPropertySet> & _rxParentProps)2757 void ODatabaseForm::doShareConnection( const Reference< XPropertySet >& _rxParentProps )
2758 {
2759 // get the conneciton of the parent
2760 Reference< XConnection > xParentConn;
2761 _rxParentProps->getPropertyValue( PROPERTY_ACTIVE_CONNECTION ) >>= xParentConn;
2762 OSL_ENSURE( xParentConn.is(), "ODatabaseForm::doShareConnection: we're a valid sub-form, but the parent has no connection?!" );
2763
2764 if ( xParentConn.is() )
2765 {
2766 // add as dispose listener to the connection
2767 Reference< XComponent > xParentConnComp( xParentConn, UNO_QUERY );
2768 OSL_ENSURE( xParentConnComp.is(), "ODatabaseForm::doShareConnection: invalid connection!" );
2769 xParentConnComp->addEventListener( static_cast< XLoadListener* >( this ) );
2770
2771 // forward the connection to our own aggreagte
2772 m_bForwardingConnection = sal_True;
2773 m_xAggregateSet->setPropertyValue( PROPERTY_ACTIVE_CONNECTION, makeAny( xParentConn ) );
2774 m_bForwardingConnection = sal_False;
2775
2776 m_bSharingConnection = sal_True;
2777 }
2778 else
2779 m_bSharingConnection = sal_False;
2780 }
2781
2782 //------------------------------------------------------------------------------
disposingSharedConnection(const Reference<XConnection> &)2783 void ODatabaseForm::disposingSharedConnection( const Reference< XConnection >& /*_rxConn*/ )
2784 {
2785 stopSharingConnection();
2786
2787 // TODO: we could think about whether or not to re-connect.
2788 unload( );
2789 }
2790
2791 //------------------------------------------------------------------------------
stopSharingConnection()2792 void ODatabaseForm::stopSharingConnection( )
2793 {
2794 OSL_ENSURE( m_bSharingConnection, "ODatabaseForm::stopSharingConnection: invalid call!" );
2795
2796 if ( m_bSharingConnection )
2797 {
2798 // get the connection
2799 Reference< XConnection > xSharedConn;
2800 m_xAggregateSet->getPropertyValue( PROPERTY_ACTIVE_CONNECTION ) >>= xSharedConn;
2801 OSL_ENSURE( xSharedConn.is(), "ODatabaseForm::stopSharingConnection: there's no conn!" );
2802
2803 // remove ourself as event listener
2804 Reference< XComponent > xSharedConnComp( xSharedConn, UNO_QUERY );
2805 if ( xSharedConnComp.is() )
2806 xSharedConnComp->removeEventListener( static_cast< XLoadListener* >( this ) );
2807
2808 // no need to dispose the conn: we're not the owner, this is our parent
2809 // (in addition, this method may be called if the connection is being disposed while we use it)
2810
2811 // reset the property
2812 xSharedConn.clear();
2813 m_bForwardingConnection = sal_True;
2814 m_xAggregateSet->setPropertyValue( PROPERTY_ACTIVE_CONNECTION, makeAny( xSharedConn ) );
2815 m_bForwardingConnection = sal_False;
2816
2817 // reset the flag
2818 m_bSharingConnection = sal_False;
2819 }
2820 }
2821
2822 //------------------------------------------------------------------------------
implEnsureConnection()2823 sal_Bool ODatabaseForm::implEnsureConnection()
2824 {
2825 try
2826 {
2827 if ( getConnection( ).is() )
2828 // if our aggregate already has a connection, nothing needs to be done about it
2829 return sal_True;
2830
2831 // see whether we're an embedded form
2832 Reference< XConnection > xOuterConnection;
2833 if ( ::dbtools::isEmbeddedInDatabase( getParent(), xOuterConnection ) )
2834 {
2835 m_xAggregateSet->setPropertyValue( PROPERTY_ACTIVE_CONNECTION, makeAny( xOuterConnection ) );
2836 return xOuterConnection.is();
2837 }
2838
2839 m_bSharingConnection = sal_False;
2840
2841 // if we're a sub form, we try to re-use the connection of our parent
2842 if (m_bSubForm)
2843 {
2844 OSL_ENSURE( Reference< XForm >( getParent(), UNO_QUERY ).is(),
2845 "ODatabaseForm::implEnsureConnection: m_bSubForm is TRUE, but the parent is no form?" );
2846
2847 Reference< XPropertySet > xParentProps( getParent(), UNO_QUERY );
2848
2849 // can we re-use (aka share) the connection of the parent?
2850 if ( canShareConnection( xParentProps ) )
2851 {
2852 // yep -> do it
2853 doShareConnection( xParentProps );
2854 // success?
2855 if ( m_bSharingConnection )
2856 // yes -> outta here
2857 return sal_True;
2858 }
2859 }
2860
2861 if (m_xAggregateSet.is())
2862 {
2863 Reference< XConnection > xConnection = connectRowset(
2864 Reference<XRowSet> (m_xAggregate, UNO_QUERY),
2865 m_xServiceFactory,
2866 sal_True // set a calculated connection as ActiveConnection
2867 );
2868 return xConnection.is();
2869 }
2870 }
2871 catch(SQLException& eDB)
2872 {
2873 onError(eDB, FRM_RES_STRING(RID_STR_CONNECTERROR));
2874 }
2875 catch( Exception )
2876 {
2877 DBG_UNHANDLED_EXCEPTION();
2878 }
2879
2880 return sal_False;
2881 }
2882
2883 //------------------------------------------------------------------------------
load_impl(sal_Bool bCausedByParentForm,sal_Bool bMoveToFirst,const Reference<XInteractionHandler> & _rxCompletionHandler)2884 void ODatabaseForm::load_impl(sal_Bool bCausedByParentForm, sal_Bool bMoveToFirst, const Reference< XInteractionHandler >& _rxCompletionHandler ) throw( RuntimeException )
2885 {
2886 ::osl::ResettableMutexGuard aGuard(m_aMutex);
2887
2888 // are we already loaded?
2889 if (isLoaded())
2890 return;
2891
2892 m_bSubForm = bCausedByParentForm;
2893
2894 // if we don't have a connection, we are not intended to be a database form or the aggregate was not able
2895 // to establish a connection
2896 sal_Bool bConnected = implEnsureConnection();
2897
2898 // we don't have to execute if we do not have a command to execute
2899 sal_Bool bExecute = bConnected && m_xAggregateSet.is() && getString(m_xAggregateSet->getPropertyValue(PROPERTY_COMMAND)).getLength();
2900
2901 // a database form always uses caching
2902 // we use starting fetchsize with at least 10 rows
2903 if (bConnected)
2904 m_xAggregateSet->setPropertyValue(PROPERTY_FETCHSIZE, makeAny((sal_Int32)40));
2905
2906 // if we're loaded as sub form we got a "rowSetChanged" from the parent rowset _before_ we got the "loaded"
2907 // so we don't need to execute the statement again, this was already done
2908 // (and there were no relevant changes between these two listener calls, the "load" of a form is quite an
2909 // atomar operation.)
2910
2911 sal_Bool bSuccess = sal_False;
2912 if (bExecute)
2913 {
2914 m_sCurrentErrorContext = FRM_RES_STRING(RID_ERR_LOADING_FORM);
2915 bSuccess = executeRowSet(aGuard, bMoveToFirst, _rxCompletionHandler);
2916 }
2917
2918 if (bSuccess)
2919 {
2920 m_bLoaded = sal_True;
2921 aGuard.clear();
2922 EventObject aEvt(static_cast<XWeak*>(this));
2923 m_aLoadListeners.notifyEach( &XLoadListener::loaded, aEvt );
2924
2925 // if we are on the insert row, we have to reset all controls
2926 // to set the default values
2927 if (bExecute && getBOOL(m_xAggregateSet->getPropertyValue(PROPERTY_ISNEW)))
2928 reset();
2929 }
2930 }
2931
2932 //------------------------------------------------------------------------------
unload()2933 void SAL_CALL ODatabaseForm::unload() throw( RuntimeException )
2934 {
2935 ::osl::ResettableMutexGuard aGuard(m_aMutex);
2936 if (!isLoaded())
2937 return;
2938
2939 DELETEZ(m_pLoadTimer);
2940
2941 aGuard.clear();
2942 EventObject aEvt(static_cast<XWeak*>(this));
2943 m_aLoadListeners.notifyEach( &XLoadListener::unloading, aEvt );
2944
2945 if (m_xAggregateAsRowSet.is())
2946 {
2947 // we may have reset the InsertOnly property on the aggregate - restore it
2948 restoreInsertOnlyState( );
2949
2950 // clear the parameters if there are any
2951 invlidateParameters();
2952
2953 try
2954 {
2955 // close the aggregate
2956 Reference<XCloseable> xCloseable;
2957 query_aggregation( m_xAggregate, xCloseable);
2958 aGuard.clear();
2959 if (xCloseable.is())
2960 xCloseable->close();
2961 }
2962 catch( const SQLException& e )
2963 {
2964 (void)e;
2965 }
2966 aGuard.reset();
2967 }
2968
2969 m_bLoaded = sal_False;
2970
2971 // if the connection we used while we were loaded is only shared with our parent, we
2972 // reset it
2973 if ( isSharingConnection() )
2974 stopSharingConnection();
2975
2976 aGuard.clear();
2977 m_aLoadListeners.notifyEach( &XLoadListener::unloaded, aEvt );
2978 }
2979
2980 //------------------------------------------------------------------------------
reload()2981 void SAL_CALL ODatabaseForm::reload() throw( RuntimeException )
2982 {
2983 reload_impl(sal_True);
2984 }
2985
2986 //------------------------------------------------------------------------------
reload_impl(sal_Bool bMoveToFirst,const Reference<XInteractionHandler> & _rxCompletionHandler)2987 void ODatabaseForm::reload_impl(sal_Bool bMoveToFirst, const Reference< XInteractionHandler >& _rxCompletionHandler ) throw( RuntimeException )
2988 {
2989 ::osl::ResettableMutexGuard aGuard(m_aMutex);
2990 if (!isLoaded())
2991 return;
2992
2993 DocumentModifyGuard aModifyGuard( *this );
2994 // ensures the document is not marked as "modified" just because we change some control's content during
2995 // reloading ...
2996
2997 EventObject aEvent(static_cast<XWeak*>(this));
2998 {
2999 // only if there is no approve listener we can post the event at this time
3000 // otherwise see approveRowsetChange
3001 // the aprrovement is done by the aggregate
3002 if (!m_aRowSetApproveListeners.getLength())
3003 {
3004 ::cppu::OInterfaceIteratorHelper aIter(m_aLoadListeners);
3005 aGuard.clear();
3006
3007 while (aIter.hasMoreElements())
3008 ((XLoadListener*)aIter.next())->reloading(aEvent);
3009
3010 aGuard.reset();
3011 }
3012 }
3013
3014 sal_Bool bSuccess = sal_True;
3015 try
3016 {
3017 m_sCurrentErrorContext = FRM_RES_STRING(RID_ERR_REFRESHING_FORM);
3018 bSuccess = executeRowSet(aGuard, bMoveToFirst, _rxCompletionHandler);
3019 }
3020 catch( const SQLException& e )
3021 {
3022 DBG_ERROR("ODatabaseForm::reload_impl : shouldn't executeRowSet catch this exception?");
3023 (void)e;
3024 }
3025
3026 if (bSuccess)
3027 {
3028 ::cppu::OInterfaceIteratorHelper aIter(m_aLoadListeners);
3029 aGuard.clear();
3030 while (aIter.hasMoreElements())
3031 ((XLoadListener*)aIter.next())->reloaded(aEvent);
3032
3033 // if we are on the insert row, we have to reset all controls
3034 // to set the default values
3035 if (getBOOL(m_xAggregateSet->getPropertyValue(PROPERTY_ISNEW)))
3036 reset();
3037 }
3038 else
3039 m_bLoaded = sal_False;
3040 }
3041
3042 //------------------------------------------------------------------------------
isLoaded()3043 sal_Bool SAL_CALL ODatabaseForm::isLoaded() throw( RuntimeException )
3044 {
3045 return m_bLoaded;
3046 }
3047
3048 //------------------------------------------------------------------------------
addLoadListener(const Reference<XLoadListener> & aListener)3049 void SAL_CALL ODatabaseForm::addLoadListener(const Reference<XLoadListener>& aListener) throw( RuntimeException )
3050 {
3051 m_aLoadListeners.addInterface(aListener);
3052 }
3053
3054 //------------------------------------------------------------------------------
removeLoadListener(const Reference<XLoadListener> & aListener)3055 void SAL_CALL ODatabaseForm::removeLoadListener(const Reference<XLoadListener>& aListener) throw( RuntimeException )
3056 {
3057 m_aLoadListeners.removeInterface(aListener);
3058 }
3059
3060 //==============================================================================
3061 // com::sun::star::sdbc::XCloseable
3062 //==============================================================================
close()3063 void SAL_CALL ODatabaseForm::close() throw( SQLException, RuntimeException )
3064 {
3065 // unload will close the aggregate
3066 unload();
3067 }
3068
3069 //==============================================================================
3070 // com::sun::star::sdbc::XRowSetListener
3071 //------------------------------------------------------------------------------
cursorMoved(const EventObject &)3072 void SAL_CALL ODatabaseForm::cursorMoved(const EventObject& /*event*/) throw( RuntimeException )
3073 {
3074 // reload the subform with the new parameters of the parent
3075 // do this handling delayed to provide of execute too many SQL Statements
3076 ::osl::ResettableMutexGuard aGuard(m_aMutex);
3077
3078 DBG_ASSERT( m_pLoadTimer, "ODatabaseForm::cursorMoved: how can this happen?!" );
3079 if ( !m_pLoadTimer )
3080 impl_createLoadTimer();
3081
3082 if ( m_pLoadTimer->IsActive() )
3083 m_pLoadTimer->Stop();
3084
3085 // and start the timer again
3086 m_pLoadTimer->Start();
3087 }
3088
3089 //------------------------------------------------------------------------------
rowChanged(const EventObject &)3090 void SAL_CALL ODatabaseForm::rowChanged(const EventObject& /*event*/) throw( RuntimeException )
3091 {
3092 // ignore it
3093 }
3094
3095 //------------------------------------------------------------------------------
rowSetChanged(const EventObject &)3096 void SAL_CALL ODatabaseForm::rowSetChanged(const EventObject& /*event*/) throw( RuntimeException )
3097 {
3098 // not interested in :
3099 // if our parent is an ODatabaseForm, too, then after this rowSetChanged we'll get a "reloaded"
3100 // or a "loaded" event.
3101 // If somebody gave us another parent which is an XRowSet but doesn't handle an execute as
3102 // "load" respectively "reload" ... can't do anything ....
3103 }
3104
3105 //------------------------------------------------------------------------------
impl_approveRowChange_throw(const EventObject & _rEvent,const bool _bAllowSQLException,::osl::ClearableMutexGuard & _rGuard)3106 bool ODatabaseForm::impl_approveRowChange_throw( const EventObject& _rEvent, const bool _bAllowSQLException,
3107 ::osl::ClearableMutexGuard& _rGuard )
3108 {
3109 ::cppu::OInterfaceIteratorHelper aIter( m_aRowSetApproveListeners );
3110 _rGuard.clear();
3111 while ( aIter.hasMoreElements() )
3112 {
3113 Reference< XRowSetApproveListener > xListener( static_cast< XRowSetApproveListener* >( aIter.next() ) );
3114 if ( !xListener.is() )
3115 continue;
3116
3117 try
3118 {
3119 if ( !xListener->approveRowSetChange( _rEvent ) )
3120 return false;
3121 }
3122 catch ( const DisposedException& e )
3123 {
3124 if ( e.Context == xListener )
3125 aIter.remove();
3126 }
3127 catch ( const RuntimeException& ) { throw; }
3128 catch ( const SQLException& )
3129 {
3130 if ( _bAllowSQLException )
3131 throw;
3132 DBG_UNHANDLED_EXCEPTION();
3133 }
3134 catch ( const Exception& )
3135 {
3136 DBG_UNHANDLED_EXCEPTION();
3137 }
3138 }
3139 return true;
3140 }
3141
3142 //------------------------------------------------------------------------------
approveCursorMove(const EventObject & event)3143 sal_Bool SAL_CALL ODatabaseForm::approveCursorMove(const EventObject& event) throw( RuntimeException )
3144 {
3145 // is our aggregate calling?
3146 if (event.Source == InterfaceRef(static_cast<XWeak*>(this)))
3147 {
3148 // Our aggregate doesn't have any ApproveRowSetListeners (expect ourself), as we re-routed the queryInterface
3149 // for XRowSetApproveBroadcaster-interface.
3150 // So we have to multiplex this approve request.
3151 ::cppu::OInterfaceIteratorHelper aIter( m_aRowSetApproveListeners );
3152 while ( aIter.hasMoreElements() )
3153 {
3154 Reference< XRowSetApproveListener > xListener( static_cast< XRowSetApproveListener* >( aIter.next() ) );
3155 if ( !xListener.is() )
3156 continue;
3157
3158 try
3159 {
3160 if ( !xListener->approveCursorMove( event ) )
3161 return sal_False;
3162 }
3163 catch ( const DisposedException& e )
3164 {
3165 if ( e.Context == xListener )
3166 aIter.remove();
3167 }
3168 catch ( const RuntimeException& ) { throw; }
3169 catch ( const Exception& )
3170 {
3171 DBG_UNHANDLED_EXCEPTION();
3172 }
3173 }
3174 return true;
3175 }
3176 else
3177 {
3178 // this is a call from our parent ...
3179 // a parent's cursor move will result in a re-execute of our own row-set, so we have to
3180 // ask our own RowSetChangesListeners, too
3181 ::osl::ClearableMutexGuard aGuard( m_aMutex );
3182 if ( !impl_approveRowChange_throw( event, false, aGuard ) )
3183 return sal_False;
3184 }
3185 return sal_True;
3186 }
3187
3188 //------------------------------------------------------------------------------
approveRowChange(const RowChangeEvent & event)3189 sal_Bool SAL_CALL ODatabaseForm::approveRowChange(const RowChangeEvent& event) throw( RuntimeException )
3190 {
3191 // is our aggregate calling?
3192 if (event.Source == InterfaceRef(static_cast<XWeak*>(this)))
3193 {
3194 // Our aggregate doesn't have any ApproveRowSetListeners (expect ourself), as we re-routed the queryInterface
3195 // for XRowSetApproveBroadcaster-interface.
3196 // So we have to multiplex this approve request.
3197 ::cppu::OInterfaceIteratorHelper aIter( m_aRowSetApproveListeners );
3198 while ( aIter.hasMoreElements() )
3199 {
3200 Reference< XRowSetApproveListener > xListener( static_cast< XRowSetApproveListener* >( aIter.next() ) );
3201 if ( !xListener.is() )
3202 continue;
3203
3204 try
3205 {
3206 if ( !xListener->approveRowChange( event ) )
3207 return false;
3208 }
3209 catch ( const DisposedException& e )
3210 {
3211 if ( e.Context == xListener )
3212 aIter.remove();
3213 }
3214 catch ( const RuntimeException& ) { throw; }
3215 catch ( const Exception& )
3216 {
3217 DBG_UNHANDLED_EXCEPTION();
3218 }
3219 }
3220 return true;
3221 }
3222 return sal_True;
3223 }
3224
3225 //------------------------------------------------------------------------------
approveRowSetChange(const EventObject & event)3226 sal_Bool SAL_CALL ODatabaseForm::approveRowSetChange(const EventObject& event) throw( RuntimeException )
3227 {
3228 if (event.Source == InterfaceRef(static_cast<XWeak*>(this))) // ignore our aggregate as we handle this approve ourself
3229 {
3230 ::osl::ClearableMutexGuard aGuard( m_aMutex );
3231 bool bWasLoaded = isLoaded();
3232 if ( !impl_approveRowChange_throw( event, false, aGuard ) )
3233 return sal_False;
3234
3235 if ( bWasLoaded )
3236 {
3237 m_aLoadListeners.notifyEach( &XLoadListener::reloading, event );
3238 }
3239 }
3240 else
3241 {
3242 // this is a call from our parent ...
3243 // a parent's cursor move will result in a re-execute of our own row-set, so we have to
3244 // ask our own RowSetChangesListeners, too
3245 ::osl::ClearableMutexGuard aGuard( m_aMutex );
3246 if ( !impl_approveRowChange_throw( event, false, aGuard ) )
3247 return sal_False;
3248 }
3249 return sal_True;
3250 }
3251
3252 //==============================================================================
3253 // com::sun::star::sdb::XRowSetApproveBroadcaster
3254 //------------------------------------------------------------------------------
addRowSetApproveListener(const Reference<XRowSetApproveListener> & _rListener)3255 void SAL_CALL ODatabaseForm::addRowSetApproveListener(const Reference<XRowSetApproveListener>& _rListener) throw( RuntimeException )
3256 {
3257 ::osl::ResettableMutexGuard aGuard(m_aMutex);
3258 m_aRowSetApproveListeners.addInterface(_rListener);
3259
3260 // do we have to multiplex ?
3261 if (m_aRowSetApproveListeners.getLength() == 1)
3262 {
3263 Reference<XRowSetApproveBroadcaster> xBroadcaster;
3264 if (query_aggregation( m_xAggregate, xBroadcaster))
3265 {
3266 Reference<XRowSetApproveListener> xListener((XRowSetApproveListener*)this);
3267 xBroadcaster->addRowSetApproveListener(xListener);
3268 }
3269 }
3270 }
3271
3272 //------------------------------------------------------------------------------
removeRowSetApproveListener(const Reference<XRowSetApproveListener> & _rListener)3273 void SAL_CALL ODatabaseForm::removeRowSetApproveListener(const Reference<XRowSetApproveListener>& _rListener) throw( RuntimeException )
3274 {
3275 ::osl::ResettableMutexGuard aGuard(m_aMutex);
3276 // do we have to remove the multiplex ?
3277 m_aRowSetApproveListeners.removeInterface(_rListener);
3278 if ( m_aRowSetApproveListeners.getLength() == 0 )
3279 {
3280 Reference<XRowSetApproveBroadcaster> xBroadcaster;
3281 if (query_aggregation( m_xAggregate, xBroadcaster))
3282 {
3283 Reference<XRowSetApproveListener> xListener((XRowSetApproveListener*)this);
3284 xBroadcaster->removeRowSetApproveListener(xListener);
3285 }
3286 }
3287 }
3288
3289 //==============================================================================
3290 // com::sun:star::form::XDatabaseParameterBroadcaster
3291 //------------------------------------------------------------------------------
addDatabaseParameterListener(const Reference<XDatabaseParameterListener> & _rListener)3292 void SAL_CALL ODatabaseForm::addDatabaseParameterListener(const Reference<XDatabaseParameterListener>& _rListener) throw( RuntimeException )
3293 {
3294 m_aParameterManager.addParameterListener( _rListener );
3295 }
3296 //------------------------------------------------------------------------------
removeDatabaseParameterListener(const Reference<XDatabaseParameterListener> & _rListener)3297 void SAL_CALL ODatabaseForm::removeDatabaseParameterListener(const Reference<XDatabaseParameterListener>& _rListener) throw( RuntimeException )
3298 {
3299 m_aParameterManager.removeParameterListener( _rListener );
3300 }
3301
3302 //------------------------------------------------------------------------------
addParameterListener(const Reference<XDatabaseParameterListener> & _rListener)3303 void SAL_CALL ODatabaseForm::addParameterListener(const Reference<XDatabaseParameterListener>& _rListener) throw( RuntimeException )
3304 {
3305 ODatabaseForm::addDatabaseParameterListener( _rListener );
3306 }
3307
3308 //------------------------------------------------------------------------------
removeParameterListener(const Reference<XDatabaseParameterListener> & _rListener)3309 void SAL_CALL ODatabaseForm::removeParameterListener(const Reference<XDatabaseParameterListener>& _rListener) throw( RuntimeException )
3310 {
3311 ODatabaseForm::removeDatabaseParameterListener( _rListener );
3312 }
3313
3314 //==============================================================================
3315 // com::sun::star::sdb::XCompletedExecution
3316 //------------------------------------------------------------------------------
executeWithCompletion(const Reference<XInteractionHandler> & _rxHandler)3317 void SAL_CALL ODatabaseForm::executeWithCompletion( const Reference< XInteractionHandler >& _rxHandler ) throw(SQLException, RuntimeException)
3318 {
3319 ::osl::ClearableMutexGuard aGuard(m_aMutex);
3320 // the difference between execute and load is, that we position on the first row in case of load
3321 // after execute we remain before the first row
3322 if (!isLoaded())
3323 {
3324 aGuard.clear();
3325 load_impl(sal_False, sal_False, _rxHandler);
3326 }
3327 else
3328 {
3329 EventObject event(static_cast< XWeak* >(this));
3330 if ( !impl_approveRowChange_throw( event, true, aGuard ) )
3331 return;
3332
3333 // we're loaded and somebody want's to execute ourself -> this means a reload
3334 reload_impl(sal_False, _rxHandler);
3335 }
3336 }
3337
3338 //==============================================================================
3339 // com::sun::star::sdbc::XRowSet
3340 //------------------------------------------------------------------------------
execute()3341 void SAL_CALL ODatabaseForm::execute() throw( SQLException, RuntimeException )
3342 {
3343 ::osl::ResettableMutexGuard aGuard(m_aMutex);
3344 // if somebody calls an execute and we're not loaded we reroute this call to our load method.
3345
3346 // the difference between execute and load is, that we position on the first row in case of load
3347 // after execute we remain before the first row
3348 if (!isLoaded())
3349 {
3350 aGuard.clear();
3351 load_impl(sal_False, sal_False);
3352 }
3353 else
3354 {
3355 EventObject event(static_cast< XWeak* >(this));
3356 if ( !impl_approveRowChange_throw( event, true, aGuard ) )
3357 return;
3358
3359 // we're loaded and somebody want's to execute ourself -> this means a reload
3360 reload_impl(sal_False);
3361 }
3362 }
3363
3364 //------------------------------------------------------------------------------
addRowSetListener(const Reference<XRowSetListener> & _rListener)3365 void SAL_CALL ODatabaseForm::addRowSetListener(const Reference<XRowSetListener>& _rListener) throw( RuntimeException )
3366 {
3367 if (m_xAggregateAsRowSet.is())
3368 m_xAggregateAsRowSet->addRowSetListener(_rListener);
3369 }
3370
3371 //------------------------------------------------------------------------------
removeRowSetListener(const Reference<XRowSetListener> & _rListener)3372 void SAL_CALL ODatabaseForm::removeRowSetListener(const Reference<XRowSetListener>& _rListener) throw( RuntimeException )
3373 {
3374 if (m_xAggregateAsRowSet.is())
3375 m_xAggregateAsRowSet->removeRowSetListener(_rListener);
3376 }
3377
3378 //==============================================================================
3379 // com::sun::star::sdbc::XResultSet
3380 //------------------------------------------------------------------------------
next()3381 sal_Bool SAL_CALL ODatabaseForm::next() throw( SQLException, RuntimeException )
3382 {
3383 return m_xAggregateAsRowSet->next();
3384 }
3385
3386 //------------------------------------------------------------------------------
isBeforeFirst()3387 sal_Bool SAL_CALL ODatabaseForm::isBeforeFirst() throw( SQLException, RuntimeException )
3388 {
3389 return m_xAggregateAsRowSet->isBeforeFirst();
3390 }
3391
3392 //------------------------------------------------------------------------------
isAfterLast()3393 sal_Bool SAL_CALL ODatabaseForm::isAfterLast() throw( SQLException, RuntimeException )
3394 {
3395 return m_xAggregateAsRowSet->isAfterLast();
3396 }
3397
3398 //------------------------------------------------------------------------------
isFirst()3399 sal_Bool SAL_CALL ODatabaseForm::isFirst() throw( SQLException, RuntimeException )
3400 {
3401 return m_xAggregateAsRowSet->isFirst();
3402 }
3403
3404 //------------------------------------------------------------------------------
isLast()3405 sal_Bool SAL_CALL ODatabaseForm::isLast() throw( SQLException, RuntimeException )
3406 {
3407 return m_xAggregateAsRowSet->isLast();
3408 }
3409
3410 //------------------------------------------------------------------------------
beforeFirst()3411 void SAL_CALL ODatabaseForm::beforeFirst() throw( SQLException, RuntimeException )
3412 {
3413 m_xAggregateAsRowSet->beforeFirst();
3414 }
3415
3416 //------------------------------------------------------------------------------
afterLast()3417 void SAL_CALL ODatabaseForm::afterLast() throw( SQLException, RuntimeException )
3418 {
3419 m_xAggregateAsRowSet->afterLast();
3420 }
3421
3422 //------------------------------------------------------------------------------
first()3423 sal_Bool SAL_CALL ODatabaseForm::first() throw( SQLException, RuntimeException )
3424 {
3425 return m_xAggregateAsRowSet->first();
3426 }
3427
3428 //------------------------------------------------------------------------------
last()3429 sal_Bool SAL_CALL ODatabaseForm::last() throw( SQLException, RuntimeException )
3430 {
3431 return m_xAggregateAsRowSet->last();
3432 }
3433
3434 //------------------------------------------------------------------------------
getRow()3435 sal_Int32 SAL_CALL ODatabaseForm::getRow() throw( SQLException, RuntimeException )
3436 {
3437 return m_xAggregateAsRowSet->getRow();
3438 }
3439
3440 //------------------------------------------------------------------------------
absolute(sal_Int32 row)3441 sal_Bool SAL_CALL ODatabaseForm::absolute(sal_Int32 row) throw( SQLException, RuntimeException )
3442 {
3443 return m_xAggregateAsRowSet->absolute(row);
3444 }
3445
3446 //------------------------------------------------------------------------------
relative(sal_Int32 rows)3447 sal_Bool SAL_CALL ODatabaseForm::relative(sal_Int32 rows) throw( SQLException, RuntimeException )
3448 {
3449 return m_xAggregateAsRowSet->relative(rows);
3450 }
3451
3452 //------------------------------------------------------------------------------
previous()3453 sal_Bool SAL_CALL ODatabaseForm::previous() throw( SQLException, RuntimeException )
3454 {
3455 return m_xAggregateAsRowSet->previous();
3456 }
3457
3458 //------------------------------------------------------------------------------
refreshRow()3459 void SAL_CALL ODatabaseForm::refreshRow() throw( SQLException, RuntimeException )
3460 {
3461 m_xAggregateAsRowSet->refreshRow();
3462 }
3463
3464 //------------------------------------------------------------------------------
rowUpdated()3465 sal_Bool SAL_CALL ODatabaseForm::rowUpdated() throw( SQLException, RuntimeException )
3466 {
3467 return m_xAggregateAsRowSet->rowUpdated();
3468 }
3469
3470 //------------------------------------------------------------------------------
rowInserted()3471 sal_Bool SAL_CALL ODatabaseForm::rowInserted() throw( SQLException, RuntimeException )
3472 {
3473 return m_xAggregateAsRowSet->rowInserted();
3474 }
3475
3476 //------------------------------------------------------------------------------
rowDeleted()3477 sal_Bool SAL_CALL ODatabaseForm::rowDeleted() throw( SQLException, RuntimeException )
3478 {
3479 return m_xAggregateAsRowSet->rowDeleted();
3480 }
3481
3482 //------------------------------------------------------------------------------
getStatement()3483 InterfaceRef SAL_CALL ODatabaseForm::getStatement() throw( SQLException, RuntimeException )
3484 {
3485 return m_xAggregateAsRowSet->getStatement();
3486 }
3487
3488 // com::sun::star::sdbc::XResultSetUpdate
3489 // exceptions during insert update and delete will be forwarded to the errorlistener
3490 //------------------------------------------------------------------------------
insertRow()3491 void SAL_CALL ODatabaseForm::insertRow() throw( SQLException, RuntimeException )
3492 {
3493 try
3494 {
3495 Reference<XResultSetUpdate> xUpdate;
3496 if (query_aggregation( m_xAggregate, xUpdate))
3497 xUpdate->insertRow();
3498 }
3499 catch( const RowSetVetoException& eVeto )
3500 {
3501 (void)eVeto;
3502 throw;
3503 }
3504 catch(SQLException& eDb)
3505 {
3506 onError(eDb, FRM_RES_STRING(RID_STR_ERR_INSERTRECORD));
3507 throw;
3508 }
3509 }
3510
3511 //------------------------------------------------------------------------------
updateRow()3512 void SAL_CALL ODatabaseForm::updateRow() throw( SQLException, RuntimeException )
3513 {
3514 try
3515 {
3516 Reference<XResultSetUpdate> xUpdate;
3517 if (query_aggregation( m_xAggregate, xUpdate))
3518 xUpdate->updateRow();
3519 }
3520 catch( const RowSetVetoException& eVeto )
3521 {
3522 (void)eVeto;
3523 throw;
3524 }
3525 catch(SQLException& eDb)
3526 {
3527 onError(eDb, FRM_RES_STRING(RID_STR_ERR_UPDATERECORD));
3528 throw;
3529 }
3530 }
3531
3532 //------------------------------------------------------------------------------
deleteRow()3533 void SAL_CALL ODatabaseForm::deleteRow() throw( SQLException, RuntimeException )
3534 {
3535 try
3536 {
3537 Reference<XResultSetUpdate> xUpdate;
3538 if (query_aggregation( m_xAggregate, xUpdate))
3539 xUpdate->deleteRow();
3540 }
3541 catch( const RowSetVetoException& eVeto )
3542 {
3543 (void)eVeto;
3544 throw;
3545 }
3546 catch(SQLException& eDb)
3547 {
3548 onError(eDb, FRM_RES_STRING(RID_STR_ERR_DELETERECORD));
3549 throw;
3550 }
3551 }
3552
3553 //------------------------------------------------------------------------------
cancelRowUpdates()3554 void SAL_CALL ODatabaseForm::cancelRowUpdates() throw( SQLException, RuntimeException )
3555 {
3556 try
3557 {
3558 Reference<XResultSetUpdate> xUpdate;
3559 if (query_aggregation( m_xAggregate, xUpdate))
3560 xUpdate->cancelRowUpdates();
3561 }
3562 catch( const RowSetVetoException& eVeto )
3563 {
3564 (void)eVeto;
3565 throw;
3566 }
3567 catch(SQLException& eDb)
3568 {
3569 onError(eDb, FRM_RES_STRING(RID_STR_ERR_INSERTRECORD));
3570 throw;
3571 }
3572 }
3573
3574 //------------------------------------------------------------------------------
moveToInsertRow()3575 void SAL_CALL ODatabaseForm::moveToInsertRow() throw( SQLException, RuntimeException )
3576 {
3577 Reference<XResultSetUpdate> xUpdate;
3578 if (query_aggregation( m_xAggregate, xUpdate))
3579 {
3580 // _always_ move to the insert row
3581 //
3582 // Formerly, the following line was conditioned with a "not is new", means we did not move the aggregate
3583 // to the insert row if it was already positioned there.
3584 //
3585 // This prevented the RowSet implementation from resetting it's column values. We, ourself, formerly
3586 // did this reset of columns in reset_impl, where we set every column to the ControlDefault, or, if this
3587 // was not present, to NULL. However, the problem with setting to NULL was #88888#, the problem with
3588 // _not_ setting to NULL (which was the original fix for #88888#) was #97955#.
3589 //
3590 // So now we
3591 // * move our aggregate to the insert row
3592 // * in reset_impl
3593 // - set the control defaults into the columns if not void
3594 // - do _not_ set the columns to NULL if no control default is set
3595 // This fixes both #88888# and #97955#
3596 //
3597 // Still, there is #72756#. During fixing this bug, DG introduced not calling the aggregate here. So
3598 // in theory, we re-introduced #72756#. But the bug described therein does not happen anymore, as the
3599 // preliminaries for it changed (no display of guessed values for new records with autoinc fields)
3600 //
3601 // BTW: the public Issuezilla bug for #97955# is #i2815#
3602 //
3603 // 16.04.2002 - 97955 - fs@openoffice.org
3604 xUpdate->moveToInsertRow();
3605
3606 // then set the default values and the parameters given from the parent
3607 reset();
3608 }
3609 }
3610
3611 //------------------------------------------------------------------------------
moveToCurrentRow()3612 void SAL_CALL ODatabaseForm::moveToCurrentRow() throw( SQLException, RuntimeException )
3613 {
3614 Reference<XResultSetUpdate> xUpdate;
3615 if (query_aggregation( m_xAggregate, xUpdate))
3616 xUpdate->moveToCurrentRow();
3617 }
3618
3619 // com::sun::star::sdbcx::XDeleteRows
3620 //------------------------------------------------------------------------------
deleteRows(const Sequence<Any> & rows)3621 Sequence<sal_Int32> SAL_CALL ODatabaseForm::deleteRows(const Sequence<Any>& rows) throw( SQLException, RuntimeException )
3622 {
3623 try
3624 {
3625 Reference<XDeleteRows> xDelete;
3626 if (query_aggregation( m_xAggregate, xDelete))
3627 return xDelete->deleteRows(rows);
3628 }
3629 catch( const RowSetVetoException& eVeto )
3630 {
3631 (void)eVeto; // make compiler happy
3632 throw;
3633 }
3634 catch(SQLException& eDb)
3635 {
3636 onError(eDb, FRM_RES_STRING(RID_STR_ERR_DELETERECORDS));
3637 throw;
3638 }
3639
3640 return Sequence< sal_Int32 >();
3641 }
3642
3643 // com::sun::star::sdbc::XParameters
3644 //------------------------------------------------------------------------------
setNull(sal_Int32 parameterIndex,sal_Int32 sqlType)3645 void SAL_CALL ODatabaseForm::setNull(sal_Int32 parameterIndex, sal_Int32 sqlType) throw( SQLException, RuntimeException )
3646 {
3647 m_aParameterManager.setNull(parameterIndex, sqlType);
3648 }
3649
3650 //------------------------------------------------------------------------------
setObjectNull(sal_Int32 parameterIndex,sal_Int32 sqlType,const::rtl::OUString & typeName)3651 void SAL_CALL ODatabaseForm::setObjectNull(sal_Int32 parameterIndex, sal_Int32 sqlType, const ::rtl::OUString& typeName) throw( SQLException, RuntimeException )
3652 {
3653 m_aParameterManager.setObjectNull(parameterIndex, sqlType, typeName);
3654 }
3655
3656 //------------------------------------------------------------------------------
setBoolean(sal_Int32 parameterIndex,sal_Bool x)3657 void SAL_CALL ODatabaseForm::setBoolean(sal_Int32 parameterIndex, sal_Bool x) throw( SQLException, RuntimeException )
3658 {
3659 m_aParameterManager.setBoolean(parameterIndex, x);
3660 }
3661
3662 //------------------------------------------------------------------------------
setByte(sal_Int32 parameterIndex,sal_Int8 x)3663 void SAL_CALL ODatabaseForm::setByte(sal_Int32 parameterIndex, sal_Int8 x) throw( SQLException, RuntimeException )
3664 {
3665 m_aParameterManager.setByte(parameterIndex, x);
3666 }
3667
3668 //------------------------------------------------------------------------------
setShort(sal_Int32 parameterIndex,sal_Int16 x)3669 void SAL_CALL ODatabaseForm::setShort(sal_Int32 parameterIndex, sal_Int16 x) throw( SQLException, RuntimeException )
3670 {
3671 m_aParameterManager.setShort(parameterIndex, x);
3672 }
3673
3674 //------------------------------------------------------------------------------
setInt(sal_Int32 parameterIndex,sal_Int32 x)3675 void SAL_CALL ODatabaseForm::setInt(sal_Int32 parameterIndex, sal_Int32 x) throw( SQLException, RuntimeException )
3676 {
3677 m_aParameterManager.setInt(parameterIndex, x);
3678 }
3679
3680 //------------------------------------------------------------------------------
setLong(sal_Int32 parameterIndex,sal_Int64 x)3681 void SAL_CALL ODatabaseForm::setLong(sal_Int32 parameterIndex, sal_Int64 x) throw( SQLException, RuntimeException )
3682 {
3683 m_aParameterManager.setLong(parameterIndex, x);
3684 }
3685
3686 //------------------------------------------------------------------------------
setFloat(sal_Int32 parameterIndex,float x)3687 void SAL_CALL ODatabaseForm::setFloat(sal_Int32 parameterIndex, float x) throw( SQLException, RuntimeException )
3688 {
3689 m_aParameterManager.setFloat(parameterIndex, x);
3690 }
3691
3692 //------------------------------------------------------------------------------
setDouble(sal_Int32 parameterIndex,double x)3693 void SAL_CALL ODatabaseForm::setDouble(sal_Int32 parameterIndex, double x) throw( SQLException, RuntimeException )
3694 {
3695 m_aParameterManager.setDouble(parameterIndex, x);
3696 }
3697
3698 //------------------------------------------------------------------------------
setString(sal_Int32 parameterIndex,const::rtl::OUString & x)3699 void SAL_CALL ODatabaseForm::setString(sal_Int32 parameterIndex, const ::rtl::OUString& x) throw( SQLException, RuntimeException )
3700 {
3701 m_aParameterManager.setString(parameterIndex, x);
3702 }
3703
3704 //------------------------------------------------------------------------------
setBytes(sal_Int32 parameterIndex,const Sequence<sal_Int8> & x)3705 void SAL_CALL ODatabaseForm::setBytes(sal_Int32 parameterIndex, const Sequence< sal_Int8 >& x) throw( SQLException, RuntimeException )
3706 {
3707 m_aParameterManager.setBytes(parameterIndex, x);
3708 }
3709
3710 //------------------------------------------------------------------------------
setDate(sal_Int32 parameterIndex,const::com::sun::star::util::Date & x)3711 void SAL_CALL ODatabaseForm::setDate(sal_Int32 parameterIndex, const ::com::sun::star::util::Date& x) throw( SQLException, RuntimeException )
3712 {
3713 m_aParameterManager.setDate(parameterIndex, x);
3714 }
3715
3716 //------------------------------------------------------------------------------
setTime(sal_Int32 parameterIndex,const::com::sun::star::util::Time & x)3717 void SAL_CALL ODatabaseForm::setTime(sal_Int32 parameterIndex, const ::com::sun::star::util::Time& x) throw( SQLException, RuntimeException )
3718 {
3719 m_aParameterManager.setTime(parameterIndex, x);
3720 }
3721
3722 //------------------------------------------------------------------------------
setTimestamp(sal_Int32 parameterIndex,const::com::sun::star::util::DateTime & x)3723 void SAL_CALL ODatabaseForm::setTimestamp(sal_Int32 parameterIndex, const ::com::sun::star::util::DateTime& x) throw( SQLException, RuntimeException )
3724 {
3725 m_aParameterManager.setTimestamp(parameterIndex, x);
3726 }
3727
3728 //------------------------------------------------------------------------------
setBinaryStream(sal_Int32 parameterIndex,const Reference<XInputStream> & x,sal_Int32 length)3729 void SAL_CALL ODatabaseForm::setBinaryStream(sal_Int32 parameterIndex, const Reference<XInputStream>& x, sal_Int32 length) throw( SQLException, RuntimeException )
3730 {
3731 m_aParameterManager.setBinaryStream(parameterIndex, x, length);
3732 }
3733
3734 //------------------------------------------------------------------------------
setCharacterStream(sal_Int32 parameterIndex,const Reference<XInputStream> & x,sal_Int32 length)3735 void SAL_CALL ODatabaseForm::setCharacterStream(sal_Int32 parameterIndex, const Reference<XInputStream>& x, sal_Int32 length) throw( SQLException, RuntimeException )
3736 {
3737 m_aParameterManager.setCharacterStream(parameterIndex, x, length);
3738 }
3739
3740 //------------------------------------------------------------------------------
setObjectWithInfo(sal_Int32 parameterIndex,const Any & x,sal_Int32 targetSqlType,sal_Int32 scale)3741 void SAL_CALL ODatabaseForm::setObjectWithInfo(sal_Int32 parameterIndex, const Any& x, sal_Int32 targetSqlType, sal_Int32 scale) throw( SQLException, RuntimeException )
3742 {
3743 m_aParameterManager.setObjectWithInfo(parameterIndex, x, targetSqlType, scale);
3744 }
3745
3746 //------------------------------------------------------------------------------
setObject(sal_Int32 parameterIndex,const Any & x)3747 void SAL_CALL ODatabaseForm::setObject(sal_Int32 parameterIndex, const Any& x) throw( SQLException, RuntimeException )
3748 {
3749 m_aParameterManager.setObject(parameterIndex, x);
3750 }
3751
3752 //------------------------------------------------------------------------------
setRef(sal_Int32 parameterIndex,const Reference<XRef> & x)3753 void SAL_CALL ODatabaseForm::setRef(sal_Int32 parameterIndex, const Reference<XRef>& x) throw( SQLException, RuntimeException )
3754 {
3755 m_aParameterManager.setRef(parameterIndex, x);
3756 }
3757
3758 //------------------------------------------------------------------------------
setBlob(sal_Int32 parameterIndex,const Reference<XBlob> & x)3759 void SAL_CALL ODatabaseForm::setBlob(sal_Int32 parameterIndex, const Reference<XBlob>& x) throw( SQLException, RuntimeException )
3760 {
3761 m_aParameterManager.setBlob(parameterIndex, x);
3762 }
3763
3764 //------------------------------------------------------------------------------
setClob(sal_Int32 parameterIndex,const Reference<XClob> & x)3765 void SAL_CALL ODatabaseForm::setClob(sal_Int32 parameterIndex, const Reference<XClob>& x) throw( SQLException, RuntimeException )
3766 {
3767 m_aParameterManager.setClob(parameterIndex, x);
3768 }
3769
3770 //------------------------------------------------------------------------------
setArray(sal_Int32 parameterIndex,const Reference<XArray> & x)3771 void SAL_CALL ODatabaseForm::setArray(sal_Int32 parameterIndex, const Reference<XArray>& x) throw( SQLException, RuntimeException )
3772 {
3773 m_aParameterManager.setArray(parameterIndex, x);
3774 }
3775
3776 //------------------------------------------------------------------------------
clearParameters()3777 void SAL_CALL ODatabaseForm::clearParameters() throw( SQLException, RuntimeException )
3778 {
3779 m_aParameterManager.clearParameters();
3780 }
3781
3782 //------------------------------------------------------------------------------
propertyChange(const PropertyChangeEvent & evt)3783 void SAL_CALL ODatabaseForm::propertyChange( const PropertyChangeEvent& evt ) throw (RuntimeException)
3784 {
3785 if ( evt.Source == m_xParent )
3786 {
3787 if ( evt.PropertyName == PROPERTY_ISNEW )
3788 {
3789 sal_Bool bCurrentIsNew( sal_False );
3790 OSL_VERIFY( evt.NewValue >>= bCurrentIsNew );
3791 if ( !bCurrentIsNew )
3792 reload_impl( sal_True );
3793 }
3794 return;
3795 }
3796 OFormComponents::propertyChange( evt );
3797 }
3798
3799 // com::sun::star::lang::XServiceInfo
3800 //------------------------------------------------------------------------------
getImplementationName_Static()3801 ::rtl::OUString SAL_CALL ODatabaseForm::getImplementationName_Static()
3802 {
3803 return ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.forms.ODatabaseForm" ) );
3804 }
3805
3806 //------------------------------------------------------------------------------
getCompatibleServiceNames_Static()3807 Sequence< ::rtl::OUString > SAL_CALL ODatabaseForm::getCompatibleServiceNames_Static()
3808 {
3809 Sequence< ::rtl::OUString > aServices( 1 );
3810 ::rtl::OUString* pServices = aServices.getArray();
3811
3812 *pServices++ = FRM_COMPONENT_FORM;
3813
3814 return aServices;
3815 }
3816
3817 //------------------------------------------------------------------------------
getCurrentServiceNames_Static()3818 Sequence< ::rtl::OUString > SAL_CALL ODatabaseForm::getCurrentServiceNames_Static()
3819 {
3820 Sequence< ::rtl::OUString > aServices( 5 );
3821 ::rtl::OUString* pServices = aServices.getArray();
3822
3823 *pServices++ = FRM_SUN_FORMCOMPONENT;
3824 *pServices++ = ::rtl::OUString::createFromAscii("com.sun.star.form.FormComponents");
3825 *pServices++ = FRM_SUN_COMPONENT_FORM;
3826 *pServices++ = FRM_SUN_COMPONENT_HTMLFORM;
3827 *pServices++ = FRM_SUN_COMPONENT_DATAFORM;
3828
3829 return aServices;
3830 }
3831
3832 //------------------------------------------------------------------------------
getSupportedServiceNames_Static()3833 Sequence< ::rtl::OUString > SAL_CALL ODatabaseForm::getSupportedServiceNames_Static()
3834 {
3835 return ::comphelper::concatSequences(
3836 getCurrentServiceNames_Static(),
3837 getCompatibleServiceNames_Static()
3838 );
3839 }
3840
3841 //------------------------------------------------------------------------------
getImplementationName()3842 ::rtl::OUString SAL_CALL ODatabaseForm::getImplementationName() throw( RuntimeException )
3843 {
3844 return getImplementationName_Static();
3845 }
3846
3847 //------------------------------------------------------------------------------
getSupportedServiceNames()3848 Sequence< ::rtl::OUString > SAL_CALL ODatabaseForm::getSupportedServiceNames() throw( RuntimeException )
3849 {
3850 // the services of our aggregate
3851 Sequence< ::rtl::OUString > aServices;
3852 Reference< XServiceInfo > xInfo;
3853 if (query_aggregation(m_xAggregate, xInfo))
3854 aServices = xInfo->getSupportedServiceNames();
3855
3856 // concat with out own services
3857 return ::comphelper::concatSequences(
3858 getCurrentServiceNames_Static(),
3859 aServices
3860 );
3861 // use getCurrentXXX instead of getSupportedXXX, because at runtime, we do not want to have
3862 // the compatible names
3863 // This is maily to be consistent with the implementation before fixing #97083#, though the
3864 // better solution _may_ be to return the compatible names at runtime, too
3865 // 04.03.2002 - fs@openoffice.org
3866 }
3867
3868 //------------------------------------------------------------------------------
supportsService(const::rtl::OUString & ServiceName)3869 sal_Bool SAL_CALL ODatabaseForm::supportsService(const ::rtl::OUString& ServiceName) throw( RuntimeException )
3870 {
3871 Sequence< ::rtl::OUString > aSupported( getSupportedServiceNames() );
3872 const ::rtl::OUString* pArray = aSupported.getConstArray();
3873 for( sal_Int32 i = 0; i < aSupported.getLength(); ++i, ++pArray )
3874 if( pArray->equals( ServiceName ) )
3875 return sal_True;
3876 return sal_False;
3877 }
3878
3879 //==============================================================================
3880 // com::sun::star::io::XPersistObject
3881 //------------------------------------------------------------------------------
3882
3883 const sal_uInt16 CYCLE = 0x0001;
3884 const sal_uInt16 DONTAPPLYFILTER = 0x0002;
3885
3886 //------------------------------------------------------------------------------
getServiceName()3887 ::rtl::OUString ODatabaseForm::getServiceName() throw( RuntimeException )
3888 {
3889 return FRM_COMPONENT_FORM; // old (non-sun) name for compatibility !
3890 }
3891
3892 //------------------------------------------------------------------------------
write(const Reference<XObjectOutputStream> & _rxOutStream)3893 void SAL_CALL ODatabaseForm::write(const Reference<XObjectOutputStream>& _rxOutStream) throw( IOException, RuntimeException )
3894 {
3895 DBG_ASSERT(m_xAggregateSet.is(), "ODatabaseForm::write : only to be called if the aggregate exists !");
3896
3897 // all children
3898 OFormComponents::write(_rxOutStream);
3899
3900 // version
3901 _rxOutStream->writeShort(0x0003);
3902
3903 // Name
3904 _rxOutStream << m_sName;
3905
3906 ::rtl::OUString sDataSource;
3907 if (m_xAggregateSet.is())
3908 m_xAggregateSet->getPropertyValue(PROPERTY_DATASOURCE) >>= sDataSource;
3909 _rxOutStream << sDataSource;
3910
3911 // former CursorSource
3912 ::rtl::OUString sCommand;
3913 if (m_xAggregateSet.is())
3914 m_xAggregateSet->getPropertyValue(PROPERTY_COMMAND) >>= sCommand;
3915 _rxOutStream << sCommand;
3916
3917 // former MasterFields
3918 _rxOutStream << m_aMasterFields;
3919 // former DetailFields
3920 _rxOutStream << m_aDetailFields;
3921
3922 // former DataSelectionType
3923 DataSelectionType eTranslated = DataSelectionType_TABLE;
3924 if (m_xAggregateSet.is())
3925 {
3926 sal_Int32 nCommandType = 0;
3927 m_xAggregateSet->getPropertyValue(PROPERTY_COMMANDTYPE) >>= nCommandType;
3928 switch (nCommandType)
3929 {
3930 case CommandType::TABLE : eTranslated = DataSelectionType_TABLE; break;
3931 case CommandType::QUERY : eTranslated = DataSelectionType_QUERY; break;
3932 case CommandType::COMMAND:
3933 {
3934 sal_Bool bEscapeProcessing = getBOOL(m_xAggregateSet->getPropertyValue(PROPERTY_ESCAPE_PROCESSING));
3935 eTranslated = bEscapeProcessing ? DataSelectionType_SQL : DataSelectionType_SQLPASSTHROUGH;
3936 }
3937 break;
3938 default : DBG_ERROR("ODatabaseForm::write : wrong CommandType !");
3939 }
3940 }
3941 _rxOutStream->writeShort((sal_Int16)eTranslated); // former DataSelectionType
3942
3943 // very old versions expect a CursorType here
3944 _rxOutStream->writeShort(DatabaseCursorType_KEYSET);
3945
3946 _rxOutStream->writeBoolean(m_eNavigation != NavigationBarMode_NONE);
3947
3948 // former DataEntry
3949 if (m_xAggregateSet.is())
3950 _rxOutStream->writeBoolean(getBOOL(m_xAggregateSet->getPropertyValue(PROPERTY_INSERTONLY)));
3951 else
3952 _rxOutStream->writeBoolean(sal_False);
3953
3954 _rxOutStream->writeBoolean(m_bAllowInsert);
3955 _rxOutStream->writeBoolean(m_bAllowUpdate);
3956 _rxOutStream->writeBoolean(m_bAllowDelete);
3957
3958 // html form stuff
3959 ::rtl::OUString sTmp = INetURLObject::decode( m_aTargetURL, '%', INetURLObject::DECODE_UNAMBIGUOUS);
3960 _rxOutStream << sTmp;
3961 _rxOutStream->writeShort( (sal_Int16)m_eSubmitMethod );
3962 _rxOutStream->writeShort( (sal_Int16)m_eSubmitEncoding );
3963 _rxOutStream << m_aTargetFrame;
3964
3965 // version 2 didn't know some options and the "default" state
3966 sal_Int32 nCycle = TabulatorCycle_RECORDS;
3967 if (m_aCycle.hasValue())
3968 {
3969 ::cppu::enum2int(nCycle, m_aCycle);
3970 if (m_aCycle == TabulatorCycle_PAGE)
3971 // unknown in earlier versions
3972 nCycle = TabulatorCycle_RECORDS;
3973 }
3974 _rxOutStream->writeShort((sal_Int16) nCycle);
3975
3976 _rxOutStream->writeShort((sal_Int16)m_eNavigation);
3977
3978 ::rtl::OUString sFilter;
3979 ::rtl::OUString sOrder;
3980 if (m_xAggregateSet.is())
3981 {
3982 m_xAggregateSet->getPropertyValue(PROPERTY_FILTER) >>= sFilter;
3983 m_xAggregateSet->getPropertyValue(PROPERTY_SORT) >>= sOrder;
3984 }
3985 _rxOutStream << sFilter;
3986 _rxOutStream << sOrder;
3987
3988
3989 // version 3
3990 sal_uInt16 nAnyMask = 0;
3991 if (m_aCycle.hasValue())
3992 nAnyMask |= CYCLE;
3993
3994 if (m_xAggregateSet.is() && !getBOOL(m_xAggregateSet->getPropertyValue(PROPERTY_APPLYFILTER)))
3995 nAnyMask |= DONTAPPLYFILTER;
3996
3997 _rxOutStream->writeShort(nAnyMask);
3998
3999 if (nAnyMask & CYCLE)
4000 {
4001 sal_Int32 nRealCycle = 0;
4002 ::cppu::enum2int(nRealCycle, m_aCycle);
4003 _rxOutStream->writeShort((sal_Int16)nRealCycle);
4004 }
4005 }
4006
4007 //------------------------------------------------------------------------------
read(const Reference<XObjectInputStream> & _rxInStream)4008 void SAL_CALL ODatabaseForm::read(const Reference<XObjectInputStream>& _rxInStream) throw( IOException, RuntimeException )
4009 {
4010 DBG_ASSERT(m_xAggregateSet.is(), "ODatabaseForm::read : only to be called if the aggregate exists !");
4011
4012 OFormComponents::read(_rxInStream);
4013
4014 // version
4015 sal_uInt16 nVersion = _rxInStream->readShort();
4016
4017 _rxInStream >> m_sName;
4018
4019 ::rtl::OUString sAggregateProp;
4020 _rxInStream >> sAggregateProp;
4021 if (m_xAggregateSet.is())
4022 m_xAggregateSet->setPropertyValue(PROPERTY_DATASOURCE, makeAny(sAggregateProp));
4023 _rxInStream >> sAggregateProp;
4024 if (m_xAggregateSet.is())
4025 m_xAggregateSet->setPropertyValue(PROPERTY_COMMAND, makeAny(sAggregateProp));
4026
4027 _rxInStream >> m_aMasterFields;
4028 _rxInStream >> m_aDetailFields;
4029
4030 sal_Int16 nCursorSourceType = _rxInStream->readShort();
4031 sal_Int32 nCommandType = 0;
4032 switch ((DataSelectionType)nCursorSourceType)
4033 {
4034 case DataSelectionType_TABLE : nCommandType = CommandType::TABLE; break;
4035 case DataSelectionType_QUERY : nCommandType = CommandType::QUERY; break;
4036 case DataSelectionType_SQL:
4037 case DataSelectionType_SQLPASSTHROUGH:
4038 {
4039 nCommandType = CommandType::COMMAND;
4040 sal_Bool bEscapeProcessing = ((DataSelectionType)nCursorSourceType) != DataSelectionType_SQLPASSTHROUGH;
4041 m_xAggregateSet->setPropertyValue(PROPERTY_ESCAPE_PROCESSING, makeAny((sal_Bool)bEscapeProcessing));
4042 }
4043 break;
4044 default : DBG_ERROR("ODatabaseForm::read : wrong CommandType !");
4045 }
4046 if (m_xAggregateSet.is())
4047 m_xAggregateSet->setPropertyValue(PROPERTY_COMMANDTYPE, makeAny(nCommandType));
4048
4049 // obsolete
4050 _rxInStream->readShort();
4051
4052 // navigation mode was a boolean in version 1
4053 // war in der version 1 ein sal_Bool
4054 sal_Bool bNavigation = _rxInStream->readBoolean();
4055 if (nVersion == 1)
4056 m_eNavigation = bNavigation ? NavigationBarMode_CURRENT : NavigationBarMode_NONE;
4057
4058 sal_Bool bInsertOnly = _rxInStream->readBoolean();
4059 if (m_xAggregateSet.is())
4060 m_xAggregateSet->setPropertyValue(PROPERTY_INSERTONLY, makeAny(bInsertOnly));
4061
4062 m_bAllowInsert = _rxInStream->readBoolean();
4063 m_bAllowUpdate = _rxInStream->readBoolean();
4064 m_bAllowDelete = _rxInStream->readBoolean();
4065
4066 // html stuff
4067 ::rtl::OUString sTmp;
4068 _rxInStream >> sTmp;
4069 m_aTargetURL = INetURLObject::decode( sTmp, '%', INetURLObject::DECODE_UNAMBIGUOUS);
4070 m_eSubmitMethod = (FormSubmitMethod)_rxInStream->readShort();
4071 m_eSubmitEncoding = (FormSubmitEncoding)_rxInStream->readShort();
4072 _rxInStream >> m_aTargetFrame;
4073
4074 if (nVersion > 1)
4075 {
4076 sal_Int32 nCycle = _rxInStream->readShort();
4077 m_aCycle = ::cppu::int2enum(nCycle, ::getCppuType(static_cast<const TabulatorCycle*>(NULL)));
4078 m_eNavigation = (NavigationBarMode)_rxInStream->readShort();
4079
4080 _rxInStream >> sAggregateProp;
4081 setPropertyValue(PROPERTY_FILTER, makeAny(sAggregateProp));
4082
4083 _rxInStream >> sAggregateProp;
4084 if (m_xAggregateSet.is())
4085 m_xAggregateSet->setPropertyValue(PROPERTY_SORT, makeAny(sAggregateProp));
4086 }
4087
4088 sal_uInt16 nAnyMask = 0;
4089 if (nVersion > 2)
4090 {
4091 nAnyMask = _rxInStream->readShort();
4092 if (nAnyMask & CYCLE)
4093 {
4094 sal_Int32 nCycle = _rxInStream->readShort();
4095 m_aCycle = ::cppu::int2enum(nCycle, ::getCppuType(static_cast<const TabulatorCycle*>(NULL)));
4096 }
4097 else
4098 m_aCycle.clear();
4099 }
4100 if (m_xAggregateSet.is())
4101 m_xAggregateSet->setPropertyValue(PROPERTY_APPLYFILTER, makeAny((sal_Bool)((nAnyMask & DONTAPPLYFILTER) == 0)));
4102 }
4103
4104 //------------------------------------------------------------------------------
implInserted(const ElementDescription * _pElement)4105 void ODatabaseForm::implInserted( const ElementDescription* _pElement )
4106 {
4107 OFormComponents::implInserted( _pElement );
4108
4109 Reference< XSQLErrorBroadcaster > xBroadcaster( _pElement->xInterface, UNO_QUERY );
4110 Reference< XForm > xForm ( _pElement->xInterface, UNO_QUERY );
4111
4112 if ( xBroadcaster.is() && !xForm.is() )
4113 { // the object is an error broadcaster, but no form itself -> add ourself as listener
4114 xBroadcaster->addSQLErrorListener( this );
4115 }
4116 }
4117
4118 //------------------------------------------------------------------------------
implRemoved(const InterfaceRef & _rxObject)4119 void ODatabaseForm::implRemoved(const InterfaceRef& _rxObject)
4120 {
4121 OFormComponents::implRemoved( _rxObject );
4122
4123 Reference<XSQLErrorBroadcaster> xBroadcaster(_rxObject, UNO_QUERY);
4124 Reference<XForm> xForm(_rxObject, UNO_QUERY);
4125 if (xBroadcaster.is() && !xForm.is())
4126 { // the object is an error broadcaster, but no form itself -> remove ourself as listener
4127 xBroadcaster->removeSQLErrorListener(this);
4128 }
4129 }
4130
4131 //------------------------------------------------------------------------------
errorOccured(const SQLErrorEvent & _rEvent)4132 void SAL_CALL ODatabaseForm::errorOccured(const SQLErrorEvent& _rEvent) throw( RuntimeException )
4133 {
4134 // give it to my own error listener
4135 onError(_rEvent);
4136 // TODO : think about extending the chain with an SQLContext object saying
4137 // "this was an error of one of my children"
4138 }
4139
4140 // com::sun::star::container::XNamed
4141 //------------------------------------------------------------------------------
getName()4142 ::rtl::OUString SAL_CALL ODatabaseForm::getName() throw( RuntimeException )
4143 {
4144 ::rtl::OUString sReturn;
4145 OPropertySetHelper::getFastPropertyValue(PROPERTY_ID_NAME) >>= sReturn;
4146 return sReturn;
4147 }
4148
4149 //------------------------------------------------------------------------------
setName(const::rtl::OUString & aName)4150 void SAL_CALL ODatabaseForm::setName(const ::rtl::OUString& aName) throw( RuntimeException )
4151 {
4152 setFastPropertyValue(PROPERTY_ID_NAME, makeAny(aName));
4153 }
4154
4155 //.........................................................................
4156 } // namespace frm
4157 //.........................................................................
4158
4159