/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include "controltype.hxx" #include "modulepcr.hxx" #include #include #include "fontdialog.hxx" #include "formcomponenthandler.hxx" #include "formlinkdialog.hxx" #include "formmetadata.hxx" #include #include #include #include "formstrings.hxx" #include "handlerhelper.hxx" #include "listselectiondlg.hxx" #include "pcrcommon.hxx" #include "selectlabeldialog.hxx" #include "standardcontrol.hxx" #include "taborder.hxx" #include "usercontrol.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace pcr { using namespace ::com::sun::star; using namespace uno; using namespace lang; using namespace beans; using namespace frame; using namespace form; using namespace util; using namespace awt; using namespace sdb; using namespace sdbc; using namespace sdbcx; using namespace report; using namespace container; using namespace ui::dialogs; using namespace inspection; using namespace ::dbtools; namespace WritingMode2 = ::com::sun::star::text::WritingMode2; //= FormComponentPropertyHandler #define PROPERTY_ID_ROWSET 1 FormComponentPropertyHandler::FormComponentPropertyHandler( const Reference< XComponentContext >& _rxContext ) :PropertyHandlerComponent( _rxContext ) ,::comphelper::OPropertyContainer(PropertyHandlerComponent::rBHelper) ,m_sDefaultValueString( PcrRes(RID_STR_STANDARD) ) ,m_eComponentClass( eUnknown ) ,m_bComponentIsSubForm( false ) ,m_bHaveListSource( false ) ,m_bHaveCommand( false ) ,m_nClassId( 0 ) { registerProperty(PROPERTY_ROWSET,PROPERTY_ID_ROWSET,0,&m_xRowSet,cppu::UnoType::get()); } FormComponentPropertyHandler::~FormComponentPropertyHandler() { } IMPLEMENT_FORWARD_XINTERFACE2(FormComponentPropertyHandler,PropertyHandlerComponent,::comphelper::OPropertyContainer) OUString FormComponentPropertyHandler::getImplementationName( ) { return u"com.sun.star.comp.extensions.FormComponentPropertyHandler"_ustr; } Sequence< OUString > FormComponentPropertyHandler::getSupportedServiceNames( ) { return { u"com.sun.star.form.inspection.FormComponentPropertyHandler"_ustr }; } namespace { // TODO: -> export from toolkit struct LanguageDependentProp { const char* pPropName; sal_Int32 nPropNameLength; }; } const LanguageDependentProp aLanguageDependentProp[] = { { "Text", 4 }, { "Label", 5 }, { "Title", 5 }, { "HelpText", 8 }, { "CurrencySymbol", 14 }, { "StringItemList", 14 }, { nullptr, 0 } }; namespace { bool lcl_isLanguageDependentProperty( std::u16string_view aName ) { bool bRet = false; const LanguageDependentProp* pLangDepProp = aLanguageDependentProp; while( pLangDepProp->pPropName != nullptr ) { if( o3tl::equalsAscii( aName, std::string_view(pLangDepProp->pPropName, pLangDepProp->nPropNameLength) )) { bRet = true; break; } pLangDepProp++; } return bRet; } Reference< resource::XStringResourceResolver > lcl_getStringResourceResolverForProperty ( const Reference< XPropertySet >& _xComponent, std::u16string_view _rPropertyName, const Any& _rPropertyValue ) { Reference< resource::XStringResourceResolver > xRet; const TypeClass eType = _rPropertyValue.getValueTypeClass(); if ( (eType == TypeClass_STRING || eType == TypeClass_SEQUENCE) && lcl_isLanguageDependentProperty( _rPropertyName ) ) { Reference< resource::XStringResourceResolver > xStringResourceResolver; try { xStringResourceResolver.set( _xComponent->getPropertyValue( u"ResourceResolver"_ustr ),UNO_QUERY); if( xStringResourceResolver.is() && xStringResourceResolver->getLocales().hasElements() ) { xRet = std::move(xStringResourceResolver); } } catch(const UnknownPropertyException&) { // nii } } return xRet; } } Any FormComponentPropertyHandler::impl_getPropertyValue_throw( const OUString& _rPropertyName ) const { const PropertyId nPropId( impl_getPropertyId_throwUnknownProperty( _rPropertyName ) ); // tdf#117159 crash with chart in database report if (!m_xComponent) return Any(); Any aPropertyValue( m_xComponent->getPropertyValue( _rPropertyName ) ); Reference< resource::XStringResourceResolver > xStringResourceResolver = lcl_getStringResourceResolverForProperty( m_xComponent, _rPropertyName, aPropertyValue ); if( xStringResourceResolver.is() ) { TypeClass eType = aPropertyValue.getValueTypeClass(); if( eType == TypeClass_STRING ) { OUString aPropStr; aPropertyValue >>= aPropStr; if( aPropStr.getLength() > 1 ) { OUString aPureIdStr = aPropStr.copy( 1 ); if( xStringResourceResolver->hasEntryForId( aPureIdStr ) ) { OUString aResourceStr = xStringResourceResolver->resolveString( aPureIdStr ); aPropertyValue <<= aResourceStr; } } } // StringItemList? else if( eType == TypeClass_SEQUENCE ) { Sequence< OUString > aStrings; aPropertyValue >>= aStrings; std::vector< OUString > aResolvedStrings; aResolvedStrings.reserve( aStrings.getLength() ); try { for (const OUString& rIdStr : aStrings) { OUString aPureIdStr = rIdStr.copy( 1 ); if( xStringResourceResolver->hasEntryForId( aPureIdStr ) ) aResolvedStrings.push_back(xStringResourceResolver->resolveString( aPureIdStr )); else aResolvedStrings.push_back(rIdStr); } } catch( const resource::MissingResourceException & ) {} aPropertyValue <<= comphelper::containerToSequence(aResolvedStrings); } } else impl_normalizePropertyValue_nothrow( aPropertyValue, nPropId ); return aPropertyValue; } Any SAL_CALL FormComponentPropertyHandler::getPropertyValue( const OUString& _rPropertyName ) { if( _rPropertyName == PROPERTY_ROWSET ) return ::comphelper::OPropertyContainer::getPropertyValue( _rPropertyName ); ::osl::MutexGuard aGuard( m_aMutex ); return impl_getPropertyValue_throw( _rPropertyName ); } void SAL_CALL FormComponentPropertyHandler::setPropertyValue( const OUString& _rPropertyName, const Any& _rValue ) { if( _rPropertyName == PROPERTY_ROWSET ) { ::comphelper::OPropertyContainer::setPropertyValue( _rPropertyName, _rValue ); return; } ::osl::MutexGuard aGuard( m_aMutex ); PropertyId nPropId( impl_getPropertyId_throwUnknownProperty( _rPropertyName ) ); // check if property is known by the handler Reference< graphic::XGraphicObject > xGrfObj; if ( PROPERTY_ID_IMAGE_URL == nPropId && ( _rValue >>= xGrfObj ) ) { DBG_ASSERT( xGrfObj.is(), "FormComponentPropertyHandler::setPropertyValue() xGrfObj is invalid"); m_xComponent->setPropertyValue(PROPERTY_GRAPHIC, uno::Any(xGrfObj->getGraphic())); } else if ( PROPERTY_ID_FONT == nPropId ) { // special handling, the value is a faked value we generated ourself in impl_executeFontDialog_nothrow Sequence< NamedValue > aFontPropertyValues; if( ! (_rValue >>= aFontPropertyValues) ) SAL_WARN("extensions.propctrlr", "setPropertyValue: unable to get property " << PROPERTY_ID_FONT); for (const NamedValue& fontPropertyValue : aFontPropertyValues) m_xComponent->setPropertyValue( fontPropertyValue.Name, fontPropertyValue.Value ); } else { Any aValue = _rValue; Reference< resource::XStringResourceResolver > xStringResourceResolver = lcl_getStringResourceResolverForProperty( m_xComponent, _rPropertyName, _rValue ); if( xStringResourceResolver.is() ) { Reference< resource::XStringResourceManager > xStringResourceManager( xStringResourceResolver, UNO_QUERY ); if( xStringResourceManager.is() ) { Any aPropertyValue( m_xComponent->getPropertyValue( _rPropertyName ) ); TypeClass eType = aPropertyValue.getValueTypeClass(); if( eType == TypeClass_STRING ) { OUString aPropStr; aPropertyValue >>= aPropStr; if( aPropStr.getLength() > 1 ) { OUString aPureIdStr = aPropStr.copy( 1 ); OUString aValueStr; _rValue >>= aValueStr; xStringResourceManager->setString( aPureIdStr, aValueStr ); aValue = std::move(aPropertyValue); // set value to force modified } } // StringItemList? else if( eType == TypeClass_SEQUENCE ) { static const char aDot[] = "."; // Put strings into resource using new ids Sequence< OUString > aNewStrings; _rValue >>= aNewStrings; const sal_Int32 nNewCount = aNewStrings.getLength(); // Create new Ids std::unique_ptr pNewPureIds(new OUString[nNewCount]); Any aNameAny = m_xComponent->getPropertyValue(PROPERTY_NAME); OUString sControlName; aNameAny >>= sControlName; OUString aIdStrBase = aDot + sControlName + aDot + _rPropertyName; sal_Int32 i; for ( i = 0; i < nNewCount; ++i ) { sal_Int32 nUniqueId = xStringResourceManager->getUniqueNumericId(); OUString aPureIdStr = OUString::number( nUniqueId ) + aIdStrBase; pNewPureIds[i] = aPureIdStr; // Force usage of next Unique Id xStringResourceManager->setString( aPureIdStr, OUString() ); } // Move strings to new Ids for all locales const Sequence< Locale > aLocaleSeq = xStringResourceManager->getLocales(); Sequence< OUString > aOldIdStrings; aPropertyValue >>= aOldIdStrings; try { const OUString* pOldIdStrings = aOldIdStrings.getConstArray(); sal_Int32 nOldIdCount = aOldIdStrings.getLength(); for ( i = 0; i < nNewCount; ++i ) { OUString aOldPureIdStr; if( i < nOldIdCount ) { OUString aOldIdStr = pOldIdStrings[i]; aOldPureIdStr = aOldIdStr.copy( 1 ); } const OUString& aNewPureIdStr = pNewPureIds[i]; for ( const Locale& rLocale : aLocaleSeq ) { OUString aResourceStr; if( !aOldPureIdStr.isEmpty() ) { if( xStringResourceManager->hasEntryForIdAndLocale( aOldPureIdStr, rLocale ) ) { aResourceStr = xStringResourceManager-> resolveStringForLocale( aOldPureIdStr, rLocale ); } } xStringResourceManager->setStringForLocale( aNewPureIdStr, aResourceStr, rLocale ); } } } catch( const resource::MissingResourceException & ) {} // Set new strings for current locale and create // new Id sequence as new property value Sequence< OUString > aNewIdStrings; aNewIdStrings.realloc( nNewCount ); OUString* pNewIdStrings = aNewIdStrings.getArray(); for ( i = 0; i < nNewCount; ++i ) { const OUString& aPureIdStr = pNewPureIds[i]; const OUString& aStr = aNewStrings[i]; xStringResourceManager->setString( aPureIdStr, aStr ); pNewIdStrings[i] = "&" + aPureIdStr; } aValue <<= aNewIdStrings; // Remove old ids from resource for all locales for (const OUString& rIdStr : aOldIdStrings) { OUString aPureIdStr = rIdStr.copy( 1 ); for ( const Locale& rLocale : aLocaleSeq ) { try { xStringResourceManager->removeIdForLocale( aPureIdStr, rLocale ); } catch( const resource::MissingResourceException & ) {} } } } } } m_xComponent->setPropertyValue( _rPropertyName, aValue ); } } Any SAL_CALL FormComponentPropertyHandler::convertToPropertyValue( const OUString& _rPropertyName, const Any& _rControlValue ) { ::osl::MutexGuard aGuard( m_aMutex ); PropertyId nPropId( impl_getPropertyId_throwUnknownProperty( _rPropertyName ) ); Property aProperty( impl_getPropertyFromId_throw( nPropId ) ); Any aPropertyValue( _rControlValue ); if ( !aPropertyValue.hasValue() ) { if ( ( aProperty.Attributes & PropertyAttribute::MAYBEVOID ) == 0 ) // default construct an instance of the proper type aPropertyValue = Any( nullptr, aProperty.Type ); // nothing to do return aPropertyValue; } /// care for the special "default" string, translate it to VOID if ( m_aPropertiesWithDefListEntry.find( _rPropertyName ) != m_aPropertiesWithDefListEntry.end() ) { // it's a control with a string list OUString sStringValue; if ( _rControlValue >>= sStringValue ) { // note that ColorListBoxes might transfer values either as string or as css.util.Color, // so this check here is important if ( sStringValue == m_sDefaultValueString ) return Any(); } } switch ( nPropId ) { case PROPERTY_ID_DATASOURCE: { OUString sControlValue; if( ! (_rControlValue >>= sControlValue) ) SAL_WARN("extensions.propctrlr", "convertToPropertyValue: unable to get property " << PROPERTY_ID_DATASOURCE); if ( !sControlValue.isEmpty() ) { Reference< XDatabaseContext > xDatabaseContext = sdb::DatabaseContext::create( m_xContext ); if ( !xDatabaseContext->hasByName( sControlValue ) ) { ::svt::OFileNotation aTransformer(sControlValue); aPropertyValue <<= aTransformer.get( ::svt::OFileNotation::N_URL ); } } } break; // case PROPERTY_ID_DATASOURCE case PROPERTY_ID_SHOW_POSITION: case PROPERTY_ID_SHOW_NAVIGATION: case PROPERTY_ID_SHOW_RECORDACTIONS: case PROPERTY_ID_SHOW_FILTERSORT: { OUString sControlValue; if( ! (_rControlValue >>= sControlValue) ) SAL_WARN("extensions.propctrlr", "convertToControlValue: unable to get property for Show/Hide"); static_assert(SAL_N_ELEMENTS(RID_RSC_ENUM_SHOWHIDE) == 2, "FormComponentPropertyHandler::convertToPropertyValue: broken resource for Show/Hide!"); bool bShow = sControlValue == PcrRes(RID_RSC_ENUM_SHOWHIDE[1]); aPropertyValue <<= bShow; } break; case PROPERTY_ID_TARGET_URL: case PROPERTY_ID_IMAGE_URL: { OUString sControlValue; if( ! (_rControlValue >>= sControlValue) ) SAL_WARN("extensions.propctrlr", "convertToPropertyValue: unable to get property for URLs"); // Don't convert a placeholder if ( nPropId == PROPERTY_ID_IMAGE_URL && sControlValue == PcrRes(RID_EMBED_IMAGE_PLACEHOLDER) ) aPropertyValue <<= sControlValue; else { INetURLObject aDocURL( impl_getDocumentURL_nothrow() ); aPropertyValue <<= URIHelper::SmartRel2Abs( aDocURL, sControlValue, Link(), false, true ); } } break; case PROPERTY_ID_DATEMIN: case PROPERTY_ID_DATEMAX: case PROPERTY_ID_DEFAULT_DATE: case PROPERTY_ID_DATE: { util::Date aDate; if( ! (_rControlValue >>= aDate) ) SAL_WARN("extensions.propctrlr", "convertToControlValue: unable to get property for date"); aPropertyValue <<= aDate; } break; case PROPERTY_ID_TIMEMIN: case PROPERTY_ID_TIMEMAX: case PROPERTY_ID_DEFAULT_TIME: case PROPERTY_ID_TIME: { util::Time aTime; if( ! (_rControlValue >>= aTime) ) SAL_WARN("extensions.propctrlr", "convertToControlValue: unable to get property for time"); aPropertyValue <<= aTime; } break; case PROPERTY_ID_WRITING_MODE: { aPropertyValue = PropertyHandlerComponent::convertToPropertyValue( _rPropertyName, _rControlValue ); sal_Int16 nNormalizedValue( 2 ); if( ! (aPropertyValue >>= nNormalizedValue) ) SAL_WARN("extensions.propctrlr", "convertToControlValue: unable to get property for " << PROPERTY_ID_WRITING_MODE); sal_Int16 nWritingMode = WritingMode2::CONTEXT; switch ( nNormalizedValue ) { case 0: nWritingMode = WritingMode2::LR_TB; break; case 1: nWritingMode = WritingMode2::RL_TB; break; case 2: nWritingMode = WritingMode2::CONTEXT; break; default: OSL_FAIL( "FormComponentPropertyHandler::convertToPropertyValue: unexpected 'normalized value' for WritingMode!" ); nWritingMode = WritingMode2::CONTEXT; break; } aPropertyValue <<= nWritingMode; } break; default: aPropertyValue = PropertyHandlerComponent::convertToPropertyValue( _rPropertyName, _rControlValue ); break; // default } // switch ( nPropId ) return aPropertyValue; } Any SAL_CALL FormComponentPropertyHandler::convertToControlValue( const OUString& _rPropertyName, const Any& _rPropertyValue, const Type& _rControlValueType ) { ::osl::MutexGuard aGuard( m_aMutex ); sal_Int32 nPropId = m_pInfoService->getPropertyId( _rPropertyName ); DBG_ASSERT( nPropId != -1, "FormComponentPropertyHandler::convertToPropertyValue: not one of my properties!!" ); impl_getPropertyFromId_throw( nPropId ); Any aControlValue( _rPropertyValue ); if ( !aControlValue.hasValue() ) { // if the property is represented with a list box or color list box, we need to // translate this into the string "Default" if ( m_aPropertiesWithDefListEntry.find( _rPropertyName ) != m_aPropertiesWithDefListEntry.end() ) aControlValue <<= m_sDefaultValueString; return aControlValue; } switch ( nPropId ) { case PROPERTY_ID_SHOW_POSITION: case PROPERTY_ID_SHOW_NAVIGATION: case PROPERTY_ID_SHOW_RECORDACTIONS: case PROPERTY_ID_SHOW_FILTERSORT: { static_assert(SAL_N_ELEMENTS(RID_RSC_ENUM_SHOWHIDE) == 2, "FormComponentPropertyHandler::convertToPropertyValue: broken resource for Show/Hide!"); OUString sControlValue = ::comphelper::getBOOL(_rPropertyValue) ? PcrRes(RID_RSC_ENUM_SHOWHIDE[1]) : PcrRes(RID_RSC_ENUM_SHOWHIDE[0]); aControlValue <<= sControlValue; } break; case PROPERTY_ID_DATASOURCE: { OSL_ENSURE( _rControlValueType.getTypeClass() == TypeClass_STRING, "FormComponentPropertyHandler::convertToControlValue: wrong ControlValueType!" ); OUString sDataSource; _rPropertyValue >>= sDataSource; if ( !sDataSource.isEmpty() ) { ::svt::OFileNotation aTransformer( sDataSource ); sDataSource = aTransformer.get( ::svt::OFileNotation::N_SYSTEM ); } aControlValue <<= sDataSource; } break; case PROPERTY_ID_CONTROLLABEL: { OUString sControlValue; Reference< XPropertySet > xSet; _rPropertyValue >>= xSet; Reference< XPropertySetInfo > xPSI; if ( xSet.is() ) xPSI = xSet->getPropertySetInfo(); if ( xPSI.is() && xPSI->hasPropertyByName( PROPERTY_LABEL ) ) { OUString sLabel; if( ! (xSet->getPropertyValue( PROPERTY_LABEL) >>= sLabel) ) SAL_WARN("extensions.propctrlr", "convertToPropertyValue: unable to get property " << PROPERTY_LABEL); sControlValue = "<" + sLabel + ">"; } aControlValue <<= sControlValue; } break; case PROPERTY_ID_DATEMIN: case PROPERTY_ID_DATEMAX: case PROPERTY_ID_DEFAULT_DATE: case PROPERTY_ID_DATE: { sal_Int32 nDate = 0; if( ! (_rPropertyValue >>= nDate) ) SAL_WARN("extensions.propctrlr", "convertToControlValue: unable to get property for dates"); aControlValue <<= DBTypeConversion::toDate( nDate ); } break; case PROPERTY_ID_TIMEMIN: case PROPERTY_ID_TIMEMAX: case PROPERTY_ID_DEFAULT_TIME: case PROPERTY_ID_TIME: { sal_Int64 nTime = 0; if( ! (_rPropertyValue >>= nTime) ) SAL_WARN("extensions.propctrlr", "convertToControlValue: unable to get property for times"); aControlValue <<= DBTypeConversion::toTime( nTime ); } break; case PROPERTY_ID_WRITING_MODE: { sal_Int16 nWritingMode( WritingMode2::CONTEXT ); if( ! (_rPropertyValue >>= nWritingMode) ) SAL_WARN("extensions.propctrlr", "convertToControlValue: unable to get property " << PROPERTY_ID_WRITING_MODE); sal_Int16 nNormalized = 2; switch ( nWritingMode ) { case WritingMode2::LR_TB: nNormalized = 0; break; case WritingMode2::RL_TB: nNormalized = 1; break; case WritingMode2::CONTEXT: nNormalized = 2; break; default: OSL_FAIL( "FormComponentPropertyHandler::convertToControlValue: unsupported API value for WritingMode!" ); nNormalized = 2; break; } aControlValue = PropertyHandlerComponent::convertToControlValue( _rPropertyName, Any( nNormalized ), _rControlValueType ); } break; case PROPERTY_ID_FONT: { FontDescriptor aFont; if( ! (_rPropertyValue >>= aFont) ) SAL_WARN("extensions.propctrlr", "convertToControlValue: unable to get property " << PROPERTY_ID_FONT); OUStringBuffer displayName; if ( aFont.Name.isEmpty() ) { displayName.append( PcrRes(RID_STR_FONT_DEFAULT) ); } else { // font name displayName.append( aFont.Name + ", " ); // font style ::FontWeight eWeight = vcl::unohelper::ConvertFontWeight( aFont.Weight ); TranslateId pStyleResID = RID_STR_FONTSTYLE_REGULAR; if ( aFont.Slant == FontSlant_ITALIC ) { if ( eWeight > WEIGHT_NORMAL ) pStyleResID = RID_STR_FONTSTYLE_BOLD_ITALIC; else pStyleResID = RID_STR_FONTSTYLE_ITALIC; } else { if ( eWeight > WEIGHT_NORMAL ) pStyleResID = RID_STR_FONTSTYLE_BOLD; } displayName.append(PcrRes(pStyleResID)); // font size if ( aFont.Height ) { displayName.append( ", " + OUString::number( sal_Int32( aFont.Height ) ) ); } } aControlValue <<= displayName.makeStringAndClear(); } break; default: aControlValue = PropertyHandlerComponent::convertToControlValue( _rPropertyName, _rPropertyValue, _rControlValueType ); break; } // switch ( nPropId ) return aControlValue; } PropertyState SAL_CALL FormComponentPropertyHandler::getPropertyState( const OUString& _rPropertyName ) { ::osl::MutexGuard aGuard( m_aMutex ); if ( m_xPropertyState.is() ) return m_xPropertyState->getPropertyState( _rPropertyName ); return PropertyState_DIRECT_VALUE; } void SAL_CALL FormComponentPropertyHandler::addPropertyChangeListener( const Reference< XPropertyChangeListener >& _rxListener ) { ::osl::MutexGuard aGuard( m_aMutex ); PropertyHandlerComponent::addPropertyChangeListener( _rxListener ); if ( m_xComponent.is() ) m_xComponent->addPropertyChangeListener( OUString(), _rxListener ); } void SAL_CALL FormComponentPropertyHandler::removePropertyChangeListener( const Reference< XPropertyChangeListener >& _rxListener ) { ::osl::MutexGuard aGuard( m_aMutex ); if ( m_xComponent.is() ) m_xComponent->removePropertyChangeListener( OUString(), _rxListener ); PropertyHandlerComponent::removePropertyChangeListener( _rxListener ); } Sequence< Property > FormComponentPropertyHandler::doDescribeSupportedProperties() const { if ( !m_xComponentPropertyInfo.is() ) return Sequence< Property >(); std::vector< Property > aProperties; Sequence< Property > aAllProperties( m_xComponentPropertyInfo->getProperties() ); aProperties.reserve( aAllProperties.getLength() ); // filter the properties PropertyId nPropId( 0 ); OUString sDisplayName; for ( Property & rProperty : asNonConstRange(aAllProperties) ) { nPropId = m_pInfoService->getPropertyId( rProperty.Name ); if ( nPropId == -1 ) continue; rProperty.Handle = nPropId; sDisplayName = m_pInfoService->getPropertyTranslation( nPropId ); if ( sDisplayName.isEmpty() ) continue; sal_uInt32 nPropertyUIFlags = m_pInfoService->getPropertyUIFlags( nPropId ); bool bIsVisibleForForms = ( nPropertyUIFlags & PROP_FLAG_FORM_VISIBLE ) != 0; bool bIsVisibleForDialogs = ( nPropertyUIFlags & PROP_FLAG_DIALOG_VISIBLE ) != 0; // depending on whether we're working for a form or a UNO dialog, some // properties are not displayed if ( ( m_eComponentClass == eFormControl && !bIsVisibleForForms ) || ( m_eComponentClass == eDialogControl && !bIsVisibleForDialogs ) ) continue; // some generic sanity checks if ( impl_shouldExcludeProperty_nothrow( rProperty ) ) continue; switch ( nPropId ) { case PROPERTY_ID_BORDER: case PROPERTY_ID_TABSTOP: // BORDER and TABSTOP are normalized (see impl_normalizePropertyValue_nothrow) // to not allow VOID values rProperty.Attributes &= ~PropertyAttribute::MAYBEVOID; break; case PROPERTY_ID_LISTSOURCE: // no cursor source if no Base is installed. if (SvtModuleOptions().IsDataBaseInstalled()) const_cast< FormComponentPropertyHandler* >( this )->m_bHaveListSource = true; break; case PROPERTY_ID_COMMAND: // no cursor source if no Base is installed. if (SvtModuleOptions().IsDataBaseInstalled()) const_cast< FormComponentPropertyHandler* >( this )->m_bHaveCommand = true; break; } // switch ( nPropId ) aProperties.push_back( rProperty ); } if ( aProperties.empty() ) return Sequence< Property >(); return comphelper::containerToSequence(aProperties); } Sequence< OUString > SAL_CALL FormComponentPropertyHandler::getSupersededProperties( ) { return Sequence< OUString >( ); } Sequence< OUString > SAL_CALL FormComponentPropertyHandler::getActuatingProperties( ) { return { PROPERTY_DATASOURCE, PROPERTY_COMMAND, PROPERTY_COMMANDTYPE, PROPERTY_LISTSOURCE, PROPERTY_LISTSOURCETYPE, PROPERTY_SUBMIT_ENCODING, PROPERTY_REPEAT, PROPERTY_TABSTOP, PROPERTY_BORDER, PROPERTY_CONTROLSOURCE, PROPERTY_DROPDOWN, PROPERTY_IMAGE_URL, PROPERTY_TARGET_URL, PROPERTY_STRINGITEMLIST, PROPERTY_BUTTONTYPE, PROPERTY_ESCAPE_PROCESSING, PROPERTY_TRISTATE, PROPERTY_DECIMAL_ACCURACY, PROPERTY_SHOWTHOUSANDSEP, PROPERTY_FORMATKEY, PROPERTY_EMPTY_IS_NULL, PROPERTY_TOGGLE }; } LineDescriptor SAL_CALL FormComponentPropertyHandler::describePropertyLine( const OUString& _rPropertyName, const Reference< XPropertyControlFactory >& _rxControlFactory ) { if ( !_rxControlFactory.is() ) throw NullPointerException(); ::osl::MutexGuard aGuard( m_aMutex ); PropertyId nPropId( impl_getPropertyId_throwUnknownProperty( _rPropertyName ) ); Property aProperty( impl_getPropertyFromId_throw( nPropId ) ); // for the MultiLine property, we have different UI translations depending on the control // type if ( nPropId == PROPERTY_ID_MULTILINE ) { if ( ( m_nClassId == FormComponentType::FIXEDTEXT ) || ( m_nClassId == FormComponentType::COMMANDBUTTON ) || ( m_nClassId == FormComponentType::RADIOBUTTON ) || ( m_nClassId == FormComponentType::CHECKBOX ) ) nPropId = PROPERTY_ID_WORDBREAK; } OUString sDisplayName = m_pInfoService->getPropertyTranslation( nPropId ); if ( sDisplayName.isEmpty() ) { OSL_FAIL( "FormComponentPropertyHandler::describePropertyLine: did getSupportedProperties not work properly?" ); throw UnknownPropertyException(); } LineDescriptor aDescriptor; aDescriptor.HelpURL = HelpIdUrl::getHelpURL( m_pInfoService->getPropertyHelpId( nPropId ) ); aDescriptor.DisplayName = sDisplayName; // for the moment, assume a text field sal_Int16 nControlType = PropertyControlType::TextField; bool bReadOnly = false; aDescriptor.Control.clear(); bool bNeedDefaultStringIfVoidAllowed = false; TypeClass eType = aProperty.Type.getTypeClass(); switch ( nPropId ) { case PROPERTY_ID_DEFAULT_SELECT_SEQ: case PROPERTY_ID_SELECTEDITEMS: aDescriptor.PrimaryButtonId = UID_PROP_DLG_SELECTION; break; case PROPERTY_ID_FILTER: aDescriptor.PrimaryButtonId = UID_PROP_DLG_FILTER; break; case PROPERTY_ID_SORT: aDescriptor.PrimaryButtonId = UID_PROP_DLG_ORDER; break; case PROPERTY_ID_MASTERFIELDS: case PROPERTY_ID_DETAILFIELDS: nControlType = PropertyControlType::StringListField; aDescriptor.PrimaryButtonId = UID_PROP_DLG_FORMLINKFIELDS; break; case PROPERTY_ID_COMMAND: aDescriptor.PrimaryButtonId = UID_PROP_DLG_SQLCOMMAND; break; case PROPERTY_ID_TABINDEX: { Reference< XControlContainer > xControlContext( impl_getContextControlContainer_nothrow() ); if ( xControlContext.is() ) aDescriptor.PrimaryButtonId = UID_PROP_DLG_TABINDEX; nControlType = PropertyControlType::NumericField; }; break; case PROPERTY_ID_FONT: bReadOnly = true; aDescriptor.PrimaryButtonId = UID_PROP_DLG_FONT_TYPE; break; case PROPERTY_ID_TARGET_URL: case PROPERTY_ID_IMAGE_URL: { std::unique_ptr xBuilder(PropertyHandlerHelper::makeBuilder(u"modules/spropctrlr/ui/urlcontrol.ui"_ustr, m_xContext)); auto pURLBox = std::make_unique(xBuilder->weld_combo_box(u"urlcontrol"_ustr)); rtl::Reference pControl = new OFileUrlControl(std::move(pURLBox), std::move(xBuilder), false); pControl->SetModifyHandler(); aDescriptor.Control = pControl; aDescriptor.PrimaryButtonId = PROPERTY_ID_TARGET_URL == nPropId ? UID_PROP_DLG_ATTR_TARGET_URL : UID_PROP_DLG_IMAGE_URL; break; } case PROPERTY_ID_ECHO_CHAR: nControlType = PropertyControlType::CharacterField; break; case PROPERTY_ID_BACKGROUNDCOLOR: case PROPERTY_ID_FILLCOLOR: case PROPERTY_ID_SYMBOLCOLOR: case PROPERTY_ID_BORDERCOLOR: case PROPERTY_ID_GRIDLINECOLOR: case PROPERTY_ID_HEADERBACKGROUNDCOLOR: case PROPERTY_ID_HEADERTEXTCOLOR: case PROPERTY_ID_ACTIVESELECTIONBACKGROUNDCOLOR: case PROPERTY_ID_ACTIVESELECTIONTEXTCOLOR: case PROPERTY_ID_INACTIVESELECTIONBACKGROUNDCOLOR: case PROPERTY_ID_INACTIVESELECTIONTEXTCOLOR: nControlType = PropertyControlType::ColorListBox; switch( nPropId ) { case PROPERTY_ID_BACKGROUNDCOLOR: aDescriptor.PrimaryButtonId = UID_PROP_DLG_BACKGROUNDCOLOR; break; case PROPERTY_ID_FILLCOLOR: aDescriptor.PrimaryButtonId = UID_PROP_DLG_FILLCOLOR; break; case PROPERTY_ID_SYMBOLCOLOR: aDescriptor.PrimaryButtonId = UID_PROP_DLG_SYMBOLCOLOR; break; case PROPERTY_ID_BORDERCOLOR: aDescriptor.PrimaryButtonId = UID_PROP_DLG_BORDERCOLOR; break; case PROPERTY_ID_GRIDLINECOLOR: aDescriptor.PrimaryButtonId = HID_PROP_GRIDLINECOLOR; break; case PROPERTY_ID_HEADERBACKGROUNDCOLOR: aDescriptor.PrimaryButtonId = HID_PROP_HEADERBACKGROUNDCOLOR; break; case PROPERTY_ID_HEADERTEXTCOLOR: aDescriptor.PrimaryButtonId = HID_PROP_HEADERTEXTCOLOR; break; case PROPERTY_ID_ACTIVESELECTIONBACKGROUNDCOLOR: aDescriptor.PrimaryButtonId = HID_PROP_ACTIVESELECTIONBACKGROUNDCOLOR; break; case PROPERTY_ID_ACTIVESELECTIONTEXTCOLOR: aDescriptor.PrimaryButtonId = HID_PROP_ACTIVESELECTIONTEXTCOLOR; break; case PROPERTY_ID_INACTIVESELECTIONBACKGROUNDCOLOR: aDescriptor.PrimaryButtonId = HID_PROP_INACTIVESELECTIONBACKGROUNDCOLOR; break; case PROPERTY_ID_INACTIVESELECTIONTEXTCOLOR: aDescriptor.PrimaryButtonId = HID_PROP_INACTIVESELECTIONTEXTCOLOR; break; } break; case PROPERTY_ID_LABEL: case PROPERTY_ID_URL: nControlType = PropertyControlType::MultiLineTextField; break; case PROPERTY_ID_DEFAULT_TEXT: { if (FormComponentType::FILECONTROL == m_nClassId) nControlType = PropertyControlType::TextField; else nControlType = PropertyControlType::MultiLineTextField; } break; case PROPERTY_ID_TEXT: if ( impl_componentHasProperty_throw( PROPERTY_MULTILINE ) ) nControlType = PropertyControlType::MultiLineTextField; break; case PROPERTY_ID_CONTROLLABEL: bReadOnly = true; aDescriptor.PrimaryButtonId = UID_PROP_DLG_CONTROLLABEL; break; case PROPERTY_ID_FORMATKEY: case PROPERTY_ID_EFFECTIVE_MIN: case PROPERTY_ID_EFFECTIVE_MAX: case PROPERTY_ID_EFFECTIVE_DEFAULT: case PROPERTY_ID_EFFECTIVE_VALUE: { // and the supplier is really available Reference< XNumberFormatsSupplier > xSupplier; m_xComponent->getPropertyValue( PROPERTY_FORMATSSUPPLIER ) >>= xSupplier; if (xSupplier.is()) { Reference< XUnoTunnel > xTunnel(xSupplier,UNO_QUERY); DBG_ASSERT(xTunnel.is(), "FormComponentPropertyHandler::describePropertyLine : xTunnel is invalid!"); if (auto pSupplier = comphelper::getFromUnoTunnel(xTunnel)) { bool bIsFormatKey = (PROPERTY_ID_FORMATKEY == nPropId); bReadOnly = bIsFormatKey; if ( bIsFormatKey ) { std::unique_ptr xBuilder(PropertyHandlerHelper::makeBuilder(u"modules/spropctrlr/ui/formattedsample.ui"_ustr, m_xContext)); auto pContainer = xBuilder->weld_container(u"formattedsample"_ustr); rtl::Reference pControl = new OFormatSampleControl(std::move(pContainer), std::move(xBuilder), false); pControl->SetModifyHandler(); pControl->SetFormatSupplier(pSupplier); aDescriptor.Control = pControl; aDescriptor.PrimaryButtonId = UID_PROP_DLG_NUMBER_FORMAT; } else { std::unique_ptr xBuilder(PropertyHandlerHelper::makeBuilder(u"modules/spropctrlr/ui/formattedcontrol.ui"_ustr, m_xContext)); auto pSpinButton = xBuilder->weld_formatted_spin_button(u"formattedcontrol"_ustr); rtl::Reference pControl = new OFormattedNumericControl(std::move(pSpinButton), std::move(xBuilder), false); pControl->SetModifyHandler(); FormatDescription aDesc; aDesc.pSupplier = pSupplier; Any aFormatKeyValue = m_xComponent->getPropertyValue(PROPERTY_FORMATKEY); if ( !( aFormatKeyValue >>= aDesc.nKey ) ) aDesc.nKey = 0; pControl->SetFormatDescription( aDesc ); aDescriptor.Control = pControl; } } } } break; case PROPERTY_ID_DATEMIN: case PROPERTY_ID_DATEMAX: case PROPERTY_ID_DEFAULT_DATE: case PROPERTY_ID_DATE: nControlType = PropertyControlType::DateField; break; case PROPERTY_ID_TIMEMIN: case PROPERTY_ID_TIMEMAX: case PROPERTY_ID_DEFAULT_TIME: case PROPERTY_ID_TIME: nControlType = PropertyControlType::TimeField; break; case PROPERTY_ID_VALUEMIN: case PROPERTY_ID_VALUEMAX: case PROPERTY_ID_DEFAULT_VALUE: case PROPERTY_ID_VALUE: { std::unique_ptr xBuilder(PropertyHandlerHelper::makeBuilder(u"modules/spropctrlr/ui/formattedcontrol.ui"_ustr, m_xContext)); auto pSpinButton = xBuilder->weld_formatted_spin_button(u"formattedcontrol"_ustr); rtl::Reference pControl = new OFormattedNumericControl(std::move(pSpinButton), std::move(xBuilder), false); pControl->SetModifyHandler(); aDescriptor.Control = pControl; // we don't set a formatter so the control uses a default (which uses the application // language and a default numeric format) // but we set the decimal digits pControl->SetDecimalDigits( ::comphelper::getINT16( m_xComponent->getPropertyValue( PROPERTY_DECIMAL_ACCURACY ) ) ); // and the default value for the property try { if (m_xPropertyState.is() && ((PROPERTY_ID_VALUEMIN == nPropId) || (PROPERTY_ID_VALUEMAX == nPropId))) { double nDefault = 0; if ( m_xPropertyState->getPropertyDefault( aProperty.Name ) >>= nDefault ) pControl->SetDefaultValue(nDefault); } } catch (const Exception&) { // just ignore it } break; } default: if ( TypeClass_BYTE <= eType && eType <= TypeClass_DOUBLE ) { sal_Int16 nDigits = 0; sal_Int16 nValueUnit = -1; sal_Int16 nDisplayUnit = -1; if ( m_eComponentClass == eFormControl ) { if ( ( nPropId == PROPERTY_ID_WIDTH ) || ( nPropId == PROPERTY_ID_ROWHEIGHT ) || ( nPropId == PROPERTY_ID_HEIGHT ) ) { nValueUnit = MeasureUnit::MM_10TH; nDisplayUnit = impl_getDocumentMeasurementUnit_throw(); nDigits = 2; } } Optional< double > aValueNotPresent( false, 0 ); aDescriptor.Control = PropertyHandlerHelper::createNumericControl( _rxControlFactory, nDigits, aValueNotPresent, aValueNotPresent ); Reference< XNumericControl > xNumericControl( aDescriptor.Control, UNO_QUERY_THROW ); if ( nValueUnit != -1 ) xNumericControl->setValueUnit( nValueUnit ); if ( nDisplayUnit != -1 ) xNumericControl->setDisplayUnit( nDisplayUnit ); } break; } if ( eType == TypeClass_SEQUENCE ) nControlType = PropertyControlType::StringListField; // boolean values if ( eType == TypeClass_BOOLEAN ) { if ( ( nPropId == PROPERTY_ID_SHOW_POSITION ) || ( nPropId == PROPERTY_ID_SHOW_NAVIGATION ) || ( nPropId == PROPERTY_ID_SHOW_RECORDACTIONS ) || ( nPropId == PROPERTY_ID_SHOW_FILTERSORT ) ) { aDescriptor.Control = PropertyHandlerHelper::createListBoxControl(_rxControlFactory, RID_RSC_ENUM_SHOWHIDE, SAL_N_ELEMENTS(RID_RSC_ENUM_SHOWHIDE), false); } else aDescriptor.Control = PropertyHandlerHelper::createListBoxControl(_rxControlFactory, RID_RSC_ENUM_YESNO, SAL_N_ELEMENTS(RID_RSC_ENUM_YESNO), false); bNeedDefaultStringIfVoidAllowed = true; } // enum properties sal_uInt32 nPropertyUIFlags = m_pInfoService->getPropertyUIFlags( nPropId ); bool bIsEnumProperty = ( nPropertyUIFlags & PROP_FLAG_ENUM ) != 0; if ( bIsEnumProperty || ( PROPERTY_ID_TARGET_FRAME == nPropId ) ) { std::vector< OUString > aEnumValues = m_pInfoService->getPropertyEnumRepresentations( nPropId ); std::vector< OUString >::const_iterator pStart = aEnumValues.begin(); std::vector< OUString >::const_iterator pEnd = aEnumValues.end(); // for a checkbox: if "ambiguous" is not allowed, remove this from the sequence if ( ( PROPERTY_ID_DEFAULT_STATE == nPropId ) || ( PROPERTY_ID_STATE == nPropId ) ) { if ( impl_componentHasProperty_throw( PROPERTY_TRISTATE ) ) { if ( !::comphelper::getBOOL( m_xComponent->getPropertyValue( PROPERTY_TRISTATE ) ) ) { // remove the last sequence element if ( pEnd > pStart ) --pEnd; } } else --pEnd; } if ( PROPERTY_ID_LISTSOURCETYPE == nPropId ) if ( FormComponentType::COMBOBOX == m_nClassId ) // remove the first sequence element -> value list not possible for combo boxes ++pStart; // copy the sequence std::vector< OUString > aListEntries( pEnd - pStart ); std::copy( pStart, pEnd, aListEntries.begin() ); // create the control if ( PROPERTY_ID_TARGET_FRAME == nPropId ) aDescriptor.Control = PropertyHandlerHelper::createComboBoxControl( _rxControlFactory, std::move(aListEntries), false ); else { aDescriptor.Control = PropertyHandlerHelper::createListBoxControl( _rxControlFactory, std::move(aListEntries), false, false ); bNeedDefaultStringIfVoidAllowed = true; } } switch( nPropId ) { case PROPERTY_ID_REPEAT_DELAY: { std::unique_ptr xBuilder(PropertyHandlerHelper::makeBuilder(u"modules/spropctrlr/ui/numericfield.ui"_ustr, m_xContext)); auto pSpinButton = xBuilder->weld_metric_spin_button(u"numericfield"_ustr, FieldUnit::MILLISECOND); rtl::Reference pControl = new ONumericControl(std::move(pSpinButton), std::move(xBuilder), bReadOnly); pControl->SetModifyHandler(); pControl->setMinValue( Optional< double >( true, 0 ) ); pControl->setMaxValue( Optional< double >( true, std::numeric_limits< double >::max() ) ); aDescriptor.Control = pControl; } break; case PROPERTY_ID_TABINDEX: case PROPERTY_ID_BOUNDCOLUMN: case PROPERTY_ID_VISIBLESIZE: case PROPERTY_ID_MAXTEXTLEN: case PROPERTY_ID_LINEINCREMENT: case PROPERTY_ID_BLOCKINCREMENT: case PROPERTY_ID_SPININCREMENT: { Optional< double > aMinValue( true, 0 ); Optional< double > aMaxValue( true, 0x7FFFFFFF ); if ( nPropId == PROPERTY_ID_MAXTEXTLEN || nPropId == PROPERTY_ID_BOUNDCOLUMN ) aMinValue.Value = -1; else if ( nPropId == PROPERTY_ID_VISIBLESIZE ) aMinValue.Value = 1; else aMinValue.Value = 0; aDescriptor.Control = PropertyHandlerHelper::createNumericControl( _rxControlFactory, 0, aMinValue, aMaxValue ); } break; case PROPERTY_ID_DECIMAL_ACCURACY: { Optional< double > aMinValue( true, 0 ); Optional< double > aMaxValue( true, 20 ); aDescriptor.Control = PropertyHandlerHelper::createNumericControl( _rxControlFactory, 0, aMinValue, aMaxValue ); } break; // DataSource case PROPERTY_ID_DATASOURCE: { aDescriptor.PrimaryButtonId = UID_PROP_DLG_ATTR_DATASOURCE; std::vector< OUString > aListEntries; Reference< XDatabaseContext > xDatabaseContext = sdb::DatabaseContext::create( m_xContext ); const Sequence< OUString > aDatasources = xDatabaseContext->getElementNames(); aListEntries.resize( aDatasources.getLength() ); std::copy( aDatasources.begin(), aDatasources.end(), aListEntries.begin() ); aDescriptor.Control = PropertyHandlerHelper::createComboBoxControl( _rxControlFactory, std::move(aListEntries), true ); } break; case PROPERTY_ID_CONTROLSOURCE: { std::vector< OUString > aFieldNames; impl_initFieldList_nothrow( aFieldNames ); aDescriptor.Control = PropertyHandlerHelper::createComboBoxControl( _rxControlFactory, std::move(aFieldNames), false ); } break; case PROPERTY_ID_COMMAND: impl_describeCursorSource_nothrow( aDescriptor, _rxControlFactory ); break; case PROPERTY_ID_LISTSOURCE: impl_describeListSourceUI_throw( aDescriptor, _rxControlFactory ); break; } if ( !aDescriptor.Control.is() ) aDescriptor.Control = _rxControlFactory->createPropertyControl( nControlType, bReadOnly ); if ( ( aProperty.Attributes & PropertyAttribute::MAYBEVOID ) != 0 ) { // insert the string "Default" string, if necessary if (bNeedDefaultStringIfVoidAllowed) { Reference< XStringListControl > xStringList( aDescriptor.Control, UNO_QUERY_THROW ); xStringList->prependListEntry( m_sDefaultValueString ); m_aPropertiesWithDefListEntry.insert( _rPropertyName ); } } if ( !aDescriptor.PrimaryButtonId.isEmpty() ) aDescriptor.HasPrimaryButton = true; if ( !aDescriptor.SecondaryButtonId.isEmpty() ) aDescriptor.HasSecondaryButton = true; bool bIsDataProperty = ( nPropertyUIFlags & PROP_FLAG_DATA_PROPERTY ) != 0; aDescriptor.Category = bIsDataProperty ? std::u16string_view(u"Data") : std::u16string_view(u"General"); return aDescriptor; } InteractiveSelectionResult SAL_CALL FormComponentPropertyHandler::onInteractivePropertySelection( const OUString& _rPropertyName, sal_Bool /*_bPrimary*/, Any& _rData, const Reference< XObjectInspectorUI >& _rxInspectorUI ) { if ( !_rxInspectorUI.is() ) throw NullPointerException(); ::osl::ClearableMutexGuard aGuard( m_aMutex ); PropertyId nPropId( impl_getPropertyId_throwUnknownProperty( _rPropertyName ) ); InteractiveSelectionResult eResult = InteractiveSelectionResult_Cancelled; switch ( nPropId ) { case PROPERTY_ID_DEFAULT_SELECT_SEQ: case PROPERTY_ID_SELECTEDITEMS: if ( impl_dialogListSelection_nothrow( _rPropertyName, aGuard ) ) eResult = InteractiveSelectionResult_Success; break; case PROPERTY_ID_FILTER: case PROPERTY_ID_SORT: { OUString sClause; if ( impl_dialogFilterOrSort_nothrow( PROPERTY_ID_FILTER == nPropId, sClause, aGuard ) ) { _rData <<= sClause; eResult = InteractiveSelectionResult_ObtainedValue; } } break; case PROPERTY_ID_MASTERFIELDS: case PROPERTY_ID_DETAILFIELDS: if ( impl_dialogLinkedFormFields_nothrow( aGuard ) ) eResult = InteractiveSelectionResult_Success; break; case PROPERTY_ID_FORMATKEY: if ( impl_dialogFormatting_nothrow( _rData, aGuard ) ) eResult = InteractiveSelectionResult_ObtainedValue; break; case PROPERTY_ID_IMAGE_URL: if ( impl_browseForImage_nothrow( _rData, aGuard ) ) eResult = InteractiveSelectionResult_ObtainedValue; break; case PROPERTY_ID_TARGET_URL: if ( impl_browseForTargetURL_nothrow( _rData, aGuard ) ) eResult = InteractiveSelectionResult_ObtainedValue; break; case PROPERTY_ID_FONT: if ( impl_executeFontDialog_nothrow( _rData, aGuard ) ) eResult = InteractiveSelectionResult_ObtainedValue; break; case PROPERTY_ID_DATASOURCE: if ( impl_browseForDatabaseDocument_throw( _rData, aGuard ) ) eResult = InteractiveSelectionResult_ObtainedValue; break; case PROPERTY_ID_BACKGROUNDCOLOR: case PROPERTY_ID_FILLCOLOR: case PROPERTY_ID_SYMBOLCOLOR: case PROPERTY_ID_BORDERCOLOR: case PROPERTY_ID_GRIDLINECOLOR: case PROPERTY_ID_HEADERBACKGROUNDCOLOR: case PROPERTY_ID_HEADERTEXTCOLOR: case PROPERTY_ID_ACTIVESELECTIONBACKGROUNDCOLOR: case PROPERTY_ID_ACTIVESELECTIONTEXTCOLOR: case PROPERTY_ID_INACTIVESELECTIONBACKGROUNDCOLOR: case PROPERTY_ID_INACTIVESELECTIONTEXTCOLOR: if ( impl_dialogColorChooser_throw( nPropId, _rData, aGuard ) ) eResult = InteractiveSelectionResult_ObtainedValue; break; case PROPERTY_ID_CONTROLLABEL: if ( impl_dialogChooseLabelControl_nothrow( _rData, aGuard ) ) eResult = InteractiveSelectionResult_ObtainedValue; break; case PROPERTY_ID_TABINDEX: if ( impl_dialogChangeTabOrder_nothrow( aGuard ) ) eResult = InteractiveSelectionResult_Success; break; case PROPERTY_ID_COMMAND: case PROPERTY_ID_LISTSOURCE: if ( impl_doDesignSQLCommand_nothrow( _rxInspectorUI, nPropId ) ) eResult = InteractiveSelectionResult_Pending; break; default: OSL_FAIL( "FormComponentPropertyHandler::onInteractivePropertySelection: request for a property which does not have dedicated UI!" ); break; } return eResult; } namespace { void lcl_rebuildAndResetCommand( const Reference< XObjectInspectorUI >& _rxInspectorUI, const Reference< XPropertyHandler >& _rxHandler ) { OSL_PRECOND( _rxInspectorUI.is(), "lcl_rebuildAndResetCommand: invalid BrowserUI!" ); OSL_PRECOND( _rxHandler.is(), "lcl_rebuildAndResetCommand: invalid handler!" ); _rxInspectorUI->rebuildPropertyUI( PROPERTY_COMMAND ); _rxHandler->setPropertyValue( PROPERTY_COMMAND, Any( OUString() ) ); } } void SAL_CALL FormComponentPropertyHandler::actuatingPropertyChanged( const OUString& _rActuatingPropertyName, const Any& _rNewValue, const Any& /*_rOldValue*/, const Reference< XObjectInspectorUI >& _rxInspectorUI, sal_Bool _bFirstTimeInit ) { if ( !_rxInspectorUI.is() ) throw NullPointerException(); ::osl::MutexGuard aGuard( m_aMutex ); PropertyId nActuatingPropId( impl_getPropertyId_nothrow( _rActuatingPropertyName ) ); std::vector< PropertyId > aDependentProperties; switch ( nActuatingPropId ) { // ----- EscapeProcessing ----- case PROPERTY_ID_ESCAPE_PROCESSING: aDependentProperties.push_back( PROPERTY_ID_FILTER ); aDependentProperties.push_back( PROPERTY_ID_SORT ); break; // case PROPERTY_ID_ESCAPE_PROCESSING // ----- CommandType ----- case PROPERTY_ID_COMMANDTYPE: // available commands (tables or queries) might have changed if ( !_bFirstTimeInit && m_bHaveCommand ) lcl_rebuildAndResetCommand( _rxInspectorUI, this ); aDependentProperties.push_back( PROPERTY_ID_COMMAND ); break; // case PROPERTY_ID_COMMANDTYPE // ----- DataSourceName ----- case PROPERTY_ID_DATASOURCE: // reset the connection, now that we have a new data source m_xRowSetConnection.clear(); // available list source values (tables or queries) might have changed if ( !_bFirstTimeInit && m_bHaveListSource ) _rxInspectorUI->rebuildPropertyUI( PROPERTY_LISTSOURCE ); // available commands (tables or queries) might have changed if ( !_bFirstTimeInit && m_bHaveCommand ) lcl_rebuildAndResetCommand( _rxInspectorUI, this ); // Command also depends on DataSource aDependentProperties.push_back( PROPERTY_ID_COMMAND ); [[fallthrough]]; // ----- Command ----- case PROPERTY_ID_COMMAND: aDependentProperties.push_back( PROPERTY_ID_FILTER ); aDependentProperties.push_back( PROPERTY_ID_SORT ); if ( m_bComponentIsSubForm ) aDependentProperties.push_back( PROPERTY_ID_DETAILFIELDS ); break; // ----- ListSourceType ----- case PROPERTY_ID_LISTSOURCETYPE: if ( !_bFirstTimeInit && m_bHaveListSource ) // available list source values (tables or queries) might have changed _rxInspectorUI->rebuildPropertyUI( PROPERTY_LISTSOURCE ); aDependentProperties.push_back( PROPERTY_ID_STRINGITEMLIST ); aDependentProperties.push_back( PROPERTY_ID_TYPEDITEMLIST ); aDependentProperties.push_back( PROPERTY_ID_BOUNDCOLUMN ); [[fallthrough]]; // ----- StringItemList ----- case PROPERTY_ID_STRINGITEMLIST: aDependentProperties.push_back( PROPERTY_ID_TYPEDITEMLIST ); aDependentProperties.push_back( PROPERTY_ID_SELECTEDITEMS ); aDependentProperties.push_back( PROPERTY_ID_DEFAULT_SELECT_SEQ ); break; // ----- ListSource ----- case PROPERTY_ID_LISTSOURCE: aDependentProperties.push_back( PROPERTY_ID_STRINGITEMLIST ); aDependentProperties.push_back( PROPERTY_ID_TYPEDITEMLIST ); break; // ----- DataField ----- case PROPERTY_ID_CONTROLSOURCE: { OUString sControlSource; _rNewValue >>= sControlSource; if ( impl_componentHasProperty_throw( PROPERTY_FILTERPROPOSAL ) ) _rxInspectorUI->enablePropertyUI( PROPERTY_FILTERPROPOSAL, !sControlSource.isEmpty() ); if ( impl_componentHasProperty_throw( PROPERTY_EMPTY_IS_NULL ) ) _rxInspectorUI->enablePropertyUI( PROPERTY_EMPTY_IS_NULL, !sControlSource.isEmpty() ); aDependentProperties.push_back( PROPERTY_ID_BOUNDCOLUMN ); aDependentProperties.push_back( PROPERTY_ID_SCALEIMAGE ); aDependentProperties.push_back( PROPERTY_ID_SCALE_MODE ); aDependentProperties.push_back( PROPERTY_ID_INPUT_REQUIRED ); } break; case PROPERTY_ID_EMPTY_IS_NULL: aDependentProperties.push_back( PROPERTY_ID_INPUT_REQUIRED ); break; // ----- SubmitEncoding ----- case PROPERTY_ID_SUBMIT_ENCODING: { FormSubmitEncoding eEncoding = FormSubmitEncoding_URL; if( ! (_rNewValue >>= eEncoding) ) SAL_WARN("extensions.propctrlr", "actuatingPropertyChanged: unable to get property " << PROPERTY_ID_SUBMIT_ENCODING); _rxInspectorUI->enablePropertyUI( PROPERTY_SUBMIT_METHOD, eEncoding == FormSubmitEncoding_URL ); } break; // ----- Repeat ----- case PROPERTY_ID_REPEAT: { bool bIsRepeating = false; if( ! (_rNewValue >>= bIsRepeating) ) SAL_WARN("extensions.propctrlr", "actuatingPropertyChanged: unable to get property " << PROPERTY_ID_REPEAT); _rxInspectorUI->enablePropertyUI( PROPERTY_REPEAT_DELAY, bIsRepeating ); } break; // ----- TabStop ----- case PROPERTY_ID_TABSTOP: { if ( !impl_componentHasProperty_throw( PROPERTY_TABINDEX ) ) break; bool bHasTabStop = false; _rNewValue >>= bHasTabStop; _rxInspectorUI->enablePropertyUI( PROPERTY_TABINDEX, bHasTabStop ); } break; // ----- Border ----- case PROPERTY_ID_BORDER: { sal_Int16 nBordeType = VisualEffect::NONE; if( ! (_rNewValue >>= nBordeType) ) SAL_WARN("extensions.propctrlr", "actuatingPropertyChanged: unable to get property " << PROPERTY_ID_BORDER); _rxInspectorUI->enablePropertyUI( PROPERTY_BORDERCOLOR, nBordeType == VisualEffect::FLAT ); } break; // ----- DropDown ----- case PROPERTY_ID_DROPDOWN: { if ( impl_isSupportedProperty_nothrow( PROPERTY_ID_LINECOUNT ) ) { bool bDropDown = true; _rNewValue >>= bDropDown; _rxInspectorUI->enablePropertyUI( PROPERTY_LINECOUNT, bDropDown ); } } break; // ----- ImageURL ----- case PROPERTY_ID_IMAGE_URL: { if ( impl_isSupportedProperty_nothrow( PROPERTY_ID_IMAGEPOSITION ) ) { OUString sImageURL; if( ! (_rNewValue >>= sImageURL) ) SAL_WARN("extensions.propctrlr", "actuatingPropertyChanged: unable to get property " << PROPERTY_ID_IMAGE_URL); _rxInspectorUI->enablePropertyUI( PROPERTY_IMAGEPOSITION, !sImageURL.isEmpty() ); } aDependentProperties.push_back( PROPERTY_ID_SCALEIMAGE ); aDependentProperties.push_back( PROPERTY_ID_SCALE_MODE ); } break; // ----- ButtonType ----- case PROPERTY_ID_BUTTONTYPE: { FormButtonType eButtonType( FormButtonType_PUSH ); if( ! (_rNewValue >>= eButtonType) ) SAL_WARN("extensions.propctrlr", "actuatingPropertyChanged: unable to get property " << PROPERTY_ID_BUTTONTYPE); _rxInspectorUI->enablePropertyUI( PROPERTY_TARGET_URL, FormButtonType_URL == eButtonType ); [[fallthrough]]; } // ----- TargetURL ----- case PROPERTY_ID_TARGET_URL: aDependentProperties.push_back( PROPERTY_ID_TARGET_FRAME ); break; // case PROPERTY_ID_TARGET_URL // ----- TriState ----- case PROPERTY_ID_TRISTATE: if ( !_bFirstTimeInit ) _rxInspectorUI->rebuildPropertyUI( m_eComponentClass == eFormControl ? PROPERTY_DEFAULT_STATE : PROPERTY_STATE ); break; // case PROPERTY_ID_TRISTATE // ----- DecimalAccuracy ----- case PROPERTY_ID_DECIMAL_ACCURACY: // ----- ShowThousandsSeparator ----- case PROPERTY_ID_SHOWTHOUSANDSEP: { bool bAccuracy = (PROPERTY_ID_DECIMAL_ACCURACY == nActuatingPropId); sal_uInt16 nNewDigits = 0; if ( bAccuracy ) { if( ! (_rNewValue >>= nNewDigits) ) SAL_WARN("extensions.propctrlr", "actuatingPropertyChanged: unable to get property " << PROPERTY_ID_DECIMAL_ACCURACY); } else { bool bUseSep = false; if( ! (_rNewValue >>= bUseSep) ) SAL_WARN("extensions.propctrlr", "actuatingPropertyChanged: unable to get property " << PROPERTY_ID_SHOWTHOUSANDSEP); } // propagate the changes to the min/max/default fields OUString aAffectedProps[] = { PROPERTY_VALUE, PROPERTY_DEFAULT_VALUE, PROPERTY_VALUEMIN, PROPERTY_VALUEMAX }; for (const OUString & aAffectedProp : aAffectedProps) { Reference< XPropertyControl > xControl; try { xControl = _rxInspectorUI->getPropertyControl( aAffectedProp ); } catch( const UnknownPropertyException& ) {} if ( xControl.is() ) { OFormattedNumericControl* pControl = dynamic_cast< OFormattedNumericControl* >( xControl.get() ); DBG_ASSERT( pControl, "FormComponentPropertyHandler::actuatingPropertyChanged: invalid control!" ); if (pControl) { if ( bAccuracy ) pControl->SetDecimalDigits( nNewDigits ); } } } } break; // ----- FormatKey ----- case PROPERTY_ID_FORMATKEY: { FormatDescription aNewDesc; Reference< XNumberFormatsSupplier > xSupplier; if( ! (m_xComponent->getPropertyValue( PROPERTY_FORMATSSUPPLIER ) >>= xSupplier) ) SAL_WARN("extensions.propctrlr", "actuatingPropertyChanged: unable to get property " << PROPERTY_ID_FORMATKEY); Reference< XUnoTunnel > xTunnel( xSupplier, UNO_QUERY ); DBG_ASSERT(xTunnel.is(), "FormComponentPropertyHandler::actuatingPropertyChanged: xTunnel is invalid!"); if ( xTunnel.is() ) { SvNumberFormatsSupplierObj* pSupplier = reinterpret_cast(xTunnel->getSomething(SvNumberFormatsSupplierObj::getUnoTunnelId())); // the same again aNewDesc.pSupplier = pSupplier; if ( !( _rNewValue >>= aNewDesc.nKey ) ) aNewDesc.nKey = 0; // give each control which has to know this an own copy of the description OUString aFormattedPropertyControls[] = { PROPERTY_EFFECTIVE_MIN, PROPERTY_EFFECTIVE_MAX, PROPERTY_EFFECTIVE_DEFAULT, PROPERTY_EFFECTIVE_VALUE }; for (const OUString & aFormattedPropertyControl : aFormattedPropertyControls) { Reference< XPropertyControl > xControl; try { xControl = _rxInspectorUI->getPropertyControl( aFormattedPropertyControl ); } catch( const UnknownPropertyException& ) {} if ( xControl.is() ) { OFormattedNumericControl* pControl = dynamic_cast< OFormattedNumericControl* >( xControl.get() ); DBG_ASSERT( pControl, "FormComponentPropertyHandler::actuatingPropertyChanged: invalid control!" ); if ( pControl ) pControl->SetFormatDescription( aNewDesc ); } } } } break; case PROPERTY_ID_TOGGLE: { bool bIsToggleButton = false; if( ! (_rNewValue >>= bIsToggleButton) ) SAL_WARN("extensions.propctrlr", "actuatingPropertyChanged: unable to get property " << PROPERTY_ID_TOGGLE); _rxInspectorUI->enablePropertyUI( PROPERTY_DEFAULT_STATE, bIsToggleButton ); } break; case -1: throw RuntimeException(); break; default: OSL_FAIL( "FormComponentPropertyHandler::actuatingPropertyChanged: did not register for this property!" ); break; } // switch ( nActuatingPropId ) for (auto const& dependentProperty : aDependentProperties) { if ( impl_isSupportedProperty_nothrow(dependentProperty) ) impl_updateDependentProperty_nothrow(dependentProperty, _rxInspectorUI); } } void FormComponentPropertyHandler::impl_updateDependentProperty_nothrow( PropertyId _nPropId, const Reference< XObjectInspectorUI >& _rxInspectorUI ) const { try { switch ( _nPropId ) { // ----- StringItemList ----- case PROPERTY_ID_STRINGITEMLIST: { ListSourceType eLSType = ListSourceType_VALUELIST; if( ! (impl_getPropertyValue_throw( PROPERTY_LISTSOURCETYPE ) >>= eLSType) ) SAL_WARN("extensions.propctrlr", "impl_updateDependentProperty_nothrow: unable to get property " << PROPERTY_LISTSOURCETYPE); OUString sListSource; { Sequence< OUString > aListSource; Any aListSourceValue( impl_getPropertyValue_throw( PROPERTY_LISTSOURCE ) ); if ( aListSourceValue >>= aListSource ) { if ( aListSource.hasElements() ) sListSource = aListSource[0]; } else if( ! (aListSourceValue >>= sListSource) ) SAL_WARN("extensions.propctrlr", "impl_updateDependentProperty_nothrow: unable to get property " << PROPERTY_LISTSOURCE); } bool bIsEnabled = ( ( eLSType == ListSourceType_VALUELIST ) || ( sListSource.isEmpty() ) ); _rxInspectorUI->enablePropertyUI( PROPERTY_STRINGITEMLIST, bIsEnabled ); } break; // case PROPERTY_ID_STRINGITEMLIST // ----- TypedItemList ----- case PROPERTY_ID_TYPEDITEMLIST: { /* TODO: anything? */ } break; // case PROPERTY_ID_TYPEDITEMLIST // ----- BoundColumn ----- case PROPERTY_ID_BOUNDCOLUMN: { ListSourceType eLSType = ListSourceType_VALUELIST; if( ! (impl_getPropertyValue_throw( PROPERTY_LISTSOURCETYPE ) >>= eLSType) ) SAL_WARN("extensions.propctrlr", "impl_updateDependentProperty_nothrow: unable to get property " << PROPERTY_LISTSOURCETYPE); _rxInspectorUI->enablePropertyUI( PROPERTY_BOUNDCOLUMN, ( eLSType != ListSourceType_VALUELIST ) ); } break; // case PROPERTY_ID_BOUNDCOLUMN // ----- ScaleImage, ScaleMode ----- case PROPERTY_ID_SCALEIMAGE: case PROPERTY_ID_SCALE_MODE: { OUString sControlSource; if ( impl_isSupportedProperty_nothrow( PROPERTY_ID_CONTROLSOURCE ) ) impl_getPropertyValue_throw( PROPERTY_CONTROLSOURCE ) >>= sControlSource; OUString sImageURL; impl_getPropertyValue_throw( PROPERTY_IMAGE_URL ) >>= sImageURL; _rxInspectorUI->enablePropertyUI( impl_getPropertyNameFromId_nothrow( _nPropId ), ( !sControlSource.isEmpty() ) || ( !sImageURL.isEmpty() ) ); } break; // case PROPERTY_ID_SCALEIMAGE, PROPERTY_ID_SCALE_MODE // ----- InputRequired ----- case PROPERTY_ID_INPUT_REQUIRED: { OUString sControlSource; if( ! (impl_getPropertyValue_throw( PROPERTY_CONTROLSOURCE ) >>= sControlSource) ) SAL_WARN("extensions.propctrlr", "impl_updateDependentProperty_nothrow: unable to get property " << PROPERTY_CONTROLSOURCE); bool bEmptyIsNULL = false; bool bHasEmptyIsNULL = impl_componentHasProperty_throw( PROPERTY_EMPTY_IS_NULL ); if ( bHasEmptyIsNULL ) if( ! (impl_getPropertyValue_throw( PROPERTY_EMPTY_IS_NULL ) >>= bEmptyIsNULL) ) SAL_WARN("extensions.propctrlr", "impl_updateDependentProperty_nothrow: unable to get property " << PROPERTY_EMPTY_IS_NULL); // if the control is not bound to a DB field, there is no sense in having the "Input required" // property // Also, if an empty input of this control are *not* written as NULL, but as empty strings, // then "Input required" does not make sense, too (since there's always an input, even if the control // is empty). _rxInspectorUI->enablePropertyUI( PROPERTY_INPUT_REQUIRED, ( !sControlSource.isEmpty() ) && ( !bHasEmptyIsNULL || bEmptyIsNULL ) ); } break; // ----- SelectedItems, DefaultSelection ----- case PROPERTY_ID_SELECTEDITEMS: case PROPERTY_ID_DEFAULT_SELECT_SEQ: { Sequence< OUString > aEntries; impl_getPropertyValue_throw( PROPERTY_STRINGITEMLIST ) >>= aEntries; bool isEnabled = aEntries.hasElements(); if ( ( m_nClassId == FormComponentType::LISTBOX ) && ( m_eComponentClass == eFormControl ) ) { ListSourceType eLSType = ListSourceType_VALUELIST; impl_getPropertyValue_throw( PROPERTY_LISTSOURCETYPE ) >>= eLSType; isEnabled &= ( eLSType == ListSourceType_VALUELIST ); } _rxInspectorUI->enablePropertyUIElements( impl_getPropertyNameFromId_nothrow( _nPropId ), PropertyLineElement::PrimaryButton, isEnabled ); } break; // case PROPERTY_ID_DEFAULT_SELECT_SEQ // ----- TargetFrame ------ case PROPERTY_ID_TARGET_FRAME: { OUString sTargetURL; impl_getPropertyValue_throw( PROPERTY_TARGET_URL ) >>= sTargetURL; FormButtonType eButtonType( FormButtonType_URL ); if ( 0 != m_nClassId ) { if( ! (impl_getPropertyValue_throw( PROPERTY_BUTTONTYPE ) >>= eButtonType) ) SAL_WARN("extensions.propctrlr", "impl_updateDependentProperty_nothrow: unable to get property " << PROPERTY_BUTTONTYPE); } // if m_nClassId is 0, then we're inspecting a form. In this case, eButtonType is always // FormButtonType_URL here _rxInspectorUI->enablePropertyUI( PROPERTY_TARGET_FRAME, ( eButtonType == FormButtonType_URL ) && ( !sTargetURL.isEmpty() ) ); } break; // ----- Order ------ case PROPERTY_ID_SORT: // ----- Filter ------ case PROPERTY_ID_FILTER: { Reference< XConnection > xConnection; bool bAllowEmptyDS = ::dbtools::isEmbeddedInDatabase( m_xComponent, xConnection ); // if there's no escape processing, we cannot enter any values for this property bool bDoEscapeProcessing( false ); impl_getPropertyValue_throw( PROPERTY_ESCAPE_PROCESSING ) >>= bDoEscapeProcessing; _rxInspectorUI->enablePropertyUI( impl_getPropertyNameFromId_nothrow( _nPropId ), bDoEscapeProcessing ); // also care for the browse button - enabled if we have escape processing, and a valid // data source signature _rxInspectorUI->enablePropertyUIElements( impl_getPropertyNameFromId_nothrow( _nPropId ), PropertyLineElement::PrimaryButton, impl_hasValidDataSourceSignature_nothrow( m_xComponent, bAllowEmptyDS ) && bDoEscapeProcessing ); } break; // case PROPERTY_ID_FILTER: // ----- Command ----- case PROPERTY_ID_COMMAND: { sal_Int32 nCommandType( CommandType::COMMAND ); if( ! (impl_getPropertyValue_throw( PROPERTY_COMMANDTYPE ) >>= nCommandType) ) SAL_WARN("extensions.propctrlr", "impl_updateDependentProperty_nothrow: unable to get property " << PROPERTY_COMMANDTYPE); impl_ensureRowsetConnection_nothrow(); Reference< XConnection > xConnection = m_xRowSetConnection.getTyped(); bool bAllowEmptyDS = false; if ( !xConnection.is() ) bAllowEmptyDS = ::dbtools::isEmbeddedInDatabase( m_xComponent, xConnection ); bool doEnable = ( nCommandType == CommandType::COMMAND ) && ( m_xRowSetConnection.is() || xConnection.is() || impl_hasValidDataSourceSignature_nothrow( m_xComponent, bAllowEmptyDS) ); _rxInspectorUI->enablePropertyUIElements( PROPERTY_COMMAND, PropertyLineElement::PrimaryButton, doEnable ); } break; // case PROPERTY_ID_COMMAND // ----- DetailFields ----- case PROPERTY_ID_DETAILFIELDS: { Reference< XConnection > xConnection; bool bAllowEmptyDS = ::dbtools::isEmbeddedInDatabase( m_xComponent, xConnection ); // both our current form, and its parent form, need to have a valid // data source signature bool bDoEnableMasterDetailFields = impl_hasValidDataSourceSignature_nothrow( m_xComponent, bAllowEmptyDS ) && impl_hasValidDataSourceSignature_nothrow( Reference< XPropertySet >( m_xObjectParent, UNO_QUERY ), bAllowEmptyDS ); // in opposite to the other properties, here in real *two* properties are // affected _rxInspectorUI->enablePropertyUIElements( PROPERTY_DETAILFIELDS, PropertyLineElement::PrimaryButton, bDoEnableMasterDetailFields ); _rxInspectorUI->enablePropertyUIElements( PROPERTY_MASTERFIELDS, PropertyLineElement::PrimaryButton, bDoEnableMasterDetailFields ); } break; default: OSL_FAIL( "FormComponentPropertyHandler::impl_updateDependentProperty_nothrow: unexpected property to update!" ); break; } // switch } catch( const Exception& ) { TOOLS_WARN_EXCEPTION( "extensions.propctrlr", "FormComponentPropertyHandler::impl_updateDependentProperty_nothrow" ); } } void SAL_CALL FormComponentPropertyHandler::disposing() { PropertyHandlerComponent::disposing(); if ( m_xCommandDesigner.is() && m_xCommandDesigner->isActive() ) m_xCommandDesigner->dispose(); } sal_Bool SAL_CALL FormComponentPropertyHandler::suspend( sal_Bool _bSuspend ) { ::osl::MutexGuard aGuard( m_aMutex ); if ( _bSuspend ) if ( m_xCommandDesigner.is() && m_xCommandDesigner->isActive() ) return m_xCommandDesigner->suspend(); return true; } void FormComponentPropertyHandler::onNewComponent() { PropertyHandlerComponent::onNewComponent(); if ( !m_xComponentPropertyInfo.is() && m_xComponent.is() ) throw NullPointerException(); m_xPropertyState.set( m_xComponent, UNO_QUERY ); m_eComponentClass = eUnknown; m_bComponentIsSubForm = m_bHaveListSource = m_bHaveCommand = false; m_nClassId = 0; try { // component class m_eComponentClass = eUnknown; if ( impl_componentHasProperty_throw( PROPERTY_WIDTH ) && impl_componentHasProperty_throw( PROPERTY_HEIGHT ) && impl_componentHasProperty_throw( PROPERTY_POSITIONX ) && impl_componentHasProperty_throw( PROPERTY_POSITIONY ) && impl_componentHasProperty_throw( PROPERTY_STEP ) && impl_componentHasProperty_throw( PROPERTY_TABINDEX ) ) { m_eComponentClass = eDialogControl; } else { m_eComponentClass = eFormControl; } // (database) sub form? Reference< XForm > xAsForm( m_xComponent, UNO_QUERY ); if ( xAsForm.is() ) { Reference< XForm > xFormsParent( xAsForm->getParent(), css::uno::UNO_QUERY ); m_bComponentIsSubForm = xFormsParent.is(); } // ClassId Reference< XChild > xCompAsChild( m_xComponent, UNO_QUERY ); if ( xCompAsChild.is() ) m_xObjectParent = xCompAsChild->getParent(); // ClassId impl_classifyControlModel_throw(); } catch( const RuntimeException& ) { throw; } catch( const Exception& ) { TOOLS_WARN_EXCEPTION( "extensions.propctrlr", "FormComponentPropertyHandler::onNewComponent" ); } } void FormComponentPropertyHandler::impl_classifyControlModel_throw( ) { if ( impl_componentHasProperty_throw( PROPERTY_CLASSID ) ) { if( ! (m_xComponent->getPropertyValue( PROPERTY_CLASSID ) >>= m_nClassId) ) SAL_WARN("extensions.propctrlr", "impl_classifyControlModel_throw: unable to get property " << PROPERTY_CLASSID); } else if ( eDialogControl == m_eComponentClass ) { Reference< XServiceInfo > xServiceInfo( m_xComponent, UNO_QUERY ); if ( xServiceInfo.is() ) { // it's a control model, and can tell about it's supported services m_nClassId = FormComponentType::CONTROL; const char* aControlModelServiceNames[] = { "UnoControlButtonModel", "UnoControlCheckBoxModel", "UnoControlComboBoxModel", "UnoControlCurrencyFieldModel", "UnoControlDateFieldModel", "UnoControlEditModel", "UnoControlFileControlModel", "UnoControlFixedTextModel", "UnoControlGroupBoxModel", "UnoControlImageControlModel", "UnoControlListBoxModel", "UnoControlNumericFieldModel", "UnoControlPatternFieldModel", "UnoControlRadioButtonModel", "UnoControlScrollBarModel", "UnoControlSpinButtonModel", "UnoControlTimeFieldModel", "UnoControlFixedLineModel", "UnoControlFormattedFieldModel", "UnoControlProgressBarModel" }; const sal_Int16 nClassIDs[] = { FormComponentType::COMMANDBUTTON, FormComponentType::CHECKBOX, FormComponentType::COMBOBOX, FormComponentType::CURRENCYFIELD, FormComponentType::DATEFIELD, FormComponentType::TEXTFIELD, FormComponentType::FILECONTROL, FormComponentType::FIXEDTEXT, FormComponentType::GROUPBOX, FormComponentType::IMAGECONTROL, FormComponentType::LISTBOX, FormComponentType::NUMERICFIELD, FormComponentType::PATTERNFIELD, FormComponentType::RADIOBUTTON, FormComponentType::SCROLLBAR, FormComponentType::SPINBUTTON, FormComponentType::TIMEFIELD, ControlType::FIXEDLINE, ControlType::FORMATTEDFIELD, ControlType::PROGRESSBAR }; sal_Int32 nKnownControlTypes = SAL_N_ELEMENTS( aControlModelServiceNames ); OSL_ENSURE( nKnownControlTypes == SAL_N_ELEMENTS( nClassIDs ), "FormComponentPropertyHandler::impl_classifyControlModel_throw: inconsistence" ); for ( sal_Int32 i = 0; i < nKnownControlTypes; ++i ) { OUString sServiceName = "com.sun.star.awt." + OUString::createFromAscii( aControlModelServiceNames[ i ] ); if ( xServiceInfo->supportsService( sServiceName ) ) { m_nClassId = nClassIDs[ i ]; break; } } } } } void FormComponentPropertyHandler::impl_normalizePropertyValue_nothrow( Any& _rValue, PropertyId _nPropId ) const { switch ( _nPropId ) { case PROPERTY_ID_TABSTOP: if ( !_rValue.hasValue() ) { switch ( m_nClassId ) { case FormComponentType::COMMANDBUTTON: case FormComponentType::RADIOBUTTON: case FormComponentType::CHECKBOX: case FormComponentType::TEXTFIELD: case FormComponentType::LISTBOX: case FormComponentType::COMBOBOX: case FormComponentType::FILECONTROL: case FormComponentType::DATEFIELD: case FormComponentType::TIMEFIELD: case FormComponentType::NUMERICFIELD: case ControlType::FORMATTEDFIELD: case FormComponentType::CURRENCYFIELD: case FormComponentType::PATTERNFIELD: _rValue <<= true; break; default: _rValue <<= false; break; } } break; } } bool FormComponentPropertyHandler::isReportModel() const { Reference xModel(impl_getContextDocument_nothrow()); Reference xReportDef(xModel, css::uno::UNO_QUERY); return xReportDef.is(); } bool FormComponentPropertyHandler::impl_shouldExcludeProperty_nothrow( const Property& _rProperty ) const { OSL_ENSURE( _rProperty.Handle == m_pInfoService->getPropertyId( _rProperty.Name ), "FormComponentPropertyHandler::impl_shouldExcludeProperty_nothrow: inconsistency in the property!" ); if ( _rProperty.Handle == PROPERTY_ID_CONTROLLABEL ) // prevent that this is caught below return false; if ( ( _rProperty.Type.getTypeClass() == TypeClass_INTERFACE ) || ( _rProperty.Type.getTypeClass() == TypeClass_UNKNOWN ) ) return true; if ( ( _rProperty.Attributes & PropertyAttribute::TRANSIENT ) && ( m_eComponentClass != eDialogControl ) ) // strange enough, dialog controls declare a lot of their properties as transient return true; if ( _rProperty.Attributes & PropertyAttribute::READONLY ) return true; switch ( _rProperty.Handle ) { case PROPERTY_ID_MASTERFIELDS: case PROPERTY_ID_DETAILFIELDS: if ( !m_bComponentIsSubForm ) // no master and detail fields for forms which are no sub forms return true; break; case PROPERTY_ID_DATASOURCE: { // don't show DataSource if the component is part of an embedded form document Reference< XConnection > xConn; if ( isEmbeddedInDatabase( m_xComponent, xConn ) ) return true; } break; case PROPERTY_ID_TEXT: // don't show the "Text" property of formatted fields if ( ControlType::FORMATTEDFIELD == m_nClassId ) return true; break; case PROPERTY_ID_FORMATKEY: case PROPERTY_ID_EFFECTIVE_MIN: case PROPERTY_ID_EFFECTIVE_MAX: case PROPERTY_ID_EFFECTIVE_DEFAULT: case PROPERTY_ID_EFFECTIVE_VALUE: // only if the set has a formats supplier, too if ( !impl_componentHasProperty_throw( PROPERTY_FORMATSSUPPLIER ) ) return true; // (form) date and time fields also have a formats supplier, but the format itself // is reflected in another property if ( ( FormComponentType::DATEFIELD == m_nClassId ) || ( FormComponentType::TIMEFIELD == m_nClassId ) ) return true; break; case PROPERTY_ID_SCALEIMAGE: if ( impl_componentHasProperty_throw( PROPERTY_SCALE_MODE ) ) // ScaleImage is superseded by ScaleMode return true; break; case PROPERTY_ID_WRITING_MODE: if ( !SvtCTLOptions::IsCTLFontEnabled() ) return true; break; } sal_uInt32 nPropertyUIFlags = m_pInfoService->getPropertyUIFlags( _rProperty.Handle ); // don't show experimental properties unless allowed to do so if ( ( nPropertyUIFlags & PROP_FLAG_EXPERIMENTAL ) != 0 ) return true; // no data properties if no Base is installed. if ( ( nPropertyUIFlags & PROP_FLAG_DATA_PROPERTY ) != 0 ) if (!SvtModuleOptions().IsDataBaseInstalled()) return true; if ((nPropertyUIFlags & PROP_FLAG_REPORT_INVISIBLE) != 0 && isReportModel()) return true; return false; } Reference< XRowSet > FormComponentPropertyHandler::impl_getRowSet_throw( ) const { Reference< XRowSet > xRowSet = m_xRowSet; if ( !xRowSet.is() ) { xRowSet.set( m_xComponent, UNO_QUERY ); if ( !xRowSet.is() ) { xRowSet.set( m_xObjectParent, UNO_QUERY ); if ( !xRowSet.is() ) { // are we inspecting a grid column? if (Reference< XGridColumnFactory >( m_xObjectParent, UNO_QUERY) .is()) { // yes Reference< XChild > xParentAsChild( m_xObjectParent, UNO_QUERY ); if ( xParentAsChild.is() ) xRowSet.set( xParentAsChild->getParent(), UNO_QUERY ); } } if ( !xRowSet.is() ) xRowSet = m_xRowSet; } DBG_ASSERT( xRowSet.is(), "FormComponentPropertyHandler::impl_getRowSet_throw: could not obtain the rowset for the introspectee!" ); } return xRowSet; } Reference< XRowSet > FormComponentPropertyHandler::impl_getRowSet_nothrow( ) const { Reference< XRowSet > xReturn; try { xReturn = impl_getRowSet_throw(); } catch( const Exception& ) { TOOLS_WARN_EXCEPTION( "extensions.propctrlr", "FormComponentPropertyHandler::impl_getRowSet_nothrow" ); } return xReturn; } void FormComponentPropertyHandler::impl_initFieldList_nothrow( std::vector< OUString >& _rFieldNames ) const { clearContainer( _rFieldNames ); try { weld::WaitObject aWaitCursor(impl_getDefaultDialogFrame_nothrow()); // get the form of the control we're inspecting Reference< XPropertySet > xFormSet( impl_getRowSet_throw(), UNO_QUERY ); if ( !xFormSet.is() ) return; OUString sObjectName; if( ! (xFormSet->getPropertyValue( PROPERTY_COMMAND ) >>= sObjectName) ) SAL_WARN("extensions.propctrlr", "impl_initFieldList_nothrow: unable to get property " << PROPERTY_COMMAND); // when there is no command we don't need to ask for columns if ( !sObjectName.isEmpty() && impl_ensureRowsetConnection_nothrow() ) { OUString aDatabaseName; if( ! (xFormSet->getPropertyValue( PROPERTY_DATASOURCE ) >>= aDatabaseName) ) SAL_WARN("extensions.propctrlr", "impl_initFieldList_nothrow: unable to get property " << PROPERTY_DATASOURCE); sal_Int32 nObjectType = CommandType::COMMAND; if( ! (xFormSet->getPropertyValue( PROPERTY_COMMANDTYPE ) >>= nObjectType) ) SAL_WARN("extensions.propctrlr", "impl_initFieldList_nothrow: unable to get property " << PROPERTY_COMMANDTYPE); const Sequence aNames = ::dbtools::getFieldNamesByCommandDescriptor( m_xRowSetConnection, nObjectType, sObjectName ); _rFieldNames.insert( _rFieldNames.end(), aNames.begin(), aNames.end() ); } } catch (const Exception&) { TOOLS_WARN_EXCEPTION( "extensions.propctrlr", "FormComponentPropertyHandler::impl_initFieldList_nothrow" ); } } void FormComponentPropertyHandler::impl_displaySQLError_nothrow( const ::dbtools::SQLExceptionInfo& _rErrorDescriptor ) const { auto pTopLevel = impl_getDefaultDialogFrame_nothrow(); ::dbtools::showError(_rErrorDescriptor, pTopLevel ? pTopLevel->GetXWindow() : nullptr, m_xContext); } bool FormComponentPropertyHandler::impl_ensureRowsetConnection_nothrow() const { if ( !m_xRowSetConnection.is() ) { uno::Reference xConnection; Any any = m_xContext->getValueByName( u"ActiveConnection"_ustr ); any >>= xConnection; m_xRowSetConnection.reset(xConnection,::dbtools::SharedConnection::NoTakeOwnership); } if ( m_xRowSetConnection.is() ) return true; Reference< XRowSet > xRowSet( impl_getRowSet_throw() ); Reference< XPropertySet > xRowSetProps( xRowSet, UNO_QUERY ); // connect the row set - this is delegated to elsewhere - while observing errors SQLExceptionInfo aError; try { if ( xRowSetProps.is() ) { weld::WaitObject aWaitCursor(impl_getDefaultDialogFrame_nothrow()); m_xRowSetConnection = ::dbtools::ensureRowSetConnection( xRowSet, m_xContext, nullptr ); } } catch ( const SQLException& ) { aError = SQLExceptionInfo( ::cppu::getCaughtException() ); } catch ( const WrappedTargetException& e ) { aError = SQLExceptionInfo( e.TargetException ); } catch ( const Exception& ) { DBG_UNHANDLED_EXCEPTION("extensions.propctrlr"); } // report errors, if necessary if ( aError.isValid() ) { OUString sDataSourceName; try { xRowSetProps->getPropertyValue( PROPERTY_DATASOURCE ) >>= sDataSourceName; } catch( const Exception& ) { TOOLS_WARN_EXCEPTION( "extensions.propctrlr", "FormComponentPropertyHandler::impl_ensureRowsetConnection_nothrow: caught an exception during error handling!" ); } // additional info about what happened INetURLObject aParser( sDataSourceName ); if ( aParser.GetProtocol() != INetProtocol::NotValid ) sDataSourceName = aParser.getBase( INetURLObject::LAST_SEGMENT, true, INetURLObject::DecodeMechanism::WithCharset ); OUString sInfo(PcrRes(RID_STR_UNABLETOCONNECT).replaceAll("$name$", sDataSourceName)); SQLContext aContext(sInfo, {}, {}, 0, aError.get(), {}); impl_displaySQLError_nothrow( aContext ); } return m_xRowSetConnection.is(); } void FormComponentPropertyHandler::impl_describeCursorSource_nothrow( LineDescriptor& _out_rProperty, const Reference< XPropertyControlFactory >& _rxControlFactory ) const { try { weld::WaitObject aWaitCursor(impl_getDefaultDialogFrame_nothrow()); // Set the UI data _out_rProperty.DisplayName = m_pInfoService->getPropertyTranslation( PROPERTY_ID_COMMAND ); _out_rProperty.HelpURL = HelpIdUrl::getHelpURL( m_pInfoService->getPropertyHelpId( PROPERTY_ID_COMMAND ) ); _out_rProperty.PrimaryButtonId = UID_PROP_DLG_SQLCOMMAND; sal_Int32 nCommandType = CommandType::COMMAND; impl_getPropertyValue_throw( PROPERTY_COMMANDTYPE ) >>= nCommandType; switch ( nCommandType ) { case CommandType::TABLE: case CommandType::QUERY: { std::vector< OUString > aNames; if ( impl_ensureRowsetConnection_nothrow() ) { if ( nCommandType == CommandType::TABLE ) impl_fillTableNames_throw( aNames ); else impl_fillQueryNames_throw( aNames ); } _out_rProperty.Control = PropertyHandlerHelper::createComboBoxControl( _rxControlFactory, std::move(aNames), true ); } break; default: _out_rProperty.Control = _rxControlFactory->createPropertyControl( PropertyControlType::MultiLineTextField, false ); break; } } catch (const Exception&) { TOOLS_WARN_EXCEPTION( "extensions.propctrlr", "FormComponentPropertyHandler::impl_describeCursorSource_nothrow"); } } void FormComponentPropertyHandler::impl_fillTableNames_throw( std::vector< OUString >& _out_rNames ) const { OSL_PRECOND( m_xRowSetConnection.is(), "FormComponentPropertyHandler::impl_fillTableNames_throw: need a connection!" ); _out_rNames.resize( 0 ); Reference< XTablesSupplier > xSupplyTables( m_xRowSetConnection, UNO_QUERY ); Reference< XNameAccess > xTableNames; if ( xSupplyTables.is() ) xTableNames = xSupplyTables->getTables(); DBG_ASSERT( xTableNames.is(), "FormComponentPropertyHandler::impl_fillTableNames_throw: no way to obtain the tables of the connection!" ); if ( !xTableNames.is() ) return; const Sequence aNames = xTableNames->getElementNames(); _out_rNames.insert( _out_rNames.end(), aNames.begin(), aNames.end() ); } void FormComponentPropertyHandler::impl_fillQueryNames_throw( std::vector< OUString >& _out_rNames ) const { OSL_PRECOND( m_xRowSetConnection.is(), "FormComponentPropertyHandler::impl_fillQueryNames_throw: need a connection!" ); _out_rNames.resize( 0 ); Reference< XQueriesSupplier > xSupplyQueries( m_xRowSetConnection, UNO_QUERY ); Reference< XNameAccess > xQueryNames; if ( xSupplyQueries.is() ) { xQueryNames = xSupplyQueries->getQueries(); impl_fillQueryNames_throw(xQueryNames,_out_rNames); } } void FormComponentPropertyHandler::impl_fillQueryNames_throw( const Reference< XNameAccess >& _xQueryNames,std::vector< OUString >& _out_rNames,std::u16string_view _sName ) const { DBG_ASSERT( _xQueryNames.is(), "FormComponentPropertyHandler::impl_fillQueryNames_throw: no way to obtain the queries of the connection!" ); if ( !_xQueryNames.is() ) return; bool bAdd = !_sName.empty(); const Sequence aQueryNames =_xQueryNames->getElementNames(); for ( const OUString& rQueryName : aQueryNames ) { OUStringBuffer sTemp; if ( bAdd ) { sTemp.append(OUString::Concat(_sName) + "/"); } sTemp.append(rQueryName); Reference< XNameAccess > xSubQueries(_xQueryNames->getByName(rQueryName),UNO_QUERY); if ( xSubQueries.is() ) impl_fillQueryNames_throw(xSubQueries,_out_rNames,sTemp); else _out_rNames.push_back( sTemp.makeStringAndClear() ); } } void FormComponentPropertyHandler::impl_describeListSourceUI_throw( LineDescriptor& _out_rDescriptor, const Reference< XPropertyControlFactory >& _rxControlFactory ) const { OSL_PRECOND( m_xComponent.is(), "FormComponentPropertyHandler::impl_describeListSourceUI_throw: no component!" ); // read out ListSourceTypes Any aListSourceType( m_xComponent->getPropertyValue( PROPERTY_LISTSOURCETYPE ) ); sal_Int32 nListSourceType = sal_Int32(ListSourceType_VALUELIST); ::cppu::enum2int( nListSourceType, aListSourceType ); ListSourceType eListSourceType = static_cast(nListSourceType); _out_rDescriptor.DisplayName = m_pInfoService->getPropertyTranslation( PROPERTY_ID_LISTSOURCE ); _out_rDescriptor.HelpURL = HelpIdUrl::getHelpURL( m_pInfoService->getPropertyHelpId( PROPERTY_ID_LISTSOURCE ) ); // set enums switch( eListSourceType ) { case ListSourceType_VALUELIST: _out_rDescriptor.Control = _rxControlFactory->createPropertyControl( PropertyControlType::StringListField, false ); break; case ListSourceType_TABLEFIELDS: case ListSourceType_TABLE: case ListSourceType_QUERY: { std::vector< OUString > aListEntries; if ( impl_ensureRowsetConnection_nothrow() ) { if ( eListSourceType == ListSourceType_QUERY ) impl_fillQueryNames_throw( aListEntries ); else impl_fillTableNames_throw( aListEntries ); } _out_rDescriptor.Control = PropertyHandlerHelper::createComboBoxControl( _rxControlFactory, std::move(aListEntries), false ); } break; case ListSourceType_SQL: case ListSourceType_SQLPASSTHROUGH: impl_ensureRowsetConnection_nothrow(); _out_rDescriptor.HasPrimaryButton = m_xRowSetConnection.is(); break; default: break; } } bool FormComponentPropertyHandler::impl_dialogListSelection_nothrow( const OUString& _rProperty, ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { OSL_PRECOND(m_pInfoService, "FormComponentPropertyHandler::impl_dialogListSelection_" "nothrow: no property meta data!"); OUString sPropertyUIName( m_pInfoService->getPropertyTranslation( m_pInfoService->getPropertyId( _rProperty ) ) ); ListSelectionDialog aDialog(impl_getDefaultDialogFrame_nothrow(), m_xComponent, _rProperty, sPropertyUIName); _rClearBeforeDialog.clear(); return ( RET_OK == aDialog.run() ); } bool FormComponentPropertyHandler::impl_dialogFilterOrSort_nothrow( bool _bFilter, OUString& _out_rSelectedClause, ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { OSL_PRECOND( Reference< XRowSet >( m_xComponent, UNO_QUERY ).is(), "FormComponentPropertyHandler::impl_dialogFilterOrSort_nothrow: to be called for forms only!" ); _out_rSelectedClause.clear(); bool bSuccess = false; SQLExceptionInfo aErrorInfo; try { if ( !impl_ensureRowsetConnection_nothrow() ) return false; // get a composer for the statement which the form is currently based on Reference< XSingleSelectQueryComposer > xComposer( ::dbtools::getCurrentSettingsComposer( m_xComponent, m_xContext, nullptr ) ); OSL_ENSURE( xComposer.is(), "FormComponentPropertyHandler::impl_dialogFilterOrSort_nothrow: could not obtain a composer!" ); if ( !xComposer.is() ) return false; OUString sPropertyUIName( m_pInfoService->getPropertyTranslation( _bFilter ? PROPERTY_ID_FILTER : PROPERTY_ID_SORT ) ); // create the dialog Reference< XExecutableDialog > xDialog; if ( _bFilter) { xDialog.set( sdb::FilterDialog::createDefault(m_xContext) ); } else { xDialog.set( sdb::OrderDialog::createDefault(m_xContext) ); } // initialize the dialog Reference< XPropertySet > xDialogProps( xDialog, UNO_QUERY_THROW ); xDialogProps->setPropertyValue(u"QueryComposer"_ustr, Any( xComposer ) ); xDialogProps->setPropertyValue(u"RowSet"_ustr, Any( m_xComponent ) ); if (auto pTopLevel = impl_getDefaultDialogFrame_nothrow()) xDialogProps->setPropertyValue(u"ParentWindow"_ustr, Any(pTopLevel->GetXWindow())); xDialogProps->setPropertyValue(u"Title"_ustr, Any( sPropertyUIName ) ); _rClearBeforeDialog.clear(); bSuccess = ( xDialog->execute() != 0 ); if ( bSuccess ) _out_rSelectedClause = _bFilter ? xComposer->getFilter() : xComposer->getOrder(); } catch (const SQLContext& e) { aErrorInfo = e; } catch (const SQLWarning& e) { aErrorInfo = e; } catch (const SQLException& e) { aErrorInfo = e; } catch( const Exception& ) { TOOLS_WARN_EXCEPTION( "extensions.propctrlr", "FormComponentPropertyHandler::impl_dialogFilterOrSort_nothrow" ); } if ( aErrorInfo.isValid() ) impl_displaySQLError_nothrow( aErrorInfo ); return bSuccess; } bool FormComponentPropertyHandler::impl_dialogLinkedFormFields_nothrow( ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { Reference< XForm > xDetailForm( m_xComponent, UNO_QUERY ); Reference< XForm > xMasterForm( m_xObjectParent, UNO_QUERY ); uno::Reference xMasterProp(m_xObjectParent,uno::UNO_QUERY); OSL_PRECOND( xDetailForm.is() && xMasterForm.is(), "FormComponentPropertyHandler::impl_dialogLinkedFormFields_nothrow: no forms!" ); if ( !xDetailForm.is() || !xMasterForm.is() ) return false; FormLinkDialog aDialog(impl_getDefaultDialogFrame_nothrow(), m_xComponent, xMasterProp, m_xContext); _rClearBeforeDialog.clear(); return ( RET_OK == aDialog.run() ); } bool FormComponentPropertyHandler::impl_dialogFormatting_nothrow( Any& _out_rNewValue, ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { bool bChanged = false; try { // create the itemset for the dialog SfxItemSet aCoreSet( SfxGetpApp()->GetPool(), svl::Items< SID_ATTR_NUMBERFORMAT_VALUE, SID_ATTR_NUMBERFORMAT_INFO>); // ripped this somewhere ... don't understand it :( // get the number formats supplier Reference< XNumberFormatsSupplier > xSupplier; m_xComponent->getPropertyValue( PROPERTY_FORMATSSUPPLIER ) >>= xSupplier; DBG_ASSERT(xSupplier.is(), "FormComponentPropertyHandler::impl_dialogFormatting_nothrow: invalid call !" ); Reference< XUnoTunnel > xTunnel( xSupplier, UNO_QUERY_THROW ); SvNumberFormatsSupplierObj* pSupplier = reinterpret_cast< SvNumberFormatsSupplierObj* >( xTunnel->getSomething( SvNumberFormatsSupplierObj::getUnoTunnelId() ) ); assert(pSupplier && "FormComponentPropertyHandler::impl_dialogFormatting_nothrow: invalid call !"); sal_Int32 nFormatKey = 0; impl_getPropertyValue_throw( PROPERTY_FORMATKEY ) >>= nFormatKey; aCoreSet.Put( SfxUInt32Item( SID_ATTR_NUMBERFORMAT_VALUE, nFormatKey ) ); SvNumberFormatter* pFormatter = pSupplier->GetNumberFormatter(); double dPreviewVal = OFormatSampleControl::getPreviewValue(pFormatter,nFormatKey); SvxNumberInfoItem aFormatter( pFormatter, dPreviewVal, PcrRes(RID_STR_TEXT_FORMAT), SID_ATTR_NUMBERFORMAT_INFO ); aCoreSet.Put( aFormatter ); // a tab dialog with a single page SfxSingleTabDialogController aDialog(impl_getDefaultDialogFrame_nothrow(), &aCoreSet, u"cui/ui/formatnumberdialog.ui"_ustr, u"FormatNumberDialog"_ustr); SvxAbstractDialogFactory* pFact = SvxAbstractDialogFactory::Create(); ::CreateTabPage fnCreatePage = pFact->GetTabPageCreatorFunc( RID_SVXPAGE_NUMBERFORMAT ); if ( !fnCreatePage ) throw RuntimeException(); // caught below aDialog.SetTabPage((*fnCreatePage)(aDialog.get_content_area(), &aDialog, &aCoreSet)); _rClearBeforeDialog.clear(); if ( RET_OK == aDialog.run() ) { const SfxItemSet* pResult = aDialog.GetOutputItemSet(); if (const SvxNumberInfoItem* pInfoItem = pResult->GetItem( SID_ATTR_NUMBERFORMAT_INFO )) { for (sal_uInt32 key : pInfoItem->GetDelFormats()) pFormatter->DeleteEntry(key); } if ( const SfxUInt32Item* pItem = pResult->GetItemIfSet( SID_ATTR_NUMBERFORMAT_VALUE, false ) ) { _out_rNewValue <<= static_cast( pItem->GetValue() ); bChanged = true; } } } catch( const Exception& ) { TOOLS_WARN_EXCEPTION( "extensions.propctrlr", "FormComponentPropertyHandler::impl_dialogFormatting_nothrow" ); } return bChanged; } bool FormComponentPropertyHandler::impl_browseForImage_nothrow( Any& _out_rNewValue, ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { bool bIsLink = true;// reflect the legacy behavior OUString aStrTrans = m_pInfoService->getPropertyTranslation( PROPERTY_ID_IMAGE_URL ); weld::Window* pWin = impl_getDefaultDialogFrame_nothrow(); ::sfx2::FileDialogHelper aFileDlg( ui::dialogs::TemplateDescription::FILEOPEN_LINK_PREVIEW, FileDialogFlags::Graphic, pWin); aFileDlg.SetContext(sfx2::FileDialogHelper::FormsInsertImage); aFileDlg.SetTitle(aStrTrans); // non-linked images ( e.g. those located in the document // stream ) only if document is available bool bHandleNonLink; { Reference< XModel > xModel( impl_getContextDocument_nothrow() ); bHandleNonLink = xModel.is(); // Not implemented in reports if (bHandleNonLink) { Reference< XReportDefinition > xReportDef( xModel, css::uno::UNO_QUERY ); bHandleNonLink = !xReportDef.is(); } } Reference< XFilePickerControlAccess > xController(aFileDlg.GetFilePicker(), UNO_QUERY); DBG_ASSERT(xController.is(), "FormComponentPropertyHandler::impl_browseForImage_nothrow: missing the controller interface on the file picker!"); if (xController.is()) { // do a preview by default xController->setValue(ExtendedFilePickerElementIds::CHECKBOX_PREVIEW, 0, css::uno::Any(true)); xController->setValue(ExtendedFilePickerElementIds::CHECKBOX_LINK, 0, css::uno::Any(bIsLink)); xController->enableControl(ExtendedFilePickerElementIds::CHECKBOX_LINK, bHandleNonLink ); } OUString sCurValue; if( ! (impl_getPropertyValue_throw( PROPERTY_IMAGE_URL ) >>= sCurValue) ) SAL_WARN("extensions.propctrlr", "impl_browseForImage_nothrow: unable to get property " << PROPERTY_IMAGE_URL); if (!sCurValue.isEmpty()) { aFileDlg.SetDisplayDirectory( sCurValue ); // TODO: need to set the display directory _and_ the default name } _rClearBeforeDialog.clear(); bool bSuccess = ( ERRCODE_NONE == aFileDlg.Execute() ); if ( bSuccess ) { if ( bHandleNonLink && xController.is() ) { xController->getValue(ExtendedFilePickerElementIds::CHECKBOX_LINK, 0) >>= bIsLink; } if ( !bIsLink ) { Graphic aGraphic; aFileDlg.GetGraphic(aGraphic); Reference< graphic::XGraphicObject > xGrfObj = graphic::GraphicObject::create( m_xContext ); xGrfObj->setGraphic( aGraphic.GetXGraphic() ); _out_rNewValue <<= xGrfObj; } else _out_rNewValue <<= aFileDlg.GetPath(); } return bSuccess; } bool FormComponentPropertyHandler::impl_browseForTargetURL_nothrow( Any& _out_rNewValue, ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { weld::Window* pWin = impl_getDefaultDialogFrame_nothrow(); ::sfx2::FileDialogHelper aFileDlg( ui::dialogs::TemplateDescription::FILEOPEN_READONLY_VERSION, FileDialogFlags::NONE, pWin); OUString sURL; if( ! (impl_getPropertyValue_throw( PROPERTY_TARGET_URL ) >>= sURL) ) SAL_WARN("extensions.propctrlr", "impl_browseForTargetURL_nothrow: unable to get property " << PROPERTY_TARGET_URL); INetURLObject aParser( sURL ); if ( INetProtocol::File == aParser.GetProtocol() ) // set the initial directory only for file-URLs. Everything else // is considered to be potentially expensive aFileDlg.SetDisplayDirectory( sURL ); _rClearBeforeDialog.clear(); bool bSuccess = ( ERRCODE_NONE == aFileDlg.Execute() ); if ( bSuccess ) _out_rNewValue <<= aFileDlg.GetPath(); return bSuccess; } bool FormComponentPropertyHandler::impl_executeFontDialog_nothrow( Any& _out_rNewValue, ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { bool bSuccess = false; // create an item set for use with the dialog std::unique_ptr pSet; rtl::Reference pPool; FontList aFontList(Application::GetDefaultDevice()); ControlCharacterDialog::createItemSet(pSet, pPool, aFontList); ControlCharacterDialog::translatePropertiesToItems(m_xComponent, pSet.get()); { // do this in an own block. The dialog needs to be destroyed before we call // destroyItemSet ControlCharacterDialog aDlg(impl_getDefaultDialogFrame_nothrow(), *pSet); _rClearBeforeDialog.clear(); if (RET_OK == aDlg.run()) { const SfxItemSet* pOut = aDlg.GetOutputItemSet(); if ( pOut ) { std::vector< NamedValue > aFontPropertyValues; ControlCharacterDialog::translateItemsToProperties( *pOut, aFontPropertyValues ); _out_rNewValue <<= comphelper::containerToSequence(aFontPropertyValues); bSuccess = true; } } } ControlCharacterDialog::destroyItemSet(pSet, pPool); return bSuccess; } bool FormComponentPropertyHandler::impl_browseForDatabaseDocument_throw( Any& _out_rNewValue, ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { weld::Window* pWin = impl_getDefaultDialogFrame_nothrow(); ::sfx2::FileDialogHelper aFileDlg( ui::dialogs::TemplateDescription::FILEOPEN_READONLY_VERSION, FileDialogFlags::NONE, u"sdatabase"_ustr, SfxFilterFlags::NONE, SfxFilterFlags::NONE, pWin); OUString sDataSource; if( ! (impl_getPropertyValue_throw( PROPERTY_DATASOURCE ) >>= sDataSource) ) SAL_WARN("extensions.propctrlr", "impl_browseForDatabaseDocument_throw: unable to get property " << PROPERTY_DATASOURCE); INetURLObject aParser( sDataSource ); if ( INetProtocol::File == aParser.GetProtocol() ) // set the initial directory only for file-URLs. Everything else // is considered to be potentially expensive aFileDlg.SetDisplayDirectory( sDataSource ); std::shared_ptr pFilter = SfxFilter::GetFilterByName(u"StarOffice XML (Base)"_ustr); OSL_ENSURE(pFilter,"Filter: StarOffice XML (Base) could not be found!"); if ( pFilter ) { aFileDlg.SetCurrentFilter(pFilter->GetUIName()); //aFileDlg.AddFilter(pFilter->GetFilterName(),pFilter->GetDefaultExtension()); } _rClearBeforeDialog.clear(); bool bSuccess = ( ERRCODE_NONE == aFileDlg.Execute() ); if ( bSuccess ) _out_rNewValue <<= aFileDlg.GetPath(); return bSuccess; } bool FormComponentPropertyHandler::impl_dialogColorChooser_throw( sal_Int32 _nColorPropertyId, Any& _out_rNewValue, ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { ::Color aColor; if( ! (impl_getPropertyValue_throw( impl_getPropertyNameFromId_nothrow( _nColorPropertyId )) >>= aColor) ) SAL_WARN("extensions.propctrlr", "impl_dialogColorChooser_throw: unable to get property " << _nColorPropertyId); SvColorDialog aColorDlg; aColorDlg.SetColor( aColor ); _rClearBeforeDialog.clear(); weld::Window* pParent = impl_getDefaultDialogFrame_nothrow(); if (!aColorDlg.Execute(pParent)) return false; _out_rNewValue <<= aColorDlg.GetColor(); return true; } bool FormComponentPropertyHandler::impl_dialogChooseLabelControl_nothrow( Any& _out_rNewValue, ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { weld::Window* pParent = impl_getDefaultDialogFrame_nothrow(); OSelectLabelDialog dlgSelectLabel(pParent, m_xComponent); _rClearBeforeDialog.clear(); bool bSuccess = (RET_OK == dlgSelectLabel.run()); if ( bSuccess ) _out_rNewValue <<= dlgSelectLabel.GetSelected(); return bSuccess; } Reference< XControlContainer > FormComponentPropertyHandler::impl_getContextControlContainer_nothrow() const { Reference< XControlContainer > xControlContext; Any any = m_xContext->getValueByName( u"ControlContext"_ustr ); any >>= xControlContext; return xControlContext; } bool FormComponentPropertyHandler::impl_dialogChangeTabOrder_nothrow( ::osl::ClearableMutexGuard& _rClearBeforeDialog ) const { OSL_PRECOND( impl_getContextControlContainer_nothrow().is(), "FormComponentPropertyHandler::impl_dialogChangeTabOrder_nothrow: invalid control context!" ); Reference< XTabControllerModel > xTabControllerModel( impl_getRowSet_nothrow(), UNO_QUERY ); TabOrderDialog aDialog(impl_getDefaultDialogFrame_nothrow(), xTabControllerModel, impl_getContextControlContainer_nothrow(), m_xContext); _rClearBeforeDialog.clear(); return RET_OK == aDialog.run(); } namespace { //- ISQLCommandPropertyUI class ISQLCommandPropertyUI : public ISQLCommandAdapter { public: /** returns the empty-string-terminated list of names of properties whose UI is to be disabled while the SQL command property is being edited. */ virtual OUString* getPropertiesToDisable() = 0; }; //- SQLCommandPropertyUI class SQLCommandPropertyUI : public ISQLCommandPropertyUI { protected: explicit SQLCommandPropertyUI( const Reference< XPropertySet >& _rxObject ) : m_xObject(_rxObject) { if ( !m_xObject.is() ) throw NullPointerException(); } protected: Reference< XPropertySet > m_xObject; }; //- FormSQLCommandUI - declaration class FormSQLCommandUI : public SQLCommandPropertyUI { public: explicit FormSQLCommandUI( const Reference< XPropertySet >& _rxForm ); // ISQLCommandAdapter virtual OUString getSQLCommand() const override; virtual bool getEscapeProcessing() const override; virtual void setSQLCommand( const OUString& _rCommand ) const override; virtual void setEscapeProcessing( const bool _bEscapeProcessing ) const override; // ISQLCommandPropertyUI virtual OUString* getPropertiesToDisable() override; }; //- FormSQLCommandUI - implementation FormSQLCommandUI::FormSQLCommandUI( const Reference< XPropertySet >& _rxForm ) :SQLCommandPropertyUI( _rxForm ) { } OUString FormSQLCommandUI::getSQLCommand() const { OUString sCommand; if( ! (m_xObject->getPropertyValue( PROPERTY_COMMAND ) >>= sCommand) ) SAL_WARN("extensions.propctrlr", "getSQLCommand: unable to get property " << PROPERTY_COMMAND); return sCommand; } bool FormSQLCommandUI::getEscapeProcessing() const { bool bEscapeProcessing( false ); if( ! (m_xObject->getPropertyValue( PROPERTY_ESCAPE_PROCESSING ) >>= bEscapeProcessing) ) SAL_WARN("extensions.propctrlr", "getSQLCommand: unable to get property " << PROPERTY_ESCAPE_PROCESSING); return bEscapeProcessing; } void FormSQLCommandUI::setSQLCommand( const OUString& _rCommand ) const { m_xObject->setPropertyValue( PROPERTY_COMMAND, Any( _rCommand ) ); } void FormSQLCommandUI::setEscapeProcessing( const bool _bEscapeProcessing ) const { m_xObject->setPropertyValue( PROPERTY_ESCAPE_PROCESSING, Any( _bEscapeProcessing ) ); } OUString* FormSQLCommandUI::getPropertiesToDisable() { static OUString s_aCommandProps[] = { PROPERTY_DATASOURCE, PROPERTY_COMMAND, PROPERTY_COMMANDTYPE, PROPERTY_ESCAPE_PROCESSING, OUString() }; return s_aCommandProps; } //- ValueListCommandUI - declaration class ValueListCommandUI : public SQLCommandPropertyUI { public: explicit ValueListCommandUI( const Reference< XPropertySet >& _rxListOrCombo ); // ISQLCommandAdapter virtual OUString getSQLCommand() const override; virtual bool getEscapeProcessing() const override; virtual void setSQLCommand( const OUString& _rCommand ) const override; virtual void setEscapeProcessing( const bool _bEscapeProcessing ) const override; // ISQLCommandPropertyUI virtual OUString* getPropertiesToDisable() override; private: mutable bool m_bPropertyValueIsList; }; //- ValueListCommandUI - implementation ValueListCommandUI::ValueListCommandUI( const Reference< XPropertySet >& _rxListOrCombo ) :SQLCommandPropertyUI( _rxListOrCombo ) ,m_bPropertyValueIsList( false ) { } OUString ValueListCommandUI::getSQLCommand() const { OUString sValue; m_bPropertyValueIsList = false; // for combo boxes, the property is a mere string Any aValue( m_xObject->getPropertyValue( PROPERTY_LISTSOURCE ) ); if ( aValue >>= sValue ) return sValue; Sequence< OUString > aValueList; if ( aValue >>= aValueList ) { m_bPropertyValueIsList = true; if ( aValueList.hasElements() ) sValue = aValueList[0]; return sValue; } OSL_FAIL( "ValueListCommandUI::getSQLCommand: unexpected property type!" ); return sValue; } bool ValueListCommandUI::getEscapeProcessing() const { ListSourceType eType = ListSourceType_SQL; if( ! (m_xObject->getPropertyValue( PROPERTY_LISTSOURCETYPE ) >>= eType) ) SAL_WARN("extensions.propctrlr", "getEscapeProcessing: unable to get property " << PROPERTY_LISTSOURCETYPE); OSL_ENSURE( ( eType == ListSourceType_SQL ) || ( eType == ListSourceType_SQLPASSTHROUGH ), "ValueListCommandUI::getEscapeProcessing: unexpected list source type!" ); return ( eType == ListSourceType_SQL ); } void ValueListCommandUI::setSQLCommand( const OUString& _rCommand ) const { Any aValue; if ( m_bPropertyValueIsList ) aValue <<= Sequence< OUString >( &_rCommand, 1 ); else aValue <<= _rCommand; m_xObject->setPropertyValue( PROPERTY_LISTSOURCE, aValue ); } void ValueListCommandUI::setEscapeProcessing( const bool _bEscapeProcessing ) const { m_xObject->setPropertyValue( PROPERTY_LISTSOURCETYPE, Any( _bEscapeProcessing ? ListSourceType_SQL : ListSourceType_SQLPASSTHROUGH ) ); } OUString* ValueListCommandUI::getPropertiesToDisable() { static OUString s_aListSourceProps[] = { PROPERTY_LISTSOURCETYPE, PROPERTY_LISTSOURCE, OUString() }; return s_aListSourceProps; } } bool FormComponentPropertyHandler::impl_doDesignSQLCommand_nothrow( const Reference< XObjectInspectorUI >& _rxInspectorUI, PropertyId _nDesignForProperty ) { try { if ( m_xCommandDesigner.is() ) { if ( m_xCommandDesigner->isActive() ) { m_xCommandDesigner->raise(); return true; } m_xCommandDesigner->dispose(); m_xCommandDesigner.clear(); } if ( !impl_ensureRowsetConnection_nothrow() ) return false; Reference< XPropertySet > xComponentProperties( m_xComponent, UNO_SET_THROW ); ::rtl::Reference< ISQLCommandPropertyUI > xCommandUI; switch ( _nDesignForProperty ) { case PROPERTY_ID_COMMAND: xCommandUI = new FormSQLCommandUI( xComponentProperties ); break; case PROPERTY_ID_LISTSOURCE: xCommandUI = new ValueListCommandUI( xComponentProperties ); break; default: OSL_FAIL( "FormComponentPropertyHandler::OnDesignerClosed: invalid property id!" ); return false; } m_xCommandDesigner.set( new SQLCommandDesigner( m_xContext, xCommandUI, m_xRowSetConnection, LINK( this, FormComponentPropertyHandler, OnDesignerClosed ) ) ); DBG_ASSERT( _rxInspectorUI.is(), "FormComponentPropertyHandler::OnDesignerClosed: no access to the property browser ui!" ); if ( m_xCommandDesigner->isActive() && _rxInspectorUI.is() ) { m_xBrowserUI = _rxInspectorUI; // disable everything which would affect this property const OUString* pToDisable = xCommandUI->getPropertiesToDisable(); while ( !pToDisable->isEmpty() ) { m_xBrowserUI->enablePropertyUIElements( *pToDisable++, PropertyLineElement::All, false ); } // but enable the browse button for the property itself - so it can be used to raise the query designer OUString sPropertyName( impl_getPropertyNameFromId_nothrow( _nDesignForProperty ) ); m_xBrowserUI->enablePropertyUIElements( sPropertyName, PropertyLineElement::PrimaryButton, true ); } } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION("extensions.propctrlr"); } return m_xCommandDesigner.is(); } IMPL_LINK_NOARG( FormComponentPropertyHandler, OnDesignerClosed, SQLCommandDesigner&, void ) { OSL_ENSURE( m_xBrowserUI.is() && m_xCommandDesigner.is(), "FormComponentPropertyHandler::OnDesignerClosed: too many NULLs!" ); if ( !(m_xBrowserUI.is() && m_xCommandDesigner.is()) ) return; try { ::rtl::Reference< ISQLCommandPropertyUI > xCommandUI( dynamic_cast< ISQLCommandPropertyUI* >( m_xCommandDesigner->getPropertyAdapter().get() ) ); if ( !xCommandUI.is() ) throw NullPointerException(); const OUString* pToEnable = xCommandUI->getPropertiesToDisable(); while ( !pToEnable->isEmpty() ) { m_xBrowserUI->enablePropertyUIElements( *pToEnable++, PropertyLineElement::All, true ); } } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION("extensions.propctrlr"); } } bool FormComponentPropertyHandler::impl_hasValidDataSourceSignature_nothrow( const Reference< XPropertySet >& _xFormProperties, bool _bAllowEmptyDataSourceName ) { bool bHas = false; if ( _xFormProperties.is() ) { try { OUString sPropertyValue; // first, we need the name of an existent data source if ( _xFormProperties->getPropertySetInfo()->hasPropertyByName(PROPERTY_DATASOURCE) ) _xFormProperties->getPropertyValue( PROPERTY_DATASOURCE ) >>= sPropertyValue; bHas = ( !sPropertyValue.isEmpty() ) || _bAllowEmptyDataSourceName; // then, the command should not be empty if ( bHas ) { if ( _xFormProperties->getPropertySetInfo()->hasPropertyByName(PROPERTY_COMMAND) ) _xFormProperties->getPropertyValue( PROPERTY_COMMAND ) >>= sPropertyValue; bHas = !sPropertyValue.isEmpty(); } } catch( const Exception& ) { TOOLS_WARN_EXCEPTION( "extensions.propctrlr", "FormComponentPropertyHandler::impl_hasValidDataSourceSignature_nothrow" ); } } return bHas; } OUString FormComponentPropertyHandler::impl_getDocumentURL_nothrow() const { OUString sURL; try { Reference< XModel > xDocument( impl_getContextDocument_nothrow() ); if ( xDocument.is() ) sURL = xDocument->getURL(); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION("extensions.propctrlr"); } return sURL; } ::cppu::IPropertyArrayHelper* FormComponentPropertyHandler::createArrayHelper( ) const { uno::Sequence< beans::Property > aProps; describeProperties(aProps); return new ::cppu::OPropertyArrayHelper(aProps); } ::cppu::IPropertyArrayHelper & FormComponentPropertyHandler::getInfoHelper() { return *getArrayHelper(); } uno::Reference< beans::XPropertySetInfo > SAL_CALL FormComponentPropertyHandler::getPropertySetInfo( ) { return ::cppu::OPropertySetHelper::createPropertySetInfo(getInfoHelper()); } } // namespace pcr extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface* extensions_propctrlr_FormComponentPropertyHandler_get_implementation( css::uno::XComponentContext* context , css::uno::Sequence const&) { return cppu::acquire(new pcr::FormComponentPropertyHandler(context)); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */