/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ #include "dbinsdlg.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "tabsh.hxx" #include "swabstdlg.hxx" #include "table.hrc" #include #include namespace swui { SwAbstractDialogFactory * GetFactory(); } using namespace ::com::sun::star; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::container; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::sdb; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::sdbcx; using namespace ::com::sun::star::uno; const char cDBFldStart = '<'; const char cDBFldEnd = '>'; // Helper structure for adding database rows as fields or text struct _DB_Column { enum ColType { DB_FILLTEXT, DB_COL_FIELD, DB_COL_TEXT, DB_SPLITPARA } eColType; union { String* pText; SwField* pField; sal_uLong nFormat; } DB_ColumnData; const SwInsDBColumn* pColInfo; _DB_Column() { pColInfo = 0; DB_ColumnData.pText = 0; eColType = DB_SPLITPARA; } _DB_Column( const String& rTxt ) { pColInfo = 0; DB_ColumnData.pText = new String( rTxt ); eColType = DB_FILLTEXT; } _DB_Column( const SwInsDBColumn& rInfo, sal_uLong nFormat ) { pColInfo = &rInfo; DB_ColumnData.nFormat = nFormat; eColType = DB_COL_TEXT; } _DB_Column( const SwInsDBColumn& rInfo, SwDBField& rFld ) { pColInfo = &rInfo; DB_ColumnData.pField = &rFld; eColType = DB_COL_FIELD; } ~_DB_Column() { if( DB_COL_FIELD == eColType ) delete DB_ColumnData.pField; else if( DB_FILLTEXT == eColType ) delete DB_ColumnData.pText; } }; struct _DB_ColumnConfigData { SwInsDBColumns aDBColumns; rtl::OUString sSource, sTable, sEdit, sTblList, sTmplNm, sTAutoFmtNm; sal_Bool bIsTable : 1, bIsField : 1, bIsHeadlineOn : 1, bIsEmptyHeadln : 1; _DB_ColumnConfigData() { bIsTable = bIsHeadlineOn = sal_True; bIsField = bIsEmptyHeadln = sal_False; } ~_DB_ColumnConfigData(); private: _DB_ColumnConfigData( const _DB_ColumnConfigData& ); _DB_ColumnConfigData& operator =( const _DB_ColumnConfigData& ); }; int SwInsDBColumn::operator<( const SwInsDBColumn& rCmp ) const { return 0 > GetAppCollator().compareString( sColumn, rCmp.sColumn ); } SwInsertDBColAutoPilot::SwInsertDBColAutoPilot( SwView& rView, Reference xDataSource, Reference xColSupp, const SwDBData& rData ) : SfxModalDialog( rView.GetWindow(), SW_RES( DLG_AP_INSERT_DB_SEL )), ConfigItem(C2U("Office.Writer/InsertData/DataSet"), CONFIG_MODE_DELAYED_UPDATE), aFtInsertData( this, SW_RES( FT_INSERT_DATA )), aRbAsTable( this, SW_RES( RB_AS_TABLE )), aRbAsField( this, SW_RES( RB_AS_FIELD )), aRbAsText( this, SW_RES( RB_AS_TEXT )), aFlHead( this, SW_RES( FL_HEAD )), aFtDbColumn( this, SW_RES( FT_DB_COLUMN )), aLbTblDbColumn( this, SW_RES( LB_TBL_DB_COLUMN )), aLbTxtDbColumn( this, SW_RES( LB_TXT_DB_COLUMN )), aFlFormat( this, SW_RES( FL_FORMAT )), aRbDbFmtFromDb( this, SW_RES( RB_DBFMT_FROM_DB )), aRbDbFmtFromUsr( this, SW_RES( RB_DBFMT_FROM_USR )), aLbDbFmtFromUsr( this, &rView, SW_RES( LB_DBFMT_FROM_USR )), aIbDbcolToEdit( this, SW_RES( IB_DBCOL_TOEDIT )), aEdDbText( this, SW_RES( ED_DB_TEXT )), aFtDbParaColl( this, SW_RES( FT_DB_PARA_COLL )), aLbDbParaColl( this, SW_RES( LB_DB_PARA_COLL )), aIbDbcolAllTo( this, SW_RES( IB_DBCOL_ALL_TO )), aIbDbcolOneTo( this, SW_RES( IB_DBCOL_ONE_TO )), aIbDbcolOneFrom( this, SW_RES( IB_DBCOL_ONE_FROM )), aIbDbcolAllFrom( this, SW_RES( IB_DBCOL_ALL_FROM )), aFtTableCol( this, SW_RES( FT_TABLE_COL )), aLbTableCol( this, SW_RES( LB_TABLE_COL )), aCbTableHeadon( this, SW_RES( CB_TABLE_HEADON )), aRbHeadlColnms( this, SW_RES( RB_HEADL_COLNMS )), aRbHeadlEmpty( this, SW_RES( RB_HEADL_EMPTY )), aPbTblFormat( this, SW_RES( PB_TBL_FORMAT )), aPbTblAutofmt( this, SW_RES( PB_TBL_AUTOFMT )), aBtOk( this, SW_RES( BT_OK )), aBtCancel( this, SW_RES( BT_CANCEL )), aBtHelp( this, SW_RES( BT_HELP )), aFlBottom( this, SW_RES( FL_BOTTOM )), aDBData(rData), aOldNumFmtLnk( aLbDbFmtFromUsr.GetSelectHdl() ), sNoTmpl( SW_RES( STR_NOTEMPL )), pView( &rView ), pTAutoFmt( 0 ), pTblSet( 0 ), pRep( 0 ) { FreeResource(); nGBFmtLen = aFlFormat.GetText().Len(); if(xColSupp.is()) { SwWrtShell& rSh = pView->GetWrtShell(); SvNumberFormatter* pNumFmtr = rSh.GetNumberFormatter(); SvNumberFormatsSupplierObj* pNumFmt = new SvNumberFormatsSupplierObj( pNumFmtr ); Reference< util::XNumberFormatsSupplier > xDocNumFmtsSupplier = pNumFmt; Reference< util::XNumberFormats > xDocNumberFormats = xDocNumFmtsSupplier->getNumberFormats(); Reference< util::XNumberFormatTypes > xDocNumberFormatTypes(xDocNumberFormats, UNO_QUERY); Reference xSourceProps(xDataSource, UNO_QUERY); Reference< util::XNumberFormats > xNumberFormats; if(xSourceProps.is()) { Any aFormats = xSourceProps->getPropertyValue(C2U("NumberFormatsSupplier")); if(aFormats.hasValue()) { Reference< util::XNumberFormatsSupplier> xSuppl; aFormats >>= xSuppl; if(xSuppl.is()) { xNumberFormats = xSuppl->getNumberFormats( ); } } } Reference xCols = xColSupp->getColumns(); Sequence aColNames = xCols->getElementNames(); const rtl::OUString* pColNames = aColNames.getConstArray(); long nCount = aColNames.getLength(); for (long n = 0; n < nCount; n++) { SwInsDBColumn* pNew = new SwInsDBColumn( pColNames[n], (sal_uInt16)n ); Any aCol = xCols->getByName(pColNames[n]); Reference xCol; aCol >>= xCol; Any aType = xCol->getPropertyValue(rtl::OUString("Type")); sal_Int32 eDataType = 0; aType >>= eDataType; switch(eDataType) { case DataType::BIT: case DataType::BOOLEAN: case DataType::TINYINT: case DataType::SMALLINT: case DataType::INTEGER: case DataType::BIGINT: case DataType::FLOAT: case DataType::REAL: case DataType::DOUBLE: case DataType::NUMERIC: case DataType::DECIMAL: case DataType::DATE: case DataType::TIME: case DataType::TIMESTAMP: { pNew->bHasFmt = sal_True; Any aFormat = xCol->getPropertyValue(C2U("FormatKey")); if(aFormat.hasValue()) { sal_Int32 nFmt = 0; aFormat >>= nFmt; if(xNumberFormats.is()) { try { Reference xNumProps = xNumberFormats->getByKey( nFmt ); Any aFormatVal = xNumProps->getPropertyValue(C2U("FormatString")); Any aLocale = xNumProps->getPropertyValue(C2U("Locale")); rtl::OUString sFormat; aFormatVal >>= sFormat; lang::Locale aLoc; aLocale >>= aLoc; long nKey = xDocNumberFormats->queryKey( sFormat, aLoc, sal_True); if(nKey < 0) { nKey = xDocNumberFormats->addNew( sFormat, aLoc ); } pNew->nDBNumFmt = nKey; } catch (const Exception&) { OSL_FAIL("illegal number format key"); } } } else { pNew->nDBNumFmt = SwNewDBMgr::GetDbtoolsClient().getDefaultNumberFormat(xCol, xDocNumberFormatTypes, LanguageTag( rSh.GetCurLang() ).getLocale()); } } break; } if( !aDBColumns.insert( pNew ).second ) { OSL_ENSURE( !this, "Spaltenname mehrfach vergeben?" ); delete pNew; } } } // fill paragraph templates-ListBox { SfxStyleSheetBasePool* pPool = pView->GetDocShell()->GetStyleSheetPool(); pPool->SetSearchMask( SFX_STYLE_FAMILY_PARA, SFXSTYLEBIT_ALL ); aLbDbParaColl.InsertEntry( sNoTmpl ); const SfxStyleSheetBase* pBase = pPool->First(); while( pBase ) { aLbDbParaColl.InsertEntry( pBase->GetName() ); pBase = pPool->Next(); } aLbDbParaColl.SelectEntryPos( 0 ); } // when the cursor is inside of a table, table must NEVER be selectable if( pView->GetWrtShell().GetTableFmt() ) { aRbAsTable.Enable( sal_False ); aRbAsField.Check( sal_True ); aRbDbFmtFromDb.Check( sal_True ); } else { aRbAsTable.Check( sal_True ); aRbDbFmtFromDb.Check( sal_True ); aIbDbcolOneFrom.Enable( sal_False ); aIbDbcolAllFrom.Enable( sal_False ); } aRbAsTable.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, PageHdl )); aRbAsField.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, PageHdl )); aRbAsText.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, PageHdl )); aRbDbFmtFromDb.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, DBFormatHdl )); aRbDbFmtFromUsr.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, DBFormatHdl )); aPbTblFormat.SetClickHdl(LINK(this, SwInsertDBColAutoPilot, TblFmtHdl )); aPbTblAutofmt.SetClickHdl(LINK(this, SwInsertDBColAutoPilot, AutoFmtHdl )); aIbDbcolAllTo.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, TblToFromHdl )); aIbDbcolOneTo.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, TblToFromHdl )); aIbDbcolOneFrom.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, TblToFromHdl )); aIbDbcolAllFrom.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, TblToFromHdl )); aIbDbcolToEdit.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, TblToFromHdl )); aCbTableHeadon.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, HeaderHdl )); aRbHeadlColnms.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, HeaderHdl )); aRbHeadlEmpty.SetClickHdl( LINK(this, SwInsertDBColAutoPilot, HeaderHdl )); aLbTxtDbColumn.SetSelectHdl( LINK( this, SwInsertDBColAutoPilot, SelectHdl )); aLbTblDbColumn.SetSelectHdl( LINK( this, SwInsertDBColAutoPilot, SelectHdl )); aLbDbFmtFromUsr.SetSelectHdl( LINK( this, SwInsertDBColAutoPilot, SelectHdl )); aLbTableCol.SetSelectHdl( LINK( this, SwInsertDBColAutoPilot, SelectHdl )); aLbTxtDbColumn.SetDoubleClickHdl( LINK( this, SwInsertDBColAutoPilot, DblClickHdl )); aLbTblDbColumn.SetDoubleClickHdl( LINK( this, SwInsertDBColAutoPilot, DblClickHdl )); aLbTableCol.SetDoubleClickHdl( LINK( this, SwInsertDBColAutoPilot, DblClickHdl )); for( sal_uInt16 n = 0; n < aDBColumns.size(); ++n ) { const String& rS = aDBColumns[ n ]->sColumn; aLbTblDbColumn.InsertEntry( rS, n ); aLbTxtDbColumn.InsertEntry( rS, n ); } aLbTxtDbColumn.SelectEntryPos( 0 ); aLbTblDbColumn.SelectEntryPos( 0 ); // read configuration Load(); // initialise Controls: PageHdl( aRbAsTable.IsChecked() ? &aRbAsTable : &aRbAsField ); } SwInsertDBColAutoPilot::~SwInsertDBColAutoPilot() { delete pTblSet; delete pRep; delete pTAutoFmt; } IMPL_LINK( SwInsertDBColAutoPilot, PageHdl, Button*, pButton ) { sal_Bool bShowTbl = pButton == &aRbAsTable; String sTxt( pButton->GetText() ); aFlHead.SetText( MnemonicGenerator::EraseAllMnemonicChars( sTxt ) ); aLbTxtDbColumn.Show( !bShowTbl ); aIbDbcolToEdit.Show( !bShowTbl ); aEdDbText.Show( !bShowTbl ); aFtDbParaColl.Show( !bShowTbl ); aLbDbParaColl.Show( !bShowTbl ); aLbTblDbColumn.Show( bShowTbl ); aIbDbcolAllTo.Show( bShowTbl ); aIbDbcolOneTo.Show( bShowTbl ); aIbDbcolOneFrom.Show( bShowTbl ); aIbDbcolAllFrom.Show( bShowTbl ); aFtTableCol.Show( bShowTbl ); aLbTableCol.Show( bShowTbl ); aCbTableHeadon.Show( bShowTbl ); aRbHeadlColnms.Show( bShowTbl ); aRbHeadlEmpty.Show( bShowTbl ); aPbTblFormat.Show( bShowTbl ); aPbTblAutofmt.Show( bShowTbl ); if( bShowTbl ) aPbTblFormat.Enable( 0 != aLbTableCol.GetEntryCount() ); SelectHdl( bShowTbl ? &aLbTblDbColumn : &aLbTxtDbColumn ); return 0; } IMPL_LINK( SwInsertDBColAutoPilot, DBFormatHdl, Button*, pButton ) { ListBox& rBox = aRbAsTable.IsChecked() ? ( 0 == aLbTableCol.GetEntryData( 0 ) ? aLbTblDbColumn : aLbTableCol ) : aLbTxtDbColumn; SwInsDBColumn aSrch( rBox.GetSelectEntry(), 0 ); SwInsDBColumns::const_iterator it = aDBColumns.find( &aSrch ); sal_Bool bFromDB = &aRbDbFmtFromDb == pButton; (*it)->bIsDBFmt = bFromDB; aLbDbFmtFromUsr.Enable( !bFromDB ); return 0; } IMPL_LINK( SwInsertDBColAutoPilot, TblToFromHdl, Button*, pButton ) { sal_Bool bChgEnable = sal_True, bEnableTo = sal_True, bEnableFrom = sal_True; aLbTblDbColumn.SetUpdateMode( sal_False ); aLbTableCol.SetUpdateMode( sal_False ); if( pButton == &aIbDbcolAllTo ) { bEnableTo = sal_False; sal_uInt16 n, nInsPos = aLbTableCol.GetSelectEntryPos(), nCnt = aLbTblDbColumn.GetEntryCount(); if( LISTBOX_APPEND == nInsPos ) for( n = 0; n < nCnt; ++n ) aLbTableCol.InsertEntry( aLbTblDbColumn.GetEntry( n ), LISTBOX_APPEND ); else for( n = 0; n < nCnt; ++n, ++nInsPos ) aLbTableCol.InsertEntry( aLbTblDbColumn.GetEntry( n ), nInsPos ); aLbTblDbColumn.Clear(); aLbTableCol.SelectEntryPos( nInsPos ); aLbTblDbColumn.SelectEntryPos( LISTBOX_APPEND ); } else if( pButton == &aIbDbcolOneTo && LISTBOX_ENTRY_NOTFOUND != aLbTblDbColumn.GetSelectEntryPos() ) { sal_uInt16 nInsPos = aLbTableCol.GetSelectEntryPos(), nDelPos = aLbTblDbColumn.GetSelectEntryPos(), nTopPos = aLbTblDbColumn.GetTopEntry(); aLbTableCol.InsertEntry( aLbTblDbColumn.GetEntry( nDelPos ), nInsPos ); aLbTblDbColumn.RemoveEntry( nDelPos ); aLbTableCol.SelectEntryPos( nInsPos ); if( nDelPos >= aLbTblDbColumn.GetEntryCount() ) nDelPos = aLbTblDbColumn.GetEntryCount() - 1; aLbTblDbColumn.SelectEntryPos( nDelPos ); aLbTblDbColumn.SetTopEntry( nTopPos ); bEnableTo = 0 != aLbTblDbColumn.GetEntryCount(); } else if( pButton == &aIbDbcolOneFrom ) { if( LISTBOX_ENTRY_NOTFOUND != aLbTableCol.GetSelectEntryPos() ) { sal_uInt16 nInsPos, nDelPos = aLbTableCol.GetSelectEntryPos(), nTopPos = aLbTableCol.GetTopEntry(); // look for the right InsertPos!! SwInsDBColumn aSrch( aLbTableCol.GetEntry( nDelPos ), 0 ); SwInsDBColumns::const_iterator it = aDBColumns.find( &aSrch ); if( it == aDBColumns.begin() || (it+1) == aDBColumns.end() ) nInsPos = it - aDBColumns.begin(); else { nInsPos = LISTBOX_ENTRY_NOTFOUND; while( ++it != aDBColumns.end() && LISTBOX_ENTRY_NOTFOUND == (nInsPos = aLbTblDbColumn. GetEntryPos( String( (*it)->sColumn ))) ) ; } aLbTblDbColumn.InsertEntry( aSrch.sColumn, nInsPos ); aLbTableCol.RemoveEntry( nDelPos ); if( nInsPos >= aLbTblDbColumn.GetEntryCount() ) nInsPos = aLbTblDbColumn.GetEntryCount() - 1; aLbTblDbColumn.SelectEntryPos( nInsPos ); if( nDelPos >= aLbTableCol.GetEntryCount() ) nDelPos = aLbTableCol.GetEntryCount() - 1; aLbTableCol.SelectEntryPos( nDelPos ); aLbTableCol.SetTopEntry( nTopPos ); } else bEnableTo = 0 != aLbTblDbColumn.GetEntryCount(); bEnableFrom = 0 != aLbTableCol.GetEntryCount(); } else if( pButton == &aIbDbcolAllFrom ) { bEnableFrom = sal_False; aLbTblDbColumn.Clear(); aLbTableCol.Clear(); for( sal_uInt16 n = 0; n < aDBColumns.size(); ++n ) aLbTblDbColumn.InsertEntry( aDBColumns[ n ]->sColumn, n ); aLbTblDbColumn.SelectEntryPos( 0 ); } else if( pButton == &aIbDbcolToEdit ) { bChgEnable = sal_False; // move data to Edit: String aFld( aLbTxtDbColumn.GetSelectEntry() ); if( aFld.Len() ) { String aStr( aEdDbText.GetText() ); sal_uInt16 nPos = (sal_uInt16)aEdDbText.GetSelection().Min(); sal_uInt16 nSel = sal_uInt16(aEdDbText.GetSelection().Max()) - nPos; if( nSel ) // first delete the existing selection aStr.Erase( nPos, nSel ); aFld.Insert( cDBFldStart, 0 ); aFld += cDBFldEnd; if( aStr.Len() ) { if( nPos ) // one blank in front { sal_Unicode c = aStr.GetChar( nPos-1 ); if( '\n' != c && '\r' != c ) aFld.Insert( ' ', 0 ); } if( nPos < aStr.Len() ) // one blank behind { sal_Unicode c = aStr.GetChar( nPos ); if( '\n' != c && '\r' != c ) aFld += ' '; } } aStr.Insert( aFld, nPos ); aEdDbText.SetText( aStr ); nPos = nPos + aFld.Len(); aEdDbText.SetSelection( Selection( nPos )); } } if( bChgEnable ) { aIbDbcolOneTo.Enable( bEnableTo ); aIbDbcolAllTo.Enable( bEnableTo ); aIbDbcolOneFrom.Enable( bEnableFrom ); aIbDbcolAllFrom.Enable( bEnableFrom ); aRbDbFmtFromDb.Enable( sal_False ); aRbDbFmtFromUsr.Enable( sal_False ); aLbDbFmtFromUsr.Enable( sal_False ); aPbTblFormat.Enable( bEnableFrom ); } aLbTblDbColumn.SetUpdateMode( sal_True ); aLbTableCol.SetUpdateMode( sal_True ); return 0; } IMPL_LINK( SwInsertDBColAutoPilot, DblClickHdl, ListBox*, pBox ) { Button* pButton = 0; if( pBox == &aLbTxtDbColumn ) pButton = &aIbDbcolToEdit; else if( pBox == &aLbTblDbColumn && aIbDbcolOneTo.IsEnabled() ) pButton = &aIbDbcolOneTo; else if( pBox == &aLbTableCol && aIbDbcolOneFrom.IsEnabled() ) pButton = &aIbDbcolOneFrom; if( pButton ) TblToFromHdl( pButton ); return 0; } IMPL_LINK( SwInsertDBColAutoPilot, TblFmtHdl, PushButton*, pButton ) { SwWrtShell& rSh = pView->GetWrtShell(); sal_Bool bNewSet = sal_False; if( !pTblSet ) { bNewSet = sal_True; pTblSet = new SfxItemSet( rSh.GetAttrPool(), SwuiGetUITableAttrRange() ); // At first acquire the simple attributes pTblSet->Put( SfxStringItem( FN_PARAM_TABLE_NAME, rSh.GetUniqueTblName() )); pTblSet->Put( SfxUInt16Item( FN_PARAM_TABLE_HEADLINE, 1 ) ); pTblSet->Put( SfxUInt16Item( SID_BACKGRND_DESTINATION, rSh.GetViewOptions()->GetTblDest() )); SvxBrushItem aBrush( RES_BACKGROUND ); pTblSet->Put( aBrush ); pTblSet->Put( aBrush, SID_ATTR_BRUSH_ROW ); pTblSet->Put( aBrush, SID_ATTR_BRUSH_TABLE ); SvxBoxInfoItem aBoxInfo( SID_ATTR_BORDER_INNER ); // table variant, when multiple table cells are selected aBoxInfo.SetTable( sal_True ); // always show gap field aBoxInfo.SetDist( sal_True); // set minimum size in tables and paragraphs aBoxInfo.SetMinDist( sal_False ); // always set default-gap aBoxInfo.SetDefDist( MIN_BORDER_DIST ); // Single lines can have DontCare-status only in tables aBoxInfo.SetValid( VALID_DISABLE, sal_True ); pTblSet->Put( aBoxInfo ); SwGetCurColNumPara aPara; const sal_uInt16 nNum = rSh.GetCurColNum( &aPara ); long nWidth; if( nNum ) { nWidth = aPara.pPrtRect->Width(); const SwFmtCol& rCol = aPara.pFrmFmt->GetCol(); const SwColumns& rCols = rCol.GetColumns(); // initialise nStart und nEnd for nNum == 0 long nWidth1 = 0, nStart1 = 0, nEnd1 = nWidth; for( sal_uInt16 i = 0; i < nNum; ++i ) { const SwColumn* pCol = &rCols[i]; nStart1 = pCol->GetLeft() + nWidth1; nWidth1 += (long)rCol.CalcColWidth( i, (sal_uInt16)nWidth ); nEnd1 = nWidth1 - pCol->GetRight(); } if(nStart1 || nEnd1 != nWidth) nWidth = nEnd1 - nStart1; } else nWidth = rSh.GetAnyCurRect( FRMTYPE_FLY_ANY & rSh.GetFrmType( 0, sal_True ) ? RECT_FLY_PRT_EMBEDDED : RECT_PAGE_PRT ).Width(); SwTabCols aTabCols; aTabCols.SetRight( nWidth ); aTabCols.SetRightMax( nWidth ); pRep = new SwTableRep( aTabCols, sal_False ); pRep->SetAlign( text::HoriOrientation::NONE ); pRep->SetSpace( nWidth ); pRep->SetWidth( nWidth ); pRep->SetWidthPercent( 100 ); pTblSet->Put( SwPtrItem( FN_TABLE_REP, pRep )); pTblSet->Put( SfxUInt16Item( SID_HTML_MODE, ::GetHtmlMode( pView->GetDocShell() ))); } if( aLbTableCol.GetEntryCount() != pRep->GetAllColCount() ) { // Number of columns has changed: then the TabCols have to be adjusted long nWidth = pRep->GetWidth(); sal_uInt16 nCols = aLbTableCol.GetEntryCount() - 1; SwTabCols aTabCols( nCols ); aTabCols.SetRight( nWidth ); aTabCols.SetRightMax( nWidth ); if( nCols ) for( sal_uInt16 n = 0, nStep = (sal_uInt16)(nWidth / (nCols+1)), nW = nStep; n < nCols; ++n, nW = nW + nStep ) { aTabCols.Insert( nW, sal_False, n ); } delete pRep; pRep = new SwTableRep( aTabCols, sal_False ); pRep->SetAlign( text::HoriOrientation::NONE ); pRep->SetSpace( nWidth ); pRep->SetWidth( nWidth ); pRep->SetWidthPercent( 100 ); pTblSet->Put( SwPtrItem( FN_TABLE_REP, pRep )); } SwAbstractDialogFactory* pFact = swui::GetFactory(); OSL_ENSURE(pFact, "SwAbstractDialogFactory fail!"); SfxAbstractTabDialog* pDlg = pFact->CreateSwTableTabDlg( pButton, rSh.GetAttrPool(),pTblSet, &rSh, DLG_FORMAT_TABLE ); OSL_ENSURE(pDlg, "Dialogdiet fail!"); if( RET_OK == pDlg->Execute() ) pTblSet->Put( *pDlg->GetOutputItemSet() ); else if( bNewSet ) { delete pTblSet, pTblSet = 0; delete pRep, pRep = 0; } delete pDlg; return 0; } IMPL_LINK( SwInsertDBColAutoPilot, AutoFmtHdl, PushButton*, pButton ) { SwAbstractDialogFactory* pFact = swui::GetFactory(); OSL_ENSURE(pFact, "SwAbstractDialogFactory fail!"); AbstractSwAutoFormatDlg* pDlg = pFact->CreateSwAutoFormatDlg(pButton, pView->GetWrtShellPtr(), sal_False, pTAutoFmt); OSL_ENSURE(pDlg, "Dialogdiet fail!"); if( RET_OK == pDlg->Execute()) pDlg->FillAutoFmtOfIndex( pTAutoFmt ); delete pDlg; return 0; } IMPL_LINK( SwInsertDBColAutoPilot, SelectHdl, ListBox*, pBox ) { ListBox* pGetBox = pBox == &aLbDbFmtFromUsr ? ( aRbAsTable.IsChecked() ? ( 0 == aLbTableCol.GetEntryData( 0 ) ? &aLbTblDbColumn : &aLbTableCol ) : &aLbTxtDbColumn ) : pBox; SwInsDBColumn aSrch( pGetBox->GetSelectEntry(), 0 ); SwInsDBColumns::const_iterator it = aDBColumns.find( &aSrch ); if( pBox == &aLbDbFmtFromUsr ) { if( !aSrch.sColumn.isEmpty() ) { aOldNumFmtLnk.Call( pBox ); (*it)->nUsrNumFmt = aLbDbFmtFromUsr.GetFormat(); } } else { // set the selected FieldName at the FormatGroupBox, so that // it's clear what field is configured by the format! String sTxt( aFlFormat.GetText().Copy( 0, nGBFmtLen )); if( aSrch.sColumn.isEmpty() ) { aRbDbFmtFromDb.Enable( sal_False ); aRbDbFmtFromUsr.Enable( sal_False ); aLbDbFmtFromUsr.Enable( sal_False ); } else { sal_Bool bEnableFmt = (*it)->bHasFmt; aRbDbFmtFromDb.Enable( bEnableFmt ); aRbDbFmtFromUsr.Enable( bEnableFmt ); if( bEnableFmt ) { (( sTxt += rtl::OUString(" (" )) += String(aSrch.sColumn) ) += (sal_Unicode)')'; } sal_Bool bIsDBFmt = (*it)->bIsDBFmt; aRbDbFmtFromDb.Check( bIsDBFmt ); aRbDbFmtFromUsr.Check( !bIsDBFmt ); aLbDbFmtFromUsr.Enable( !bIsDBFmt ); if( !bIsDBFmt ) aLbDbFmtFromUsr.SetDefFormat( (*it)->nUsrNumFmt ); } aFlFormat.SetText( sTxt ); // to know later on, what ListBox was the "active", a Flag // is remembered in the 1st entry void* pPtr = pBox == &aLbTableCol ? &aLbTableCol : 0; aLbTableCol.SetEntryData( 0, pPtr ); } return 0; } IMPL_LINK( SwInsertDBColAutoPilot, HeaderHdl, Button*, pButton ) { if( pButton == &aCbTableHeadon ) { sal_Bool bEnable = aCbTableHeadon.IsChecked(); aRbHeadlColnms.Enable( bEnable ); aRbHeadlEmpty.Enable( bEnable ); } return 0; } static void lcl_InsTextInArr( const String& rTxt, _DB_Columns& rColArr ) { _DB_Column* pNew; sal_uInt16 nSttPos = 0, nFndPos; while( STRING_NOTFOUND != ( nFndPos = rTxt.Search( '\x0A', nSttPos )) ) { if( 1 < nFndPos ) { pNew = new _DB_Column( rTxt.Copy( nSttPos, nFndPos -1 ) ); rColArr.push_back( pNew ); } pNew = new _DB_Column; rColArr.push_back( pNew ); nSttPos = nFndPos + 1; } if( nSttPos < rTxt.Len() ) { pNew = new _DB_Column( rTxt.Copy( nSttPos ) ); rColArr.push_back( pNew ); } } sal_Bool SwInsertDBColAutoPilot::SplitTextToColArr( const String& rTxt, _DB_Columns& rColArr, sal_Bool bInsField ) { // create each of the database columns from the text again // and then save in an array // database columns are in <> and must be present in the columns' array: String sTxt( rTxt ); sal_uInt16 nFndPos, nEndPos, nSttPos = 0; while( STRING_NOTFOUND != ( nFndPos = sTxt.Search( cDBFldStart, nSttPos ))) { nSttPos = nFndPos + 1; if( STRING_NOTFOUND != ( nEndPos = sTxt.Search( cDBFldEnd, nSttPos+1 ))) { // Text in <> brackets found: what is it: SwInsDBColumn aSrch( sTxt.Copy( nSttPos, nEndPos - nSttPos ), 0); SwInsDBColumns::const_iterator it = aDBColumns.find( &aSrch ); if( it != aDBColumns.end() ) { // that is a valid field // so surely the text "before": const SwInsDBColumn& rFndCol = **it; _DB_Column* pNew; if( 1 < nSttPos ) { ::lcl_InsTextInArr( sTxt.Copy( 0, nSttPos-1 ), rColArr ); sTxt.Erase( 0, nSttPos-1 ); } sTxt.Erase( 0, (xub_StrLen)(rFndCol.sColumn.getLength() + 2) ); nSttPos = 0; sal_uInt16 nSubType = 0; sal_uLong nFormat; if( rFndCol.bHasFmt ) { if( rFndCol.bIsDBFmt ) nFormat = rFndCol.nDBNumFmt; else { nFormat = rFndCol.nUsrNumFmt; nSubType = nsSwExtendedSubType::SUB_OWN_FMT; } } else nFormat = 0; if( bInsField ) { SwWrtShell& rSh = pView->GetWrtShell(); SwDBFieldType aFldType( rSh.GetDoc(), aSrch.sColumn, aDBData ); pNew = new _DB_Column( rFndCol, *new SwDBField( (SwDBFieldType*)rSh.InsertFldType( aFldType ), nFormat ) ); if( nSubType ) pNew->DB_ColumnData.pField->SetSubType( nSubType ); } else pNew = new _DB_Column( rFndCol, nFormat ); rColArr.push_back( pNew ); } } } // don't forget the last text if( sTxt.Len() ) ::lcl_InsTextInArr( sTxt, rColArr ); return !rColArr.empty(); } void SwInsertDBColAutoPilot::DataToDoc( const Sequence& rSelection, Reference< XDataSource> xSource, Reference< XConnection> xConnection, Reference< sdbc::XResultSet > xResultSet ) { const Any* pSelection = rSelection.getLength() ? rSelection.getConstArray() : 0; SwWrtShell& rSh = pView->GetWrtShell(); //with the drag and drop interface no result set is initially available sal_Bool bDisposeResultSet = sal_False; // we don't have a cursor, so we have to create our own RowSet if ( !xResultSet.is() ) { xResultSet = SwNewDBMgr::createCursor(aDBData.sDataSource,aDBData.sCommand,aDBData.nCommandType,xConnection); bDisposeResultSet = xResultSet.is(); } Reference< sdbc::XRow > xRow(xResultSet, UNO_QUERY); if ( !xRow.is() ) return; rSh.StartAllAction(); sal_Bool bUndo = rSh.DoesUndo(); if( bUndo ) rSh.StartUndo( UNDO_EMPTY ); sal_Bool bAsTable = aRbAsTable.IsChecked(); SvNumberFormatter& rNumFmtr = *rSh.GetNumberFormatter(); if( rSh.HasSelection() ) rSh.DelRight(); ::std::auto_ptr pWait; Reference< XColumnsSupplier > xColsSupp( xResultSet, UNO_QUERY ); Reference xCols = xColsSupp->getColumns(); do{ // middle checked loop!! if( bAsTable ) // Daten als Tabelle einfuegen { rSh.DoUndo( sal_False ); sal_uInt16 n, nRows = 0, nCols = aLbTableCol.GetEntryCount(); if( aCbTableHeadon.IsChecked() ) nRows++; if( pSelection ) nRows = nRows + (sal_uInt16)rSelection.getLength(); else ++nRows; // prepare the array for the selected columns std::vector aColFlds; for( n = 0; n < nCols; ++n ) { SwInsDBColumn aSrch( aLbTableCol.GetEntry( n ), 0 ); SwInsDBColumns::const_iterator it = aDBColumns.find( &aSrch ); if( it != aDBColumns.end() ) aColFlds[ n ] = *it; else { OSL_ENSURE( !this, "database column not found" ); } } if( nCols != aColFlds.size() ) { OSL_ENSURE( !this, "not all database columns found" ); nCols = aColFlds.size(); } if(!nRows || !nCols) { OSL_ENSURE( !this, "wrong parameters" ); break; } const SwModuleOptions* pModOpt = SW_MOD()->GetModuleConfig(); sal_Bool bHTML = 0 != (::GetHtmlMode( pView->GetDocShell() ) & HTMLMODE_ON); rSh.InsertTable( pModOpt->GetInsTblFlags(bHTML), nRows, nCols, text::HoriOrientation::FULL, (pSelection ? pTAutoFmt : 0) ); rSh.MoveTable( GetfnTablePrev(), GetfnTableStart() ); if( pSelection && pTblSet ) SetTabSet(); SfxItemSet aTblSet( rSh.GetAttrPool(), RES_BOXATR_FORMAT, RES_BOXATR_VALUE ); sal_Bool bIsAutoUpdateCells = rSh.IsAutoUpdateCells(); rSh.SetAutoUpdateCells( sal_False ); if( aCbTableHeadon.IsChecked() ) { for( n = 0; n < nCols; ++n ) { if( aRbHeadlColnms.IsChecked() ) { rSh.SwEditShell::Insert2( aColFlds[ n ]->sColumn ); } rSh.GoNextCell(); } } else rSh.SetRowsToRepeat( 0 ); for( sal_Int32 i = 0 ; ; ++i ) { sal_Bool bBreak = sal_False; try { if(pSelection) { sal_Int32 nPos = 0; pSelection[i] >>= nPos; bBreak = !xResultSet->absolute(nPos); } else if(!i) bBreak = !xResultSet->first(); } catch (const Exception&) { bBreak = sal_True; } if(bBreak) break; for( n = 0; n < nCols; ++n ) { // at the very first time, NO GoNextCell, because we're // already in it. Also no GoNextCell after the Insert, // because an empty line is added at the end. if( i || n ) rSh.GoNextCell(); const SwInsDBColumn* pEntry = aColFlds[ n ]; Reference< XColumn > xColumn; xCols->getByName(pEntry->sColumn) >>= xColumn; Reference< XPropertySet > xColumnProps( xColumn, UNO_QUERY ); sal_Int32 eDataType = 0; if( xColumnProps.is() ) { Any aType = xColumnProps->getPropertyValue(C2U("Type")); aType >>= eDataType; } try { if( pEntry->bHasFmt ) { SwTblBoxNumFormat aNumFmt( pEntry->bIsDBFmt ? pEntry->nDBNumFmt : pEntry->nUsrNumFmt ); aTblSet.Put(aNumFmt); if( xColumn.is() ) { double fVal = xColumn->getDouble(); if( xColumn->wasNull() ) aTblSet.ClearItem( RES_BOXATR_VALUE ); else { if(rNumFmtr.GetType(aNumFmt.GetValue()) & NUMBERFORMAT_DATE) { ::Date aStandard(1,1,1900); if (*rNumFmtr.GetNullDate() != aStandard) fVal += (aStandard - *rNumFmtr.GetNullDate()); } aTblSet.Put( SwTblBoxValue( fVal )); } } else aTblSet.ClearItem( RES_BOXATR_VALUE ); rSh.SetTblBoxFormulaAttrs( aTblSet ); } //#i60207# don't insert binary data as string - creates a loop else if( DataType::BINARY == eDataType || DataType::VARBINARY == eDataType || DataType::LONGVARBINARY== eDataType || DataType::SQLNULL == eDataType || DataType::OTHER == eDataType || DataType::OBJECT == eDataType || DataType::DISTINCT == eDataType || DataType::STRUCT == eDataType || DataType::ARRAY == eDataType || DataType::BLOB == eDataType || DataType::CLOB == eDataType || DataType::REF == eDataType ) { // do nothing } else { rtl::OUString sVal = xColumn->getString(); if(!xColumn->wasNull()) { rSh.SwEditShell::Insert2( sVal ); } } } catch (const Exception& rExcept) { OSL_FAIL(rtl::OUStringToOString(rExcept.Message, osl_getThreadTextEncoding()).getStr()); (void)rExcept; } } if( !pSelection ) { if ( !xResultSet->next() ) break; } else if( i+1 >= rSelection.getLength() ) break; if( 10 == i ) pWait = ::std::auto_ptr(new SwWait( *pView->GetDocShell(), sal_True )); } rSh.MoveTable( GetfnTableCurr(), GetfnTableStart() ); if( !pSelection && ( pTblSet || pTAutoFmt )) { if( pTblSet ) SetTabSet(); if( pTAutoFmt ) rSh.SetTableAutoFmt( *pTAutoFmt ); } rSh.SetAutoUpdateCells( bIsAutoUpdateCells ); } else // add data as fields/text { _DB_Columns aColArr; if( SplitTextToColArr( aEdDbText.GetText(), aColArr, aRbAsField.IsChecked() ) ) { // now for each data set, we can iterate over the array // and add the data if( !rSh.IsSttPara() ) rSh.SwEditShell::SplitNode(); if( !rSh.IsEndPara() ) { rSh.SwEditShell::SplitNode(); rSh.SwCrsrShell::Left(1,CRSR_SKIP_CHARS); } rSh.DoUndo( sal_False ); SwTxtFmtColl* pColl = 0; { String sTmplNm( aLbDbParaColl.GetSelectEntry() ); if( sNoTmpl != sTmplNm ) { pColl = rSh.FindTxtFmtCollByName( sTmplNm ); if( !pColl ) { sal_uInt16 nId = SwStyleNameMapper::GetPoolIdFromUIName( sTmplNm, nsSwGetPoolIdFromName::GET_POOLID_TXTCOLL ); if( USHRT_MAX != nId ) pColl = rSh.GetTxtCollFromPool( nId ); else pColl = rSh.MakeTxtFmtColl( sTmplNm ); } rSh.SetTxtFmtColl( pColl ); } } // for adding as fields -> insert a "NextField" after // every data set SwDBFormatData aDBFormatData; Reference< XMultiServiceFactory > xMgr( ::comphelper::getProcessServiceFactory() ); Reference< XComponentContext > xContext( ::comphelper::getProcessComponentContext() ); aDBFormatData.xFormatter = Reference(util::NumberFormatter::create(xContext), UNO_QUERY_THROW) ; Reference xSourceProps(xSource, UNO_QUERY); if(xSourceProps.is()) { Any aFormats = xSourceProps->getPropertyValue(C2U("NumberFormatsSupplier")); if(aFormats.hasValue()) { Reference< util::XNumberFormatsSupplier> xSuppl; aFormats >>= xSuppl; if(xSuppl.is()) { Reference< XPropertySet > xSettings = xSuppl->getNumberFormatSettings(); Any aNull = xSettings->getPropertyValue(C2U("NullDate")); aNull >>= aDBFormatData.aNullDate; if(aDBFormatData.xFormatter.is()) aDBFormatData.xFormatter->attachNumberFormatsSupplier(xSuppl); } } } aDBFormatData.aLocale = LanguageTag( rSh.GetCurLang() ).getLocale(); SwDBNextSetField aNxtDBFld( (SwDBNextSetFieldType*)rSh. GetFldType( 0, RES_DBNEXTSETFLD ), rtl::OUString("1"), aEmptyStr, aDBData ); sal_Bool bSetCrsr = sal_True; sal_uInt16 n = 0, nCols = aColArr.size(); ::sw::mark::IMark* pMark = NULL; for( sal_Int32 i = 0 ; ; ++i ) { sal_Bool bBreak = sal_False; try { if(pSelection) { sal_Int32 nPos = 0; pSelection[i] >>= nPos; bBreak = !xResultSet->absolute(nPos); } else if(!i) bBreak = !xResultSet->first(); } catch (const Exception&) { bBreak = sal_True; } if(bBreak) break; for( n = 0; n < nCols; ++n ) { _DB_Column* pDBCol = &aColArr[ n ]; String sIns; switch( pDBCol->eColType ) { case _DB_Column::DB_FILLTEXT: sIns = *pDBCol->DB_ColumnData.pText; break; case _DB_Column::DB_SPLITPARA: rSh.SplitNode(); // when the template is not the same as the follow template, // the selected has to be set newly if( pColl && &pColl->GetNextTxtFmtColl() != pColl ) rSh.SetTxtFmtColl( pColl ); break; case _DB_Column::DB_COL_FIELD: { SwDBField *const pFld = static_cast( pDBCol->DB_ColumnData.pField->CopyField()); double nValue = DBL_MAX; Reference< XPropertySet > xColumnProps; xCols->getByName(pDBCol->pColInfo->sColumn) >>= xColumnProps; pFld->SetExpansion( SwNewDBMgr::GetDBField( xColumnProps, aDBFormatData, &nValue ) ); if( DBL_MAX != nValue ) { Any aType = xColumnProps->getPropertyValue(C2U("Type")); sal_Int32 eDataType = 0; aType >>= eDataType; if( DataType::DATE == eDataType || DataType::TIME == eDataType || DataType::TIMESTAMP == eDataType) { ::Date aStandard(1,1,1900); ::Date aCompare(aDBFormatData.aNullDate.Day , aDBFormatData.aNullDate.Month, aDBFormatData.aNullDate.Year); if(aStandard != aCompare) nValue += (aStandard - aCompare); } pFld->ChgValue( nValue, sal_True ); } pFld->SetInitialized(); rSh.Insert( *pFld ); delete pFld; } break; case _DB_Column::DB_COL_TEXT: { double nValue = DBL_MAX; Reference< XPropertySet > xColumnProps; xCols->getByName(pDBCol->pColInfo->sColumn) >>= xColumnProps; sIns = SwNewDBMgr::GetDBField( xColumnProps, aDBFormatData, &nValue ); if( pDBCol->DB_ColumnData.nFormat && DBL_MAX != nValue ) { Color* pCol; if(rNumFmtr.GetType(pDBCol->DB_ColumnData.nFormat) & NUMBERFORMAT_DATE) { ::Date aStandard(1,1,1900); if (*rNumFmtr.GetNullDate() != aStandard) nValue += (aStandard - *rNumFmtr.GetNullDate()); } rNumFmtr.GetOutputString( nValue, pDBCol->DB_ColumnData.nFormat, sIns, &pCol ); } } break; } if( sIns.Len() ) rSh.Insert( sIns ); if( bSetCrsr && sIns.Len() ) { // to the beginning and set a mark, so that // the cursor can be set to the initial position // at the end. rSh.SwCrsrShell::MovePara( GetfnParaCurr(), GetfnParaStart() ); pMark = rSh.SetBookmark( KeyCode(), ::rtl::OUString(), ::rtl::OUString(), IDocumentMarkAccess::UNO_BOOKMARK ); rSh.SwCrsrShell::MovePara( GetfnParaCurr(), GetfnParaEnd() ); bSetCrsr = sal_False; } } if( !pSelection ) { sal_Bool bNext = xResultSet->next(); if(!bNext) break; } else if( i+1 >= rSelection.getLength() ) break; if( aRbAsField.IsChecked() ) rSh.Insert( aNxtDBFld ); if( !rSh.IsSttPara() ) rSh.SwEditShell::SplitNode(); if( 10 == i ) pWait = ::std::auto_ptr(new SwWait( *pView->GetDocShell(), sal_True )); } if( !bSetCrsr && pMark != NULL) { rSh.SetMark(); rSh.GotoMark( pMark ); rSh.getIDocumentMarkAccess()->deleteMark( pMark ); break; } } } // write configuration Commit(); }while( sal_False ); // middle checked loop if( bUndo ) { rSh.DoUndo( sal_True ); rSh.AppendUndoForInsertFromDB( bAsTable ); rSh.EndUndo( UNDO_EMPTY ); } rSh.ClearMark(); rSh.EndAllAction(); if ( bDisposeResultSet ) ::comphelper::disposeComponent(xResultSet); } void SwInsertDBColAutoPilot::SetTabSet() { SwWrtShell& rSh = pView->GetWrtShell(); const SfxPoolItem* pItem; if( pTAutoFmt ) { if( pTAutoFmt->IsFrame() ) { // border is from AutoFormat pTblSet->ClearItem( RES_BOX ); pTblSet->ClearItem( SID_ATTR_BORDER_INNER ); } if( pTAutoFmt->IsBackground() ) { pTblSet->ClearItem( RES_BACKGROUND ); pTblSet->ClearItem( SID_ATTR_BRUSH_ROW ); pTblSet->ClearItem( SID_ATTR_BRUSH_TABLE ); } } else { // remove the defaults again, it makes no sense to set them SvxBrushItem aBrush( RES_BACKGROUND ); static sal_uInt16 aIds[3] = { RES_BACKGROUND, SID_ATTR_BRUSH_ROW, SID_ATTR_BRUSH_TABLE }; for( int i = 0; i < 3; ++i ) if( SFX_ITEM_SET == pTblSet->GetItemState( aIds[ i ], sal_False, &pItem ) && *pItem == aBrush ) pTblSet->ClearItem( aIds[ i ] ); } if( SFX_ITEM_SET == pTblSet->GetItemState( FN_PARAM_TABLE_NAME, sal_False, &pItem ) && ((const SfxStringItem*)pItem)->GetValue() == rSh.GetTableFmt()->GetName() ) pTblSet->ClearItem( FN_PARAM_TABLE_NAME ); rSh.MoveTable( GetfnTableCurr(), GetfnTableStart() ); rSh.SetMark(); rSh.MoveTable( GetfnTableCurr(), GetfnTableEnd() ); ItemSetToTableParam( *pTblSet, rSh ); rSh.ClearMark(); rSh.MoveTable( GetfnTableCurr(), GetfnTableStart() ); } _DB_ColumnConfigData::~_DB_ColumnConfigData() {} static Sequence lcl_createSourceNames(const String& rNodeName) { Sequence aSourceNames(11); rtl::OUString* pNames = aSourceNames.getArray(); String sTmp( rNodeName ); const xub_StrLen nPos = sTmp.Len(); pNames[0] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/DataSource" )); pNames[1] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/Command" )); pNames[2] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/CommandType" )); pNames[3] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/ColumnsToText" )); pNames[4] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/ColumnsToTable" )); pNames[5] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/ParaStyle" )); pNames[6] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/TableAutoFormat" )); pNames[7] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/IsTable" )); pNames[8] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/IsField" )); pNames[9] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/IsHeadlineOn" )); pNames[10] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/IsEmptyHeadline" )); return aSourceNames; } static Sequence lcl_CreateSubNames( const String& rSubNodeName ) { Sequence aSubSourceNames(6); rtl::OUString* pNames = aSubSourceNames.getArray(); String sTmp( rSubNodeName ); const xub_StrLen nPos = sTmp.Len(); pNames[0] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/ColumnName" )); pNames[1] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/ColumnIndex" )); pNames[2] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/IsNumberFormat" )); pNames[3] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/IsNumberFormatFromDataBase" )); pNames[4] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/NumberFormat" )); pNames[5] = sTmp.ReplaceAscii( nPos, STRING_MAXLEN, RTL_CONSTASCII_STRINGPARAM( "/NumberFormatLocale" )); return aSubSourceNames; } static rtl::OUString lcl_CreateUniqueName(const Sequence& aNames) { sal_Int32 nIdx = aNames.getLength(); const rtl::OUString* pNames = aNames.getConstArray(); rtl::OUString sTest(C2U("_")); rtl::OUString sRet; while(sal_True) { sRet = sTest; sRet += rtl::OUString::valueOf(nIdx++); sal_Bool bFound = sal_False; for(sal_Int32 i = 0; i < aNames.getLength(); i++) { if(pNames[i] == sRet) { bFound = sal_True; break; } } if(!bFound) break; } return sRet; } void SwInsertDBColAutoPilot::Notify( const ::com::sun::star::uno::Sequence< rtl::OUString >& ) {} void SwInsertDBColAutoPilot::Commit() { Sequence aNames = GetNodeNames(rtl::OUString()); const rtl::OUString* pNames = aNames.getArray(); //remove entries that contain this data source + table at first for(sal_Int32 nNode = 0; nNode < aNames.getLength(); nNode++) { Sequence aSourceNames(2); rtl::OUString* pSourceNames = aSourceNames.getArray(); pSourceNames[0] = pNames[nNode]; pSourceNames[0] += C2U("/DataSource"); pSourceNames[1] = pNames[nNode]; pSourceNames[1] += C2U("/Command"); Sequence aSourceProperties = GetProperties(aSourceNames); const Any* pSourceProps = aSourceProperties.getArray(); rtl::OUString sSource, sCommand; pSourceProps[0] >>= sSource; pSourceProps[1] >>= sCommand; if(sSource.equals(aDBData.sDataSource) && sCommand.equals(aDBData.sCommand)) { Sequence aElements(1); aElements.getArray()[0] = pNames[nNode]; ClearNodeElements(rtl::OUString(), aElements); } } aNames = GetNodeNames(rtl::OUString()); rtl::OUString sNewNode = lcl_CreateUniqueName(aNames); Sequence aNodeNames = lcl_createSourceNames(sNewNode); Sequence aValues(aNodeNames.getLength()); PropertyValue* pValues = aValues.getArray(); const rtl::OUString* pNodeNames = aNodeNames.getConstArray(); rtl::OUString sSlash(C2U("/")); for(sal_Int32 i = 0; i < aNodeNames.getLength(); i++) { pValues[i].Name = sSlash; pValues[i].Name += pNodeNames[i]; } pValues[0].Value <<= rtl::OUString(aDBData.sDataSource); pValues[1].Value <<= rtl::OUString(aDBData.sCommand); pValues[2].Value <<= aDBData.nCommandType; pValues[3].Value <<= rtl::OUString(aEdDbText.GetText()); String sTmp; for( sal_uInt16 n = 0, nCnt = aLbTableCol.GetEntryCount(); n < nCnt; ++n ) ( sTmp += aLbTableCol.GetEntry( n ) ) += '\x0a'; if( sTmp.Len() ) pValues[4].Value <<= rtl::OUString(sTmp); if( sNoTmpl != (sTmp = aLbDbParaColl.GetSelectEntry()) ) pValues[5].Value <<= rtl::OUString(sTmp); if( pTAutoFmt ) pValues[6].Value <<= rtl::OUString(pTAutoFmt->GetName()); const Type& rBoolType = ::getBooleanCppuType(); sal_Bool bTmp = aRbAsTable.IsChecked(); pValues[7].Value.setValue(&bTmp, rBoolType); bTmp = aRbAsField.IsChecked(); pValues[8].Value.setValue(&bTmp, rBoolType); bTmp = aCbTableHeadon.IsChecked(); pValues[9].Value.setValue(&bTmp, rBoolType); bTmp = aRbHeadlEmpty.IsChecked(); pValues[10].Value.setValue(&bTmp, rBoolType); SetSetProperties(rtl::OUString(), aValues); sNewNode += C2U("/ColumnSet"); rtl::OUString sDelim("/__"); LanguageType ePrevLang = (LanguageType)-1; rtl::OUString sPrevLang; SvNumberFormatter& rNFmtr = *pView->GetWrtShell().GetNumberFormatter(); for(sal_uInt16 nCol = 0; nCol < aDBColumns.size(); nCol++) { rtl::OUString sColumnNode = sNewNode; SwInsDBColumn* pColumn = aDBColumns[nCol]; String sColumnInsertNode(sColumnNode); sColumnInsertNode += sDelim; if( nCol < 100 ) sColumnInsertNode += '0'; if( nCol < 10 ) sColumnInsertNode += '0'; sColumnInsertNode += String::CreateFromInt32( nCol ); Sequence aSubNodeNames = lcl_CreateSubNames(sColumnInsertNode); Sequence aSubValues(aSubNodeNames.getLength()); PropertyValue* pSubValues = aSubValues.getArray(); const rtl::OUString* pSubNodeNames = aSubNodeNames.getConstArray(); sal_Int32 i; for( i = 0; i < aSubNodeNames.getLength(); i++) pSubValues[i].Name = pSubNodeNames[i]; pSubValues[0].Value <<= pColumn->sColumn; pSubValues[1].Value <<= i; sal_Bool bVal = pColumn->bHasFmt; pSubValues[2].Value.setValue(&bVal, rBoolType); bVal = pColumn->bIsDBFmt; pSubValues[3].Value.setValue(&bVal, rBoolType); SwStyleNameMapper::FillUIName( RES_POOLCOLL_STANDARD, sTmp ); const SvNumberformat* pNF = rNFmtr.GetEntry( pColumn->nUsrNumFmt ); LanguageType eLang; if( pNF ) { pSubValues[4].Value <<= rtl::OUString(pNF->GetFormatstring()); eLang = pNF->GetLanguage(); } else { pSubValues[4].Value <<= rtl::OUString(sTmp); eLang = GetAppLanguage(); } if( eLang != ePrevLang ) { /* FIXME-BCP47: handle language tags! */ lang::Locale aLocale( LanguageTag( eLang ).getLocale()); /* umm.. what's this string anyway, "Country-Language" instead of * "Language-Country" ??? */ (( sPrevLang = aLocale.Country ) += rtl::OUString( '-' )) += aLocale.Language; ePrevLang = eLang; } pSubValues[5].Value <<= sPrevLang; SetSetProperties(sColumnNode, aSubValues); } } void SwInsertDBColAutoPilot::Load() { Sequence aNames = GetNodeNames(rtl::OUString()); const rtl::OUString* pNames = aNames.getArray(); SvNumberFormatter& rNFmtr = *pView->GetWrtShell().GetNumberFormatter(); for(sal_Int32 nNode = 0; nNode < aNames.getLength(); nNode++) { //search for entries with the appropriate data source and table Sequence aSourceNames = lcl_createSourceNames(pNames[nNode]); Sequence< Any> aDataSourceProps = GetProperties(aSourceNames); const Any* pDataSourceProps = aDataSourceProps.getConstArray(); rtl::OUString sSource, sCommand; sal_Int16 nCommandType; pDataSourceProps[0] >>= sSource; pDataSourceProps[1] >>= sCommand; pDataSourceProps[2] >>= nCommandType; if(sSource.equals(aDBData.sDataSource) && sCommand.equals(aDBData.sCommand)) { _DB_ColumnConfigData* pNewData = new _DB_ColumnConfigData; pNewData->sSource = sSource; pNewData->sTable = sCommand; pDataSourceProps[3] >>= pNewData->sEdit; pDataSourceProps[4] >>= pNewData->sTblList; pDataSourceProps[5] >>= pNewData->sTmplNm; pDataSourceProps[6] >>= pNewData->sTAutoFmtNm; if(pDataSourceProps[7].hasValue()) pNewData->bIsTable = *(sal_Bool*)pDataSourceProps[7].getValue(); if(pDataSourceProps[8].hasValue()) pNewData->bIsField = *(sal_Bool*)pDataSourceProps[8].getValue(); if(pDataSourceProps[9].hasValue()) pNewData->bIsHeadlineOn = *(sal_Bool*)pDataSourceProps[9].getValue(); if(pDataSourceProps[10].hasValue()) pNewData->bIsEmptyHeadln = *(sal_Bool*)pDataSourceProps[10].getValue(); rtl::OUString sSubNodeName(pNames[nNode]); sSubNodeName += C2U("/ColumnSet/"); Sequence aSubNames = GetNodeNames(sSubNodeName); const rtl::OUString* pSubNames = aSubNames.getConstArray(); for(sal_Int32 nSub = 0; nSub < aSubNames.getLength(); nSub++) { rtl::OUString sSubSubNodeName(sSubNodeName); sSubSubNodeName += pSubNames[nSub]; Sequence aSubNodeNames = lcl_CreateSubNames(sSubSubNodeName); Sequence< Any> aSubProps = GetProperties(aSubNodeNames); const Any* pSubProps = aSubProps.getConstArray(); rtl::OUString sColumn; pSubProps[0] >>= sColumn; //check for existance of the loaded column name sal_Bool bFound = sal_False; for(sal_uInt16 nRealColumn = 0; nRealColumn < aDBColumns.size(); nRealColumn++) { if(aDBColumns[nRealColumn]->sColumn == sColumn) { bFound = sal_True; break; } } if(!bFound) continue; sal_Int16 nIndex = 0; pSubProps[1] >>= nIndex; SwInsDBColumn* pInsDBColumn = new SwInsDBColumn(sColumn, nIndex); if(pSubProps[2].hasValue()) pInsDBColumn->bHasFmt = *(sal_Bool*)pSubProps[2].getValue(); if(pSubProps[3].hasValue()) pInsDBColumn->bIsDBFmt = *(sal_Bool*)pSubProps[3].getValue(); pSubProps[4] >>= pInsDBColumn->sUsrNumFmt; rtl::OUString sNumberFormatLocale; pSubProps[5] >>= sNumberFormatLocale; /* FIXME-BCP47: handle language tags, and cope with the wrong * Country-Language string that * SwInsertDBColAutoPilot::Commit() writes so far! */ lang::Locale aLocale; aLocale.Language = sNumberFormatLocale.copy(0, 2); aLocale.Country = sNumberFormatLocale.copy(3, 2); pInsDBColumn->eUsrNumFmtLng = LanguageTag( aLocale ).getLanguageType(); pInsDBColumn->nUsrNumFmt = rNFmtr.GetEntryKey( pInsDBColumn->sUsrNumFmt, pInsDBColumn->eUsrNumFmtLng ); pNewData->aDBColumns.insert(pInsDBColumn); } sal_uInt16 n = 0; String sTmp( pNewData->sTblList ); if( sTmp.Len() ) { do { String sEntry( sTmp.GetToken( 0, '\x0a', n ) ); //preselect column - if they still exist! if(aLbTblDbColumn.GetEntryPos(sEntry) != LISTBOX_ENTRY_NOTFOUND) { aLbTableCol.InsertEntry( sEntry ); aLbTblDbColumn.RemoveEntry( sEntry ); } } while( n < sTmp.Len() ); if( !aLbTblDbColumn.GetEntryCount() ) { aIbDbcolAllTo.Enable( sal_False ); aIbDbcolOneTo.Enable( sal_False ); } aIbDbcolOneFrom.Enable( sal_True ); aIbDbcolAllFrom.Enable( sal_True ); } aEdDbText.SetText( pNewData->sEdit ); sTmp = pNewData->sTmplNm; if( sTmp.Len() ) aLbDbParaColl.SelectEntry( sTmp ); else aLbDbParaColl.SelectEntryPos( 0 ); delete pTAutoFmt, pTAutoFmt = 0; sTmp = pNewData->sTAutoFmtNm; if( sTmp.Len() ) { // then load the AutoFmt file and look for Autoformat first SwTableAutoFmtTbl aAutoFmtTbl; aAutoFmtTbl.Load(); for( sal_uInt16 nAutoFmt = aAutoFmtTbl.size(); nAutoFmt; ) if( sTmp == aAutoFmtTbl[ --nAutoFmt ].GetName() ) { pTAutoFmt = new SwTableAutoFmt( aAutoFmtTbl[ nAutoFmt ] ); break; } } aRbAsTable.Check( pNewData->bIsTable ); aRbAsField.Check( pNewData->bIsField ); aRbAsText.Check( !pNewData->bIsTable && !pNewData->bIsField ); aCbTableHeadon.Check( pNewData->bIsHeadlineOn ); aRbHeadlColnms.Check( !pNewData->bIsEmptyHeadln ); aRbHeadlEmpty.Check( pNewData->bIsEmptyHeadln ); HeaderHdl(&aCbTableHeadon); // now copy the user defined Numberformat strings to the // Shell. Then only these are available as ID for( n = 0; n < aDBColumns.size() ; ++n ) { SwInsDBColumn& rSet = *aDBColumns[ n ]; for( sal_uInt16 m = 0; m < pNewData->aDBColumns.size() ; ++m ) { SwInsDBColumn& rGet = *pNewData->aDBColumns[ m ]; if(rGet.sColumn == rSet.sColumn) { if( rGet.bHasFmt && !rGet.bIsDBFmt ) { rSet.bIsDBFmt = sal_False; rSet.nUsrNumFmt = rNFmtr.GetEntryKey( rGet.sUsrNumFmt, rGet.eUsrNumFmtLng ); if( NUMBERFORMAT_ENTRY_NOT_FOUND == rSet.nUsrNumFmt ) { sal_Int32 nCheckPos; short nType; rNFmtr.PutEntry( rGet.sUsrNumFmt, nCheckPos, nType, rSet.nUsrNumFmt, rGet.eUsrNumFmtLng ); } } break; } } } // when the cursor is inside of a table, table must NEVER be selectable if( !aRbAsTable.IsEnabled() && aRbAsTable.IsChecked() ) aRbAsField.Check( sal_True ); delete pNewData; break; } } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */