/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include const long STD_WIN_SIZE_X = 120; const long STD_WIN_SIZE_Y = 150; const long LISTBOX_BORDER = 2; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::sdb; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::datatransfer; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::form; using namespace ::com::sun::star::container; using namespace ::com::sun::star; using namespace ::svxform; using namespace ::svx; using namespace ::dbtools; namespace { struct ColumnInfo { OUString sColumnName; explicit ColumnInfo(const OUString& i_sColumnName) : sColumnName(i_sColumnName) { } }; } static void lcl_addToList( SvTreeListBox& _rListBox, const uno::Reference< container::XNameAccess>& i_xColumns ) { uno::Sequence< OUString > aEntries = i_xColumns->getElementNames(); const OUString* pEntries = aEntries.getConstArray(); sal_Int32 nEntries = aEntries.getLength(); for ( sal_Int32 i = 0; i < nEntries; ++i, ++pEntries ) { uno::Reference< beans::XPropertySet> xColumn(i_xColumns->getByName(*pEntries),UNO_QUERY_THROW); OUString sLabel; if ( xColumn->getPropertySetInfo()->hasPropertyByName(FM_PROP_LABEL) ) xColumn->getPropertyValue(FM_PROP_LABEL) >>= sLabel; if ( !sLabel.isEmpty() ) _rListBox.InsertEntry( sLabel, nullptr, false, TREELIST_APPEND, new ColumnInfo(*pEntries) ); else _rListBox.InsertEntry( *pEntries, nullptr, false, TREELIST_APPEND, new ColumnInfo(*pEntries) ); } } FmFieldWinListBox::FmFieldWinListBox( FmFieldWin* pParent ) :SvTreeListBox( pParent, WB_HASBUTTONS|WB_BORDER ) ,pTabWin( pParent ) { SetHelpId( HID_FIELD_SEL ); SetHighlightRange( ); } FmFieldWinListBox::~FmFieldWinListBox() { disposeOnce(); } void FmFieldWinListBox::dispose() { pTabWin.clear(); SvTreeListBox::dispose(); } sal_Int8 FmFieldWinListBox::AcceptDrop( const AcceptDropEvent& /*rEvt*/ ) { return DND_ACTION_NONE; } sal_Int8 FmFieldWinListBox::ExecuteDrop( const ExecuteDropEvent& /*rEvt*/ ) { return DND_ACTION_NONE; } bool FmFieldWinListBox::DoubleClickHdl() { if ( pTabWin->createSelectionControls() ) return true; return SvTreeListBox::DoubleClickHdl(); } void FmFieldWinListBox::StartDrag( sal_Int8 /*_nAction*/, const Point& /*_rPosPixel*/ ) { SvTreeListEntry* pSelected = FirstSelected(); if (!pSelected) // no drag without a field return; svx::ODataAccessDescriptor aDescriptor; aDescriptor[ DataAccessDescriptorProperty::DataSource ] <<= pTabWin->GetDatabaseName(); aDescriptor[ DataAccessDescriptorProperty::Connection ] <<= pTabWin->GetConnection().getTyped(); aDescriptor[ DataAccessDescriptorProperty::Command ] <<= pTabWin->GetObjectName(); aDescriptor[ DataAccessDescriptorProperty::CommandType ]<<= pTabWin->GetObjectType(); ColumnInfo* pInfo = static_cast(pSelected->GetUserData()); aDescriptor[ DataAccessDescriptorProperty::ColumnName ] <<= pInfo->sColumnName; rtl::Reference pTransferColumn = new OColumnTransferable( aDescriptor, ColumnTransferFormatFlags::FIELD_DESCRIPTOR | ColumnTransferFormatFlags::CONTROL_EXCHANGE | ColumnTransferFormatFlags::COLUMN_DESCRIPTOR ); EndSelection(); pTransferColumn->StartDrag( this, DND_ACTION_COPY ); } FmFieldWin::FmFieldWin(SfxBindings* _pBindings, SfxChildWindow* _pMgr, vcl::Window* _pParent) :SfxFloatingWindow(_pBindings, _pMgr, _pParent, WinBits(WB_STDMODELESS|WB_SIZEABLE)) ,SfxControllerItem(SID_FM_FIELDS_CONTROL, *_pBindings) ,::comphelper::OPropertyChangeListener(m_aMutex) ,m_nObjectType(0) { SetHelpId( HID_FIELD_SEL_WIN ); SetBackground( Wallpaper( Application::GetSettings().GetStyleSettings().GetFaceColor()) ); pListBox = VclPtr::Create( this ); pListBox->Show(); UpdateContent(nullptr); SetSizePixel(Size(STD_WIN_SIZE_X,STD_WIN_SIZE_Y)); } FmFieldWin::~FmFieldWin() { disposeOnce(); } void FmFieldWin::dispose() { if (m_pChangeListener.is()) { m_pChangeListener->dispose(); m_pChangeListener.clear(); } pListBox.disposeAndClear(); ::SfxControllerItem::dispose(); SfxFloatingWindow::dispose(); } void FmFieldWin::GetFocus() { if ( pListBox ) pListBox->GrabFocus(); else SfxFloatingWindow::GetFocus(); } bool FmFieldWin::createSelectionControls( ) { SvTreeListEntry* pSelected = pListBox->FirstSelected(); if ( pSelected ) { // build a descriptor for the currently selected field ODataAccessDescriptor aDescr; aDescr.setDataSource(GetDatabaseName()); aDescr[ DataAccessDescriptorProperty::Connection ] <<= GetConnection().getTyped(); aDescr[ DataAccessDescriptorProperty::Command ] <<= GetObjectName(); aDescr[ DataAccessDescriptorProperty::CommandType ] <<= GetObjectType(); ColumnInfo* pInfo = static_cast(pSelected->GetUserData()); aDescr[ DataAccessDescriptorProperty::ColumnName ] <<= pInfo->sColumnName;//OUString( pListBox->GetEntryText( pSelected) ); // transfer this to the SFX world SfxUnoAnyItem aDescriptorItem( SID_FM_DATACCESS_DESCRIPTOR, makeAny( aDescr.createPropertyValueSequence() ) ); const SfxPoolItem* pArgs[] = { &aDescriptorItem, nullptr }; // execute the create slot GetBindings().Execute( SID_FM_CREATE_FIELDCONTROL, pArgs ); } return nullptr != pSelected; } bool FmFieldWin::PreNotify( NotifyEvent& _rNEvt ) { if ( MouseNotifyEvent::KEYINPUT == _rNEvt.GetType() ) { const vcl::KeyCode& rKeyCode = _rNEvt.GetKeyEvent()->GetKeyCode(); if ( ( 0 == rKeyCode.GetModifier() ) && ( KEY_RETURN == rKeyCode.GetCode() ) ) { if ( createSelectionControls() ) return true; } } return SfxFloatingWindow::PreNotify( _rNEvt ); } void FmFieldWin::_propertyChanged(const css::beans::PropertyChangeEvent& evt) { css::uno::Reference< css::form::XForm > xForm(evt.Source, css::uno::UNO_QUERY); UpdateContent(xForm); } void FmFieldWin::StateChanged(sal_uInt16 nSID, SfxItemState eState, const SfxPoolItem* pState) { if (!pState || SID_FM_FIELDS_CONTROL != nSID) return; if (eState >= SfxItemState::DEFAULT) { FmFormShell* pShell = dynamic_cast( static_cast(pState)->GetShell() ); UpdateContent(pShell); } else UpdateContent(nullptr); } void FmFieldWin::UpdateContent(FmFormShell const * pShell) { pListBox->Clear(); OUString aTitle(SvxResId(RID_STR_FIELDSELECTION)); SetText( aTitle ); if (!pShell || !pShell->GetImpl()) return; Reference const xForm = pShell->GetImpl()->getCurrentForm_Lock(); if ( xForm.is() ) UpdateContent( xForm ); } void FmFieldWin::UpdateContent(const css::uno::Reference< css::form::XForm > & xForm) { try { // delete ListBox pListBox->Clear(); OUString aTitle(SvxResId(RID_STR_FIELDSELECTION)); SetText(aTitle); if (!xForm.is()) return; Reference< XPropertySet > xSet(xForm, UNO_QUERY); m_aObjectName = ::comphelper::getString(xSet->getPropertyValue(FM_PROP_COMMAND)); m_aDatabaseName = ::comphelper::getString(xSet->getPropertyValue(FM_PROP_DATASOURCE)); m_nObjectType = ::comphelper::getINT32(xSet->getPropertyValue(FM_PROP_COMMANDTYPE)); // get the connection of the form m_aConnection.reset( connectRowset( Reference< XRowSet >( xForm, UNO_QUERY ), ::comphelper::getProcessComponentContext() ), SharedConnection::NoTakeOwnership ); // TODO: When incompatible changes (such as extending the "virtualdbtools" interface by ensureRowSetConnection) // are allowed, again, we should change this: dbtools should consistently use SharedConnection all over // the place, and connectRowset should be replaced with ensureRowSetConnection // get the fields of the object if ( m_aConnection.is() && !m_aObjectName.isEmpty() ) { Reference< XComponent > xKeepFieldsAlive; Reference< XNameAccess > xColumns = getFieldsByCommandDescriptor( m_aConnection, m_nObjectType, m_aObjectName,xKeepFieldsAlive ); if ( xColumns.is() ) lcl_addToList(*pListBox,xColumns); } // set prefix OUString aPrefix; switch (m_nObjectType) { case CommandType::TABLE: aPrefix = SvxResId(RID_RSC_TABWIN_PREFIX[0]); break; case CommandType::QUERY: aPrefix = SvxResId(RID_RSC_TABWIN_PREFIX[1]); break; default: aPrefix = SvxResId(RID_RSC_TABWIN_PREFIX[2]); break; } // listen for changes at ControlSource in PropertySet if (m_pChangeListener.is()) { m_pChangeListener->dispose(); m_pChangeListener.clear(); } m_pChangeListener = new ::comphelper::OPropertyChangeMultiplexer(this, xSet); m_pChangeListener->addProperty(FM_PROP_DATASOURCE); m_pChangeListener->addProperty(FM_PROP_COMMAND); m_pChangeListener->addProperty(FM_PROP_COMMANDTYPE); // set title aTitle += " " + aPrefix + " " + m_aObjectName; SetText( aTitle ); } catch( const Exception& ) { OSL_FAIL( "FmTabWin::UpdateContent: caught an exception!" ); } } void FmFieldWin::Resize() { SfxFloatingWindow::Resize(); Size aOutputSize( GetOutputSizePixel() ); // adapt size of css::form::ListBox Point aLBPos( LISTBOX_BORDER, LISTBOX_BORDER ); Size aLBSize( aOutputSize ); aLBSize.Width() -= (2*LISTBOX_BORDER); aLBSize.Height() -= (2*LISTBOX_BORDER); pListBox->SetPosSizePixel( aLBPos, aLBSize ); } void FmFieldWin::FillInfo( SfxChildWinInfo& rInfo ) const { rInfo.bVisible = false; } SFX_IMPL_FLOATINGWINDOW(FmFieldWinMgr, SID_FM_ADD_FIELD) FmFieldWinMgr::FmFieldWinMgr(vcl::Window* _pParent, sal_uInt16 _nId, SfxBindings* _pBindings, SfxChildWinInfo const * _pInfo) :SfxChildWindow(_pParent, _nId) { SetWindow( VclPtr::Create(_pBindings, this, _pParent) ); SetHideNotDelete(true); static_cast(GetWindow())->Initialize( _pInfo ); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */