1 /*************************************************************************
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5 * Copyright 2008 by Sun Microsystems, Inc.
7 * OpenOffice.org - a multi-platform office productivity suite
9 * $RCSfile: propertyhandler.cxx,v $
12 * This file is part of OpenOffice.org.
14 * OpenOffice.org is free software: you can redistribute it and/or modify
15 * it under the terms of the GNU Lesser General Public License version 3
16 * only, as published by the Free Software Foundation.
18 * OpenOffice.org is distributed in the hope that it will be useful,
19 * but WITHOUT ANY WARRANTY; without even the implied warranty of
20 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 * GNU Lesser General Public License version 3 for more details
22 * (a copy is included in the LICENSE file that accompanied this code).
24 * You should have received a copy of the GNU Lesser General Public License
25 * version 3 along with OpenOffice.org. If not, see
26 * <http://www.openoffice.org/license.html>
27 * for a copy of the LGPLv3 License.
29 ************************************************************************/
31 // MARKER(update_precomp.py): autogen include statement, do not remove
32 #include "precompiled_extensions.hxx"
33 #include "propertyhandler.hxx"
34 #include "formmetadata.hxx"
35 #include "formbrowsertools.hxx"
36 #include "handlerhelper.hxx"
37 #include "formstrings.hxx"
39 /** === begin UNO includes === **/
40 #include <com/sun/star/beans/PropertyAttribute.hpp>
41 #include <com/sun/star/lang/NullPointerException.hpp>
42 #include <com/sun/star/util/XModifiable.hpp>
43 /** === end UNO includes === **/
45 #include <tools/debug.hxx>
46 #include <unotools/confignode.hxx>
47 #include <unotools/localedatawrapper.hxx>
48 #include <svtools/syslocale.hxx>
49 #include <toolkit/helper/vclunohelper.hxx>
53 //........................................................................
56 //........................................................................
58 using namespace ::com::sun::star::uno
;
59 using namespace ::com::sun::star::awt
;
60 using namespace ::com::sun::star::beans
;
61 using namespace ::com::sun::star::script
;
62 using namespace ::com::sun::star::lang
;
63 using namespace ::com::sun::star::util
;
64 using namespace ::com::sun::star::frame
;
65 using namespace ::com::sun::star::inspection
;
66 using namespace ::comphelper
;
68 //====================================================================
70 //====================================================================
71 DBG_NAME( PropertyHandler
)
72 //--------------------------------------------------------------------
73 PropertyHandler::PropertyHandler( const Reference
< XComponentContext
>& _rxContext
)
74 :PropertyHandler_Base( m_aMutex
)
75 ,m_bSupportedPropertiesAreKnown( false )
76 ,m_aPropertyListeners( m_aMutex
)
77 ,m_aContext( _rxContext
)
78 ,m_pInfoService ( new OPropertyInfoService
)
80 DBG_CTOR( PropertyHandler
, NULL
);
82 m_xTypeConverter
= Reference
< XTypeConverter
>(
83 m_aContext
.createComponent( "com.sun.star.script.Converter" ),
88 //--------------------------------------------------------------------
89 PropertyHandler::~PropertyHandler()
91 DBG_DTOR( PropertyHandler
, NULL
);
94 //--------------------------------------------------------------------
95 void SAL_CALL
PropertyHandler::inspect( const Reference
< XInterface
>& _rxIntrospectee
) throw (RuntimeException
, NullPointerException
)
97 if ( !_rxIntrospectee
.is() )
98 throw NullPointerException();
100 ::osl::MutexGuard
aGuard( m_aMutex
);
102 Reference
< XPropertySet
> xNewComponent( _rxIntrospectee
, UNO_QUERY
);
103 if ( xNewComponent
== m_xComponent
)
106 // remove all old property change listeners
107 ::std::auto_ptr
< ::cppu::OInterfaceIteratorHelper
> removeListener
= m_aPropertyListeners
.createIterator();
108 ::std::auto_ptr
< ::cppu::OInterfaceIteratorHelper
> readdListener
= m_aPropertyListeners
.createIterator(); // will copy the container as needed
109 while ( removeListener
->hasMoreElements() )
110 removePropertyChangeListener( static_cast< XPropertyChangeListener
* >( removeListener
->next() ) );
111 OSL_ENSURE( m_aPropertyListeners
.empty(), "PropertyHandler::inspect: derived classes are expected to forward the removePropertyChangeListener call to their base class (me)!" );
113 // remember the new component, and give derived classes the chance to react on it
114 m_xComponent
= xNewComponent
;
117 // add the listeners, again
118 while ( readdListener
->hasMoreElements() )
119 addPropertyChangeListener( static_cast< XPropertyChangeListener
* >( readdListener
->next() ) );
122 //--------------------------------------------------------------------
123 void PropertyHandler::onNewComponent()
125 if ( m_xComponent
.is() )
126 m_xComponentPropertyInfo
= m_xComponent
->getPropertySetInfo();
128 m_xComponentPropertyInfo
.clear();
130 m_bSupportedPropertiesAreKnown
= false;
131 m_aSupportedProperties
.realloc( 0 );
134 //--------------------------------------------------------------------
135 Sequence
< Property
> SAL_CALL
PropertyHandler::getSupportedProperties() throw (RuntimeException
)
137 ::osl::MutexGuard
aGuard( m_aMutex
);
138 if ( !m_bSupportedPropertiesAreKnown
)
140 m_aSupportedProperties
= doDescribeSupportedProperties();
141 m_bSupportedPropertiesAreKnown
= true;
143 return (Sequence
< Property
>)m_aSupportedProperties
;
146 //--------------------------------------------------------------------
147 Sequence
< ::rtl::OUString
> SAL_CALL
PropertyHandler::getSupersededProperties( ) throw (RuntimeException
)
149 return Sequence
< ::rtl::OUString
>();
152 //--------------------------------------------------------------------
153 Sequence
< ::rtl::OUString
> SAL_CALL
PropertyHandler::getActuatingProperties( ) throw (RuntimeException
)
155 return Sequence
< ::rtl::OUString
>();
158 //--------------------------------------------------------------------
159 Any SAL_CALL
PropertyHandler::convertToPropertyValue( const ::rtl::OUString
& _rPropertyName
, const Any
& _rControlValue
) throw (UnknownPropertyException
, RuntimeException
)
161 ::osl::MutexGuard
aGuard( m_aMutex
);
162 PropertyId nPropId
= m_pInfoService
->getPropertyId( _rPropertyName
);
163 Property
aProperty( impl_getPropertyFromName_throw( _rPropertyName
) );
166 if ( !_rControlValue
.hasValue() )
167 // NULL is converted to NULL
168 return aPropertyValue
;
170 if ( ( m_pInfoService
->getPropertyUIFlags( nPropId
) & PROP_FLAG_ENUM
) != 0 )
172 ::rtl::OUString sControlValue
;
173 OSL_VERIFY( _rControlValue
>>= sControlValue
);
174 ::rtl::Reference
< IPropertyEnumRepresentation
> aEnumConversion(
175 new DefaultEnumRepresentation( *m_pInfoService
, aProperty
.Type
, nPropId
) );
176 // TODO/UNOize: cache those converters?
177 aEnumConversion
->getValueFromDescription( sControlValue
, aPropertyValue
);
180 aPropertyValue
= PropertyHandlerHelper::convertToPropertyValue(
181 m_aContext
.getContext(),m_xTypeConverter
, aProperty
, _rControlValue
);
182 return aPropertyValue
;
185 //--------------------------------------------------------------------
186 Any SAL_CALL
PropertyHandler::convertToControlValue( const ::rtl::OUString
& _rPropertyName
, const Any
& _rPropertyValue
, const Type
& _rControlValueType
) throw (UnknownPropertyException
, RuntimeException
)
188 ::osl::MutexGuard
aGuard( m_aMutex
);
189 PropertyId nPropId
= m_pInfoService
->getPropertyId( _rPropertyName
);
191 if ( ( m_pInfoService
->getPropertyUIFlags( nPropId
) & PROP_FLAG_ENUM
) != 0 )
193 DBG_ASSERT( _rControlValueType
.getTypeClass() == TypeClass_STRING
, "PropertyHandler::convertToControlValue: ENUM, but not STRING?" );
195 ::rtl::Reference
< IPropertyEnumRepresentation
> aEnumConversion(
196 new DefaultEnumRepresentation( *m_pInfoService
, _rPropertyValue
.getValueType(), nPropId
) );
197 // TODO/UNOize: cache those converters?
198 return makeAny( aEnumConversion
->getDescriptionForValue( _rPropertyValue
) );
201 return PropertyHandlerHelper::convertToControlValue(
202 m_aContext
.getContext(),m_xTypeConverter
, _rPropertyValue
, _rControlValueType
);
205 //--------------------------------------------------------------------
206 PropertyState SAL_CALL
PropertyHandler::getPropertyState( const ::rtl::OUString
& /*_rPropertyName*/ ) throw (UnknownPropertyException
, RuntimeException
)
208 return PropertyState_DIRECT_VALUE
;
211 //--------------------------------------------------------------------
212 LineDescriptor SAL_CALL
PropertyHandler::describePropertyLine( const ::rtl::OUString
& _rPropertyName
,
213 const Reference
< XPropertyControlFactory
>& _rxControlFactory
)
214 throw (UnknownPropertyException
, NullPointerException
, RuntimeException
)
216 if ( !_rxControlFactory
.is() )
217 throw NullPointerException();
219 ::osl::MutexGuard
aGuard( m_aMutex
);
220 PropertyId
nPropId( impl_getPropertyId_throw( _rPropertyName
) );
221 const Property
& rProperty( impl_getPropertyFromId_throw( nPropId
) );
223 LineDescriptor aDescriptor
;
224 if ( ( m_pInfoService
->getPropertyUIFlags( nPropId
) & PROP_FLAG_ENUM
) != 0 )
226 aDescriptor
.Control
= PropertyHandlerHelper::createListBoxControl(
227 _rxControlFactory
, m_pInfoService
->getPropertyEnumRepresentations( nPropId
),
228 PropertyHandlerHelper::requiresReadOnlyControl( rProperty
.Attributes
), sal_False
);
231 PropertyHandlerHelper::describePropertyLine( rProperty
, aDescriptor
, _rxControlFactory
);
233 aDescriptor
.HelpURL
= HelpIdUrl::getHelpURL( m_pInfoService
->getPropertyHelpId( nPropId
) );
234 aDescriptor
.DisplayName
= m_pInfoService
->getPropertyTranslation( nPropId
);
236 if ( ( m_pInfoService
->getPropertyUIFlags( nPropId
) & PROP_FLAG_DATA_PROPERTY
) != 0 )
237 aDescriptor
.Category
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Data" ) );
239 aDescriptor
.Category
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "General" ) );
243 //--------------------------------------------------------------------
244 ::sal_Bool SAL_CALL
PropertyHandler::isComposable( const ::rtl::OUString
& _rPropertyName
) throw (UnknownPropertyException
, RuntimeException
)
246 ::osl::MutexGuard
aGuard( m_aMutex
);
247 return m_pInfoService
->isComposeable( _rPropertyName
);
250 //--------------------------------------------------------------------
251 InteractiveSelectionResult SAL_CALL
PropertyHandler::onInteractivePropertySelection( const ::rtl::OUString
& /*_rPropertyName*/, sal_Bool
/*_bPrimary*/, Any
& /*_rData*/, const Reference
< XObjectInspectorUI
>& /*_rxInspectorUI*/ ) throw (UnknownPropertyException
, NullPointerException
, RuntimeException
)
253 DBG_ERROR( "PropertyHandler::onInteractivePropertySelection: not implemented!" );
254 return InteractiveSelectionResult_Cancelled
;
257 //--------------------------------------------------------------------
258 void SAL_CALL
PropertyHandler::actuatingPropertyChanged( const ::rtl::OUString
& /*_rActuatingPropertyName*/, const Any
& /*_rNewValue*/, const Any
& /*_rOldValue*/, const Reference
< XObjectInspectorUI
>& /*_rxInspectorUI*/, sal_Bool
/*_bFirstTimeInit*/ ) throw (NullPointerException
, RuntimeException
)
260 DBG_ERROR( "PropertyHandler::actuatingPropertyChanged: not implemented!" );
263 //--------------------------------------------------------------------
264 void SAL_CALL
PropertyHandler::addPropertyChangeListener( const Reference
< XPropertyChangeListener
>& _rxListener
) throw (RuntimeException
)
266 ::osl::MutexGuard
aGuard( m_aMutex
);
267 if ( !_rxListener
.is() )
268 throw NullPointerException();
269 m_aPropertyListeners
.addListener( _rxListener
);
272 //--------------------------------------------------------------------
273 void SAL_CALL
PropertyHandler::removePropertyChangeListener( const Reference
< XPropertyChangeListener
>& _rxListener
) throw (RuntimeException
)
275 ::osl::MutexGuard
aGuard( m_aMutex
);
276 m_aPropertyListeners
.removeListener( _rxListener
);
279 //--------------------------------------------------------------------
280 sal_Bool SAL_CALL
PropertyHandler::suspend( sal_Bool
/*_bSuspend*/ ) throw (RuntimeException
)
285 //--------------------------------------------------------------------
286 IMPLEMENT_FORWARD_XCOMPONENT( PropertyHandler
, PropertyHandler_Base
)
287 //--------------------------------------------------------------------
288 void SAL_CALL
PropertyHandler::disposing()
290 m_xComponent
.clear();
291 m_aPropertyListeners
.clear();
292 m_xTypeConverter
.clear();
293 m_aSupportedProperties
.realloc( 0 );
296 //--------------------------------------------------------------------
297 void PropertyHandler::firePropertyChange( const ::rtl::OUString
& _rPropName
, PropertyId _nPropId
, const Any
& _rOldValue
, const Any
& _rNewValue
) SAL_THROW(())
299 PropertyChangeEvent aEvent
;
300 aEvent
.Source
= m_xComponent
;
301 aEvent
.PropertyHandle
= _nPropId
;
302 aEvent
.PropertyName
= _rPropName
;
303 aEvent
.OldValue
= _rOldValue
;
304 aEvent
.NewValue
= _rNewValue
;
305 m_aPropertyListeners
.notify( aEvent
, &XPropertyChangeListener::propertyChange
);
308 //--------------------------------------------------------------------
309 const Property
* PropertyHandler::impl_getPropertyFromId_nothrow( PropertyId _nPropId
) const
311 const_cast< PropertyHandler
* >( this )->getSupportedProperties();
312 const Property
* pFound
= ::std::find_if( m_aSupportedProperties
.begin(), m_aSupportedProperties
.end(),
313 FindPropertyByHandle( _nPropId
)
315 if ( pFound
!= m_aSupportedProperties
.end() )
320 //--------------------------------------------------------------------
321 const Property
& PropertyHandler::impl_getPropertyFromId_throw( PropertyId _nPropId
) const
323 const Property
* pProperty
= impl_getPropertyFromId_nothrow( _nPropId
);
325 throw UnknownPropertyException();
330 //--------------------------------------------------------------------
331 const Property
& PropertyHandler::impl_getPropertyFromName_throw( const ::rtl::OUString
& _rPropertyName
) const
333 const_cast< PropertyHandler
* >( this )->getSupportedProperties();
334 StlSyntaxSequence
< Property
>::const_iterator pFound
= ::std::find_if( m_aSupportedProperties
.begin(), m_aSupportedProperties
.end(),
335 FindPropertyByName( _rPropertyName
)
337 if ( pFound
== m_aSupportedProperties
.end() )
338 throw UnknownPropertyException();
343 //--------------------------------------------------------------------
344 void PropertyHandler::implAddPropertyDescription( ::std::vector
< Property
>& _rProperties
, const ::rtl::OUString
& _rPropertyName
, const Type
& _rType
, sal_Int16 _nAttribs
) const
346 _rProperties
.push_back( Property(
348 m_pInfoService
->getPropertyId( _rPropertyName
),
354 //------------------------------------------------------------------------
355 Window
* PropertyHandler::impl_getDefaultDialogParent_nothrow() const
357 return PropertyHandlerHelper::getDialogParentWindow( m_aContext
);
360 //------------------------------------------------------------------------
361 PropertyId
PropertyHandler::impl_getPropertyId_throw( const ::rtl::OUString
& _rPropertyName
) const
363 PropertyId nPropId
= m_pInfoService
->getPropertyId( _rPropertyName
);
365 throw UnknownPropertyException();
369 //------------------------------------------------------------------------
370 void PropertyHandler::impl_setContextDocumentModified_nothrow() const
372 Reference
< XModifiable
> xModifiable( impl_getContextDocument_nothrow(), UNO_QUERY
);
373 if ( xModifiable
.is() )
374 xModifiable
->setModified( sal_True
);
377 //------------------------------------------------------------------------
378 bool PropertyHandler::impl_componentHasProperty_throw( const ::rtl::OUString
& _rPropName
) const
380 return m_xComponentPropertyInfo
.is() && m_xComponentPropertyInfo
->hasPropertyByName( _rPropName
);
383 //--------------------------------------------------------------------
384 sal_Int16
PropertyHandler::impl_getDocumentMeasurementUnit_throw() const
386 FieldUnit eUnit
= FUNIT_NONE
;
388 Reference
< XServiceInfo
> xDocumentSI( impl_getContextDocument_nothrow(), UNO_QUERY
);
389 OSL_ENSURE( xDocumentSI
.is(), "PropertyHandlerHelper::impl_getDocumentMeasurementUnit_throw: No context document - where do I live?" );
390 if ( xDocumentSI
.is() )
392 // determine the application type we live in
393 ::rtl::OUString sConfigurationLocation
;
394 ::rtl::OUString sConfigurationProperty
;
395 if ( xDocumentSI
->supportsService( SERVICE_WEB_DOCUMENT
) )
397 sConfigurationLocation
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "/org.openoffice.Office.WriterWeb/Layout/Other" ) );
398 sConfigurationProperty
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "MeasureUnit" ) );
400 else if ( xDocumentSI
->supportsService( SERVICE_TEXT_DOCUMENT
) )
402 sConfigurationLocation
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "/org.openoffice.Office.Writer/Layout/Other" ) );
403 sConfigurationProperty
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "MeasureUnit" ) );
405 else if ( xDocumentSI
->supportsService( SERVICE_SPREADSHEET_DOCUMENT
) )
407 sConfigurationLocation
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "/org.openoffice.Office.Calc/Layout/Other/MeasureUnit" ) );
408 sConfigurationProperty
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Metric" ) );
410 else if ( xDocumentSI
->supportsService( SERVICE_DRAWING_DOCUMENT
) )
412 sConfigurationLocation
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "/org.openoffice.Office.Draw/Layout/Other/MeasureUnit" ) );
413 sConfigurationProperty
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Metric" ) );
415 else if ( xDocumentSI
->supportsService( SERVICE_PRESENTATION_DOCUMENT
) )
417 sConfigurationLocation
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "/org.openoffice.Office.Impress/Layout/Other/MeasureUnit" ) );
418 sConfigurationProperty
= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Metric" ) );
421 // read the measurement unit from the configuration
422 if ( sConfigurationLocation
.getLength() && sConfigurationProperty
.getLength() )
424 ::utl::OConfigurationTreeRoot
aConfigTree( ::utl::OConfigurationTreeRoot::createWithServiceFactory(
425 m_aContext
.getLegacyServiceFactory(), sConfigurationLocation
, -1, ::utl::OConfigurationTreeRoot::CM_READONLY
) );
426 sal_Int32 nUnitAsInt
= (sal_Int32
)FUNIT_NONE
;
427 aConfigTree
.getNodeValue( sConfigurationProperty
) >>= nUnitAsInt
;
429 // if this denotes a valid (and accepted) unit, then use it
430 if ( ( nUnitAsInt
> FUNIT_NONE
) && ( nUnitAsInt
<= FUNIT_100TH_MM
) )
431 eUnit
= static_cast< FieldUnit
>( nUnitAsInt
);
435 if ( FUNIT_NONE
== eUnit
)
437 MeasurementSystem eSystem
= SvtSysLocale().GetLocaleData().getMeasurementSystemEnum();
438 eUnit
= MEASURE_METRIC
== eSystem
? FUNIT_CM
: FUNIT_INCH
;
441 return VCLUnoHelper::ConvertToMeasurementUnit( eUnit
, 1 );
444 //====================================================================
445 //= PropertyHandlerComponent
446 //====================================================================
447 //------------------------------------------------------------------------
448 PropertyHandlerComponent::PropertyHandlerComponent( const Reference
< XComponentContext
>& _rxContext
)
449 :PropertyHandler( _rxContext
)
453 //--------------------------------------------------------------------
454 IMPLEMENT_FORWARD_XINTERFACE2( PropertyHandlerComponent
, PropertyHandler
, PropertyHandlerComponent_Base
)
455 IMPLEMENT_FORWARD_XTYPEPROVIDER2( PropertyHandlerComponent
, PropertyHandler
, PropertyHandlerComponent_Base
)
457 //--------------------------------------------------------------------
458 ::sal_Bool SAL_CALL
PropertyHandlerComponent::supportsService( const ::rtl::OUString
& ServiceName
) throw (RuntimeException
)
460 StlSyntaxSequence
< ::rtl::OUString
> aAllServices( getSupportedServiceNames() );
461 return ::std::find( aAllServices
.begin(), aAllServices
.end(), ServiceName
) != aAllServices
.end();
464 //........................................................................
466 //........................................................................