3299 lines
135 KiB
C++
3299 lines
135 KiB
C++
/* -*- 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 <propctrlr.h>
|
|
#include <helpids.h>
|
|
#include "fontdialog.hxx"
|
|
#include "formcomponenthandler.hxx"
|
|
#include "formlinkdialog.hxx"
|
|
#include "formmetadata.hxx"
|
|
#include <strings.hrc>
|
|
#include <showhide.hrc>
|
|
#include <yesno.hrc>
|
|
#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 <com/sun/star/lang/NullPointerException.hpp>
|
|
#include <com/sun/star/lang/XServiceInfo.hpp>
|
|
#include <com/sun/star/form/FormComponentType.hpp>
|
|
#include <com/sun/star/beans/PropertyAttribute.hpp>
|
|
#include <com/sun/star/util/XNumberFormatsSupplier.hpp>
|
|
#include <com/sun/star/container/XNameAccess.hpp>
|
|
#include <com/sun/star/form/XForm.hpp>
|
|
#include <com/sun/star/container/XChild.hpp>
|
|
#include <com/sun/star/sdb/OrderDialog.hpp>
|
|
#include <com/sun/star/sdb/FilterDialog.hpp>
|
|
#include <com/sun/star/sdbc/XConnection.hpp>
|
|
#include <com/sun/star/sdb/CommandType.hpp>
|
|
#include <com/sun/star/sdb/DatabaseContext.hpp>
|
|
#include <com/sun/star/form/XGridColumnFactory.hpp>
|
|
#include <com/sun/star/sdb/SQLContext.hpp>
|
|
#include <com/sun/star/sdbcx/XTablesSupplier.hpp>
|
|
#include <com/sun/star/sdb/XQueriesSupplier.hpp>
|
|
#include <com/sun/star/form/ListSourceType.hpp>
|
|
#include <com/sun/star/ui/dialogs/XExecutableDialog.hpp>
|
|
#include <com/sun/star/ui/dialogs/XFilePicker3.hpp>
|
|
#include <com/sun/star/sdb/XSingleSelectQueryComposer.hpp>
|
|
#include <com/sun/star/ui/dialogs/XFilePickerControlAccess.hpp>
|
|
#include <com/sun/star/ui/dialogs/TemplateDescription.hpp>
|
|
#include <com/sun/star/ui/dialogs/ExtendedFilePickerElementIds.hpp>
|
|
#include <com/sun/star/awt/XTabControllerModel.hpp>
|
|
#include <com/sun/star/form/FormSubmitEncoding.hpp>
|
|
#include <com/sun/star/awt/VisualEffect.hpp>
|
|
#include <com/sun/star/form/FormButtonType.hpp>
|
|
#include <com/sun/star/inspection/PropertyControlType.hpp>
|
|
#include <com/sun/star/util/MeasureUnit.hpp>
|
|
#include <com/sun/star/inspection/XObjectInspectorUI.hpp>
|
|
#include <com/sun/star/inspection/PropertyLineElement.hpp>
|
|
#include <com/sun/star/resource/XStringResourceManager.hpp>
|
|
#include <com/sun/star/resource/MissingResourceException.hpp>
|
|
#include <com/sun/star/report/XReportDefinition.hpp>
|
|
#include <com/sun/star/graphic/GraphicObject.hpp>
|
|
#include <com/sun/star/text/WritingMode2.hpp>
|
|
|
|
#include <comphelper/extract.hxx>
|
|
#include <comphelper/types.hxx>
|
|
#include <connectivity/dbconversion.hxx>
|
|
#include <connectivity/dbexception.hxx>
|
|
#include <cppuhelper/exc_hlp.hxx>
|
|
#include <sfx2/app.hxx>
|
|
#include <sfx2/basedlgs.hxx>
|
|
#include <sfx2/docfilt.hxx>
|
|
#include <sfx2/filedlghelper.hxx>
|
|
#include <svl/ctloptions.hxx>
|
|
#include <svtools/colrdlg.hxx>
|
|
#include <svl/filenotation.hxx>
|
|
#include <svl/intitem.hxx>
|
|
#include <svl/itemset.hxx>
|
|
#include <svl/numformat.hxx>
|
|
#include <unotools/moduleoptions.hxx>
|
|
#include <svl/numuno.hxx>
|
|
#include <svl/urihelper.hxx>
|
|
#include <svx/dialogs.hrc>
|
|
#include <svx/numinf.hxx>
|
|
#include <svx/svxdlg.hxx>
|
|
#include <svx/svxids.hrc>
|
|
#include <vcl/graph.hxx>
|
|
#include <vcl/unohelp.hxx>
|
|
#include <comphelper/diagnose_ex.hxx>
|
|
#include <sal/macros.h>
|
|
#include <sal/log.hxx>
|
|
#include <vcl/svapp.hxx>
|
|
|
|
#include <limits>
|
|
#include <memory>
|
|
#include <string_view>
|
|
|
|
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<decltype(m_xRowSet)>::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<OUString[]> 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<OUString *, bool>(), 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<weld::Builder> xBuilder(PropertyHandlerHelper::makeBuilder(u"modules/spropctrlr/ui/urlcontrol.ui"_ustr, m_xContext));
|
|
auto pURLBox = std::make_unique<SvtURLBox>(xBuilder->weld_combo_box(u"urlcontrol"_ustr));
|
|
rtl::Reference<OFileUrlControl> 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<SvNumberFormatsSupplierObj>(xTunnel))
|
|
{
|
|
bool bIsFormatKey = (PROPERTY_ID_FORMATKEY == nPropId);
|
|
|
|
bReadOnly = bIsFormatKey;
|
|
|
|
if ( bIsFormatKey )
|
|
{
|
|
std::unique_ptr<weld::Builder> xBuilder(PropertyHandlerHelper::makeBuilder(u"modules/spropctrlr/ui/formattedsample.ui"_ustr, m_xContext));
|
|
auto pContainer = xBuilder->weld_container(u"formattedsample"_ustr);
|
|
rtl::Reference<OFormatSampleControl> 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<weld::Builder> xBuilder(PropertyHandlerHelper::makeBuilder(u"modules/spropctrlr/ui/formattedcontrol.ui"_ustr, m_xContext));
|
|
auto pSpinButton = xBuilder->weld_formatted_spin_button(u"formattedcontrol"_ustr);
|
|
rtl::Reference<OFormattedNumericControl> 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<weld::Builder> xBuilder(PropertyHandlerHelper::makeBuilder(u"modules/spropctrlr/ui/formattedcontrol.ui"_ustr, m_xContext));
|
|
auto pSpinButton = xBuilder->weld_formatted_spin_button(u"formattedcontrol"_ustr);
|
|
rtl::Reference<OFormattedNumericControl> 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<weld::Builder> 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<ONumericControl> 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<SvNumberFormatsSupplierObj*>(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> xModel(impl_getContextDocument_nothrow());
|
|
Reference<XReportDefinition> 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<OUString> 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<sdbc::XConnection> 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<OUString> 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<OUString> 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<ListSourceType>(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<beans::XPropertySet> 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<sal_Int32>( 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<SfxItemSet> pSet;
|
|
rtl::Reference<SfxItemPool> 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<const SfxFilter> 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<css::uno::Any> const&)
|
|
{
|
|
return cppu::acquire(new pcr::FormComponentPropertyHandler(context));
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|