/************************************************************************* * * $RCSfile: gridctrl.cxx,v $ * * $Revision: 1.22 $ * * last change: $Author: fs $ $Date: 2001-05-16 14:16:06 $ * * The Contents of this file are made available subject to the terms of * either of the following licenses * * - GNU Lesser General Public License Version 2.1 * - Sun Industry Standards Source License Version 1.1 * * Sun Microsystems Inc., October, 2000 * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2000 by Sun Microsystems, Inc. * 901 San Antonio Road, Palo Alto, CA 94303, USA * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software Foundation. * * This library 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 for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA * * * Sun Industry Standards Source License Version 1.1 * ================================================= * The contents of this file are subject to the Sun Industry Standards * Source License Version 1.1 (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.openoffice.org/license.html. * * Software provided under this License is provided on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, * WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS, * MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING. * See the License for the specific provisions governing your rights and * obligations concerning the Software. * * The Initial Developer of the Original Code is: Sun Microsystems, Inc. * * Copyright: 2000 by Sun Microsystems, Inc. * * All Rights Reserved. * * Contributor(s): _______________________________________ * * ************************************************************************/ #pragma hdrstop #ifndef _SVX_FMHELP_HRC #include "fmhelp.hrc" #endif #ifndef _SVX_GRIDCTRL_HXX #include "gridctrl.hxx" #endif #ifndef _SVX_GRIDCELL_HXX #include "gridcell.hxx" #endif #ifndef _SVX_FMTOOLS_HXX #include "fmtools.hxx" #endif #ifndef _SVX_FMPROP_HXX #include "fmprop.hxx" #endif #ifndef _SVX_FMPROP_HRC #include "fmprop.hrc" #endif #ifndef _SVTOOLS_STRINGTRANSFER_HXX_ #include #endif #ifndef _COM_SUN_STAR_SDB_XRESULTSETACCESS_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDBC_XRESULTSETUPDATE_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDBCX_PRIVILEGE_HPP_ #include #endif #ifndef _COM_SUN_STAR_CONTAINER_XCHILD_HPP_ #include #endif #ifndef _COM_SUN_STAR_UTIL_XNUMBERFORMATTER_HPP_ #include #endif #ifndef _COM_SUN_STAR_UTIL_XNUMBERFORMATSSUPPLIER_HPP_ #include #endif #ifndef _COM_SUN_STAR_UTIL_XCLONEABLE_HPP_ #include #endif #ifndef _COM_SUN_STAR_BEANS_XPROPERTYSET_HPP_ #include #endif #ifndef _COM_SUN_STAR_BEANS_PROPERTYCHANGEEVENT_HPP_ #include #endif #ifndef _COMPHELPER_EXTRACT_HXX_ #include #endif #ifndef _TOOLS_RESID_HXX //autogen #include #endif #ifndef _SV_SYSTEM_HXX //autogen #include #endif #ifndef _SV_SOUND_HXX //autogen #include #endif #ifndef _SV_DRAG_HXX //autogen #include #endif #ifndef _SV_MENU_HXX //autogen #include #endif #ifndef _SV_CLIP_HXX //autogen #include #endif #ifndef _SVX_FMRESIDS_HRC #include "fmresids.hrc" #endif #ifndef _SVX_SVXIDS_HRC #include "svxids.hrc" #endif #ifndef _SHL_HXX #include #endif #ifndef _SVX_DIALMGR_HXX #include "dialmgr.hxx" #endif #ifndef _SVX_FMSERVS_HXX #include "fmservs.hxx" #endif #define CURSORPOSITION_UNKNOWN -2 #define HANDLE_ID 0 String INVALIDTEXT = String::CreateFromAscii("###"); String OBJECTTEXT = String::CreateFromAscii(""); #ifndef _COMPHELPER_STLTYPES_HXX_ #include #endif #ifndef _CONNECTIVITY_DBTOOLS_HXX_ #include #endif #ifndef _DBHELPER_DBCONVERSION_HXX_ #include #endif #ifndef _COMPHELPER_DATETIME_HXX_ #include #endif #ifndef _COMPHELPER_PROPERTY_HXX_ #include #endif #ifndef _TRACE_HXX_ #include "trace.hxx" #endif using namespace ::dbtools; using namespace ::svxform; using namespace ::svt; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::datatransfer; #define ROWSTATUS(row) !row.Is() ? "NULL" : row->GetStatus() == GRS_CLEAN ? "CLEAN" : row->GetStatus() == GRS_MODIFIED ? "MODIFIED" : row->GetStatus() == GRS_DELETED ? "DELETED" : "INVALID" //============================================================================== class GridFieldValueListener; DECLARE_STL_MAP(sal_uInt16, GridFieldValueListener*, ::std::less, ColumnFieldValueListeners); //============================================================================== DBG_NAME(GridFieldValueListener); class GridFieldValueListener : protected ::comphelper::OPropertyChangeListener { osl::Mutex m_aMutex; DbGridControl& m_rParent; ::comphelper::OPropertyChangeMultiplexer* m_pRealListener; sal_uInt16 m_nId; sal_Int16 m_nSuspended; sal_Bool m_bDisposed : 1; public: GridFieldValueListener(DbGridControl& _rParent, const ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet >& xField, sal_uInt16 _nId); virtual ~GridFieldValueListener(); virtual void _propertyChanged(const ::com::sun::star::beans::PropertyChangeEvent& evt) throw( ::com::sun::star::uno::RuntimeException ); void suspend() { ++m_nSuspended; } void resume() { --m_nSuspended; } void dispose(); }; //------------------------------------------------------------------------------ GridFieldValueListener::GridFieldValueListener(DbGridControl& _rParent, const ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet >& _rField, sal_uInt16 _nId) :OPropertyChangeListener(m_aMutex) ,m_rParent(_rParent) ,m_nId(_nId) ,m_pRealListener(NULL) ,m_bDisposed(sal_False) ,m_nSuspended(0) { DBG_CTOR(GridFieldValueListener, NULL); if (_rField.is()) { m_pRealListener = new ::comphelper::OPropertyChangeMultiplexer(this, _rField); m_pRealListener->addProperty(FM_PROP_VALUE); m_pRealListener->acquire(); } } //------------------------------------------------------------------------------ GridFieldValueListener::~GridFieldValueListener() { DBG_DTOR(GridFieldValueListener, NULL); dispose(); } //------------------------------------------------------------------------------ void GridFieldValueListener::_propertyChanged(const ::com::sun::star::beans::PropertyChangeEvent& _evt) throw( ::com::sun::star::uno::RuntimeException ) { DBG_ASSERT(m_nSuspended>=0, "GridFieldValueListener::_propertyChanged : resume > suspend !"); if (m_nSuspended <= 0) m_rParent.FieldValueChanged(m_nId, _evt); } //------------------------------------------------------------------------------ void GridFieldValueListener::dispose() { if (m_bDisposed) { DBG_ASSERT(m_pRealListener == NULL, "GridFieldValueListener::dispose : inconsistent !"); return; } if (m_pRealListener) { m_pRealListener->dispose(); m_pRealListener->release(); m_pRealListener = NULL; } m_bDisposed = sal_True; m_rParent.FieldListenerDisposing(m_nId); } //============================================================================== class DisposeListenerGridBridge : public FmXDisposeListener { osl::Mutex m_aMutex; DbGridControl& m_rParent; FmXDisposeMultiplexer* m_pRealListener; public: DisposeListenerGridBridge( DbGridControl& _rParent, const ::com::sun::star::uno::Reference< ::com::sun::star::lang::XComponent >& _rxObject, sal_Int16 _rId = -1); virtual ~DisposeListenerGridBridge(); virtual void disposing(const ::com::sun::star::lang::EventObject& _rEvent, sal_Int16 _nId) throw( ::com::sun::star::uno::RuntimeException ) { m_rParent.disposing(_nId, _rEvent); } }; //============================================================================== DBG_NAME(DisposeListenerGridBridge); //------------------------------------------------------------------------------ DisposeListenerGridBridge::DisposeListenerGridBridge(DbGridControl& _rParent, const ::com::sun::star::uno::Reference< ::com::sun::star::lang::XComponent >& _rxObject, sal_Int16 _rId) :FmXDisposeListener(m_aMutex) ,m_rParent(_rParent) ,m_pRealListener(NULL) { DBG_CTOR(DisposeListenerGridBridge,NULL); if (_rxObject.is()) { m_pRealListener = new FmXDisposeMultiplexer(this, _rxObject, _rId); m_pRealListener->acquire(); } } //------------------------------------------------------------------------------ DisposeListenerGridBridge::~DisposeListenerGridBridge() { if (m_pRealListener) { m_pRealListener->dispose(); m_pRealListener->release(); m_pRealListener = NULL; } DBG_DTOR(DisposeListenerGridBridge,NULL); } //============================================================================== static sal_uInt16 ControlMap[] = { DbGridControl::NavigationBar::RECORD_TEXT, DbGridControl::NavigationBar::RECORD_ABSOLUTE, DbGridControl::NavigationBar::RECORD_OF, DbGridControl::NavigationBar::RECORD_COUNT, DbGridControl::NavigationBar::RECORD_FIRST, DbGridControl::NavigationBar::RECORD_NEXT, DbGridControl::NavigationBar::RECORD_PREV, DbGridControl::NavigationBar::RECORD_LAST, DbGridControl::NavigationBar::RECORD_NEW, 0 }; //------------------------------------------------------------------------------ sal_Bool CompareBookmark(const ::com::sun::star::uno::Any& aLeft, const ::com::sun::star::uno::Any& aRight) { return ::comphelper::compare(aLeft, aRight); } //============================================================================== class FmXGridSourcePropListener : public ::comphelper::OPropertyChangeListener { DbGridControl* m_pParent; // a DbGridControl has no mutex, so we use our own as the base class expects one osl::Mutex m_aMutex; sal_Int16 m_nSuspended; public: FmXGridSourcePropListener(DbGridControl* _pParent); void suspend() { ++m_nSuspended; } void resume() { --m_nSuspended; } virtual void _propertyChanged(const ::com::sun::star::beans::PropertyChangeEvent& evt) throw( ::com::sun::star::uno::RuntimeException ); }; //------------------------------------------------------------------------------ FmXGridSourcePropListener::FmXGridSourcePropListener(DbGridControl* _pParent) :OPropertyChangeListener(m_aMutex) ,m_pParent(_pParent) ,m_nSuspended(0) { DBG_ASSERT(m_pParent, "FmXGridSourcePropListener::FmXGridSourcePropListener : invalid parent !"); } //------------------------------------------------------------------------------ void FmXGridSourcePropListener::_propertyChanged(const ::com::sun::star::beans::PropertyChangeEvent& evt) throw( ::com::sun::star::uno::RuntimeException ) { DBG_ASSERT(m_nSuspended>=0, "FmXGridSourcePropListener::_propertyChanged : resume > suspend !"); if (m_nSuspended <= 0) m_pParent->DataSourcePropertyChanged(evt); } //============================================================================== //------------------------------------------------------------------------------ DbGridControl::NavigationBar::AbsolutePos::AbsolutePos(Window* pParent, WinBits nStyle) :NumericField(pParent, nStyle) { SetMin(1); SetFirst(1); SetSpinSize(1); International rInt = GetInternational(); rInt.EnableNumThousandSep(sal_False); SetInternational(rInt); SetDecimalDigits(0); SetStrictFormat(sal_True); } //------------------------------------------------------------------------------ void DbGridControl::NavigationBar::AbsolutePos::KeyInput(const KeyEvent& rEvt) { if (rEvt.GetKeyCode() == KEY_RETURN && GetText().Len()) { sal_Int32 nRecord = GetValue(); if (nRecord < GetMin() || nRecord > GetMax()) return; else ((NavigationBar*)GetParent())->PositionDataSource(nRecord); } else if (rEvt.GetKeyCode() == KEY_TAB) GetParent()->GetParent()->GrabFocus(); else NumericField::KeyInput(rEvt); } //------------------------------------------------------------------------------ void DbGridControl::NavigationBar::AbsolutePos::LoseFocus() { NumericField::LoseFocus(); sal_Int32 nRecord = GetValue(); if (nRecord < GetMin() || nRecord > GetMax()) return; else { ((NavigationBar*)GetParent())->PositionDataSource(nRecord); ((NavigationBar*)GetParent())->InvalidateState(NavigationBar::RECORD_ABSOLUTE); } } //------------------------------------------------------------------------------ void DbGridControl::NavigationBar::PositionDataSource(sal_Int32 nRecord) { if (m_bPositioning) return; // the MoveToPosition may cause a LoseFocus which would lead to a second MoveToPosition, so protect agains this // recursion // 68167 - 13.08.99 - FS m_bPositioning = sal_True; ((DbGridControl*)GetParent())->MoveToPosition(nRecord - 1); m_bPositioning = sal_False; } //------------------------------------------------------------------------------ DbGridControl::NavigationBar::NavigationBar(Window* pParent, WinBits nStyle) :Control(pParent, nStyle) ,m_aRecordText(this) ,m_aAbsolute(this) ,m_aRecordOf(this) ,m_aRecordCount(this, WinBits(WB_CENTER)) ,m_aFirstBtn(this, WB_RECTSTYLE|WB_NOPOINTERFOCUS) ,m_aPrevBtn(this, WB_REPEAT|WB_RECTSTYLE|WB_NOPOINTERFOCUS) ,m_aNextBtn(this, WB_REPEAT|WB_RECTSTYLE|WB_NOPOINTERFOCUS) ,m_aLastBtn(this, WB_RECTSTYLE|WB_NOPOINTERFOCUS) ,m_aNewBtn(this, WB_RECTSTYLE|WB_NOPOINTERFOCUS) ,m_nDefaultWidth(0) ,m_nCurrentPos(-1) ,m_bPositioning(sal_False) { m_aFirstBtn.SetSymbol(SYMBOL_FIRST); m_aPrevBtn.SetSymbol(SYMBOL_PREV); m_aNextBtn.SetSymbol(SYMBOL_NEXT); m_aLastBtn.SetSymbol(SYMBOL_LAST); m_aNewBtn.SetImage(((DbGridControl*)pParent)->GetImage(DbBrowseBox::NEW)); m_aFirstBtn.SetHelpId(HID_GRID_TRAVEL_FIRST); m_aPrevBtn.SetHelpId(HID_GRID_TRAVEL_PREV); m_aNextBtn.SetHelpId(HID_GRID_TRAVEL_NEXT); m_aLastBtn.SetHelpId(HID_GRID_TRAVEL_LAST); m_aNewBtn.SetHelpId(HID_GRID_TRAVEL_NEW); m_aAbsolute.SetHelpId(HID_GRID_TRAVEL_ABSOLUTE); m_aRecordCount.SetHelpId(HID_GRID_NUMBEROFRECORDS); // Handler fuer Buttons einrichten m_aFirstBtn.SetClickHdl(LINK(this,NavigationBar,OnClick)); m_aPrevBtn.SetClickHdl(LINK(this,NavigationBar,OnClick)); m_aNextBtn.SetClickHdl(LINK(this,NavigationBar,OnClick)); m_aLastBtn.SetClickHdl(LINK(this,NavigationBar,OnClick)); m_aNewBtn.SetClickHdl(LINK(this,NavigationBar,OnClick)); m_aRecordText.SetText(XubString(SVX_RES(RID_STR_REC_TEXT))); m_aRecordOf.SetText(XubString(SVX_RES(RID_STR_REC_FROM_TEXT))); m_aRecordCount.SetText('?'); m_nDefaultWidth = ArrangeControls(); m_aFirstBtn.Disable(); m_aPrevBtn.Disable(); m_aNextBtn.Disable(); m_aLastBtn.Disable(); m_aNewBtn.Disable(); m_aRecordText.Disable(); m_aRecordOf.Disable(); m_aRecordCount.Disable(); m_aAbsolute.Disable(); AllSettings aSettings = m_aNextBtn.GetSettings(); MouseSettings aMouseSettings = aSettings.GetMouseSettings(); aMouseSettings.SetButtonRepeat(aMouseSettings.GetButtonRepeat() / 4); aSettings.SetMouseSettings(aMouseSettings); m_aNextBtn.SetSettings(aSettings, sal_True); m_aPrevBtn.SetSettings(aSettings, sal_True); m_aFirstBtn.Show(); m_aPrevBtn.Show(); m_aNextBtn.Show(); m_aLastBtn.Show(); m_aNewBtn.Show(); m_aRecordText.Show(); m_aRecordOf.Show(); m_aRecordCount.Show(); m_aAbsolute.Show(); } //------------------------------------------------------------------------------ sal_uInt16 DbGridControl::NavigationBar::ArrangeControls() { // Positionierung der Controls // Basisgroessen ermitteln sal_uInt16 nX = 0; sal_uInt16 nY = 0; Rectangle aRect(((DbGridControl*)GetParent())->GetControlArea()); const long nH = aRect.GetSize().Height() - 1; Size aBorder = LogicToPixel(Size(3, 3),MAP_APPFONT); aBorder = Size(CalcZoom(aBorder.Width()), CalcZoom(aBorder.Height())); // Controls Groessen und Positionen setzen // XubString aText = m_aRecordText.GetText(); long nTextWidth = m_aRecordText.GetTextWidth(aText); m_aRecordText.SetPosPixel(Point(nX,nY+1) ); m_aRecordText.SetSizePixel(Size(nTextWidth,nH)); nX += (sal_uInt16)(nTextWidth + aBorder.Width()); m_aAbsolute.SetPosPixel( Point(nX,nY)); m_aAbsolute.SetSizePixel( Size(3*nH,aRect.GetSize().Height()) ); // Heuristik XXXXXXX nX += (sal_uInt16)((3*nH) + aBorder.Width()); aText = m_aRecordOf.GetText(); nTextWidth = m_aRecordOf.GetTextWidth(aText); m_aRecordOf.SetPosPixel(Point(nX,nY+1) ); m_aRecordOf.SetSizePixel(Size(nTextWidth,nH)); nX += (sal_uInt16)(nTextWidth + aBorder.Width()); nTextWidth = m_aRecordCount.GetTextWidth( String::CreateFromAscii("0000000 (00000)") ); m_aRecordCount.SetPosPixel(Point(nX,nY+1) ); m_aRecordCount.SetSizePixel(Size(nTextWidth,nH)); nX += (sal_uInt16)(nTextWidth + aBorder.Width()); m_aFirstBtn.SetPosPixel( Point(nX,nY) ); m_aFirstBtn.SetSizePixel( Size(nH,nH) ); nX += (sal_uInt16)nH; m_aPrevBtn.SetPosPixel( Point(nX,nY) ); m_aPrevBtn.SetSizePixel( Size(nH,nH) ); nX += (sal_uInt16)nH; m_aNextBtn.SetPosPixel( Point(nX,nY) ); m_aNextBtn.SetSizePixel( Size(nH,nH) ); nX += (sal_uInt16)nH; m_aLastBtn.SetPosPixel( Point(nX,nY) ); m_aLastBtn.SetSizePixel( Size(nH,nH) ); nX += (sal_uInt16)nH; m_aNewBtn.SetPosPixel( Point(nX,nY) ); m_aNewBtn.SetSizePixel( Size(nH,nH) ); nX += (sal_uInt16)(nH + aBorder.Width()); // Ist der Font des Edits groesser als das Feld? Font aOutputFont = m_aAbsolute.GetFont(); if (aOutputFont.GetSize().Height() > nH) { Font aApplFont = System::GetStandardFont(STDFONT_SWISS); aApplFont.SetSize(Size(0, nH-2)); m_aAbsolute.SetFont(aApplFont); aApplFont.SetTransparent( sal_True ); m_aRecordText.SetFont( aApplFont ); m_aRecordOf.SetFont( aApplFont ); m_aRecordCount.SetFont( aApplFont ); } return nX; } //------------------------------------------------------------------------------ IMPL_LINK(DbGridControl::NavigationBar, OnClick, Button *, pButton ) { DbGridControl* pParent = (DbGridControl*)GetParent(); if (pParent->m_aMasterSlotExecutor.IsSet()) { long lResult = 0; if (pButton == &m_aFirstBtn) lResult = pParent->m_aMasterSlotExecutor.Call((void*)RECORD_FIRST); else if( pButton == &m_aPrevBtn ) lResult = pParent->m_aMasterSlotExecutor.Call((void*)RECORD_PREV); else if( pButton == &m_aNextBtn ) lResult = pParent->m_aMasterSlotExecutor.Call((void*)RECORD_NEXT); else if( pButton == &m_aLastBtn ) lResult = pParent->m_aMasterSlotExecutor.Call((void*)RECORD_LAST); else if( pButton == &m_aNewBtn ) lResult = pParent->m_aMasterSlotExecutor.Call((void*)RECORD_NEW); if (lResult) // the link already handled it return 0; } if (pButton == &m_aFirstBtn) pParent->MoveToFirst(); else if( pButton == &m_aPrevBtn ) pParent->MoveToPrev(); else if( pButton == &m_aNextBtn ) pParent->MoveToNext(); else if( pButton == &m_aLastBtn ) pParent->MoveToLast(); else if( pButton == &m_aNewBtn ) pParent->AppendNew(); return 0; } //------------------------------------------------------------------------------ void DbGridControl::NavigationBar::InvalidateAll(sal_uInt32 nCurrentPos, sal_Bool bAll) { if (m_nCurrentPos != nCurrentPos || nCurrentPos < 0 || bAll) { DbGridControl* pParent = (DbGridControl*)GetParent(); // Wann muß alles invalidiert werden if (bAll || m_nCurrentPos <= 0 || nCurrentPos <= 0 || m_nCurrentPos >= (pParent->GetRowCount() - ((pParent->GetOptions() & DbGridControl::OPT_INSERT) ? 2 : 1)) || nCurrentPos >= (pParent->GetRowCount() - ((pParent->GetOptions() & DbGridControl::OPT_INSERT) ? 2 : 1))) { m_nCurrentPos = nCurrentPos; int i = 0; while (ControlMap[i]) SetState(ControlMap[i++]); } else // befindet sich in der Mitte { m_nCurrentPos = nCurrentPos; SetState(NavigationBar::RECORD_COUNT); SetState(NavigationBar::RECORD_ABSOLUTE); } } } //------------------------------------------------------------------------------ sal_Bool DbGridControl::NavigationBar::GetState(sal_uInt16 nWhich) const { DbGridControl* pParent = (DbGridControl*)GetParent(); if (!pParent->IsOpen() || pParent->IsDesignMode() || !pParent->IsEnabled() || pParent->IsFilterMode() ) return sal_False; else { // check if we have a master state provider if (pParent->m_aMasterStateProvider.IsSet()) { long nState = pParent->m_aMasterStateProvider.Call((void*)nWhich); if (nState>=0) return (nState>0); } sal_Bool bAvailable = sal_True; switch (nWhich) { case NavigationBar::RECORD_FIRST: case NavigationBar::RECORD_PREV: bAvailable = m_nCurrentPos > 0; break; case NavigationBar::RECORD_NEXT: if(pParent->m_bRecordCountFinal) { bAvailable = m_nCurrentPos < (pParent->GetRowCount() - 1); if (!bAvailable && pParent->GetOptions() & DbGridControl::OPT_INSERT) bAvailable = (m_nCurrentPos == (pParent->GetRowCount() - 2)) && pParent->IsModified(); } break; case NavigationBar::RECORD_LAST: if(pParent->m_bRecordCountFinal) { if (pParent->GetOptions() & DbGridControl::OPT_INSERT) bAvailable = pParent->IsCurrentAppending() ? pParent->GetRowCount() > 1 : m_nCurrentPos != (pParent->GetRowCount() - 2); else bAvailable = m_nCurrentPos != (pParent->GetRowCount() - 1); } break; case NavigationBar::RECORD_NEW: bAvailable = (pParent->GetOptions() & DbGridControl::OPT_INSERT) && pParent->GetRowCount() && m_nCurrentPos < (pParent->GetRowCount() - 1); break; case NavigationBar::RECORD_ABSOLUTE: bAvailable = pParent->GetRowCount() > 0; break; } return bAvailable; } } //------------------------------------------------------------------------------ void DbGridControl::NavigationBar::SetState(sal_uInt16 nWhich) { sal_Bool bAvailable = GetState(nWhich); DbGridControl* pParent = (DbGridControl*)GetParent(); Window* pWnd = NULL; switch (nWhich) { case NavigationBar::RECORD_FIRST: pWnd = &m_aFirstBtn; break; case NavigationBar::RECORD_PREV: pWnd = &m_aPrevBtn; break; case NavigationBar::RECORD_NEXT: pWnd = &m_aNextBtn; break; case NavigationBar::RECORD_LAST: pWnd = &m_aLastBtn; break; case NavigationBar::RECORD_NEW: pWnd = &m_aNewBtn; break; case NavigationBar::RECORD_ABSOLUTE: pWnd = &m_aAbsolute; if (bAvailable) { if (pParent->m_nTotalCount >= 0) { if (pParent->IsCurrentAppending()) m_aAbsolute.SetMax(pParent->m_nTotalCount + 1); else m_aAbsolute.SetMax(pParent->m_nTotalCount); } else m_aAbsolute.SetMax(LONG_MAX); m_aAbsolute.SetValue(m_nCurrentPos + 1); } else m_aAbsolute.SetText(String()); break; case NavigationBar::RECORD_TEXT: pWnd = &m_aRecordText; break; case NavigationBar::RECORD_OF: pWnd = &m_aRecordOf; break; case NavigationBar::RECORD_COUNT: { pWnd = &m_aRecordCount; String aText; if (bAvailable) { if (pParent->GetOptions() & DbGridControl::OPT_INSERT) { if (pParent->IsCurrentAppending() && !pParent->IsModified()) aText = String::CreateFromInt32(pParent->GetRowCount()); else aText = String::CreateFromInt32(pParent->GetRowCount() - 1); } else aText = String::CreateFromInt32(pParent->GetRowCount()); } else aText = String(); // add the number of selected rows, if applicable if (pParent->GetSelectRowCount()) { String aExtendedInfo(aText); aExtendedInfo.AppendAscii(" ("); aExtendedInfo += String::CreateFromInt32(pParent->GetSelectRowCount()); aExtendedInfo += ')'; pWnd->SetText(aExtendedInfo); } else pWnd->SetText(aText); { vos::OGuard aPaintSafety(Application::GetSolarMutex()); // we want to update only the window, not our parent, so lock the latter // (In fact, if we are in DbGridControl::RecalcRows, perhaps as a result of an setDataSource or // a VisibleRowsChanged, the grid will be frozen and a SeekRow triggered implicitly by the update // of pWnd will fail.) // (the SetUpdateMode call goes to the data window : it's sufficient to prevent SeekRow's, but it // avoids the Invalidate which would be triggered by BrowseBox::SetUpdateMode (which lead to massive // flicker when scrolling)) // FS - 06.10.99 // don't use SetUpdateMode in those situations as all necessary paints get lost DG // so update only if necessary (DG) if (pParent->IsPaintEnabled()) { pWnd->Update(); pWnd->Flush(); } } pParent->SetRealRowCount(aText); } break; } DBG_ASSERT(pWnd, "kein Fenster"); if (pWnd && (pWnd->IsEnabled() != bAvailable)) // this "pWnd->IsEnabled() != bAvailable" is a little hack : Window::Enable always generates a user // event (ImplGenerateMouseMove) even if nothing happened. This may lead to some unwanted effects, so we // do this check. // For further explanation see Bug 69900. // FS - 18.11.99 pWnd->Enable(bAvailable); } //------------------------------------------------------------------------------ void DbGridControl::NavigationBar::Resize() { Control::Resize(); ArrangeControls(); } //------------------------------------------------------------------------------ void DbGridControl::NavigationBar::Paint(const Rectangle& rRect) { Control::Paint(rRect); Point aAbsolutePos = m_aAbsolute.GetPosPixel(); Size aAbsoluteSize = m_aAbsolute.GetSizePixel(); DrawLine(Point(aAbsolutePos.X() - 1, 0 ), Point(aAbsolutePos.X() - 1, aAbsolutePos.Y() + aAbsoluteSize.Height())); DrawLine(Point(aAbsolutePos.X() + aAbsoluteSize.Width() + 1, 0 ), Point(aAbsolutePos.X() + aAbsoluteSize.Width() + 1, aAbsolutePos.Y() + aAbsoluteSize.Height())); } //------------------------------------------------------------------------------ void DbGridControl::NavigationBar::StateChanged( StateChangedType nType ) { Control::StateChanged(nType); if (STATE_CHANGE_ZOOM == nType) { Fraction aZoom = GetZoom(); m_aRecordText.SetZoom(aZoom); m_aAbsolute.SetZoom(aZoom); m_aRecordOf.SetZoom(aZoom); m_aRecordCount.SetZoom(aZoom); m_aFirstBtn.SetZoom(aZoom); m_aPrevBtn.SetZoom(aZoom); m_aNextBtn.SetZoom(aZoom); m_aLastBtn.SetZoom(aZoom); m_aNewBtn.SetZoom(aZoom); // not all of these controls need to know the new zoom, but to be sure ... // rearrange the controls m_nDefaultWidth = ArrangeControls(); } } //------------------------------------------------------------------------------ DbGridRow::DbGridRow(CursorWrapper* pCur, sal_Bool bPaintCursor) :m_bIsNew(sal_False) { if (pCur && pCur->Is()) { ::com::sun::star::uno::Reference< ::com::sun::star::container::XIndexAccess > xColumns(pCur->getColumns(), ::com::sun::star::uno::UNO_QUERY); DataColumn* pColumn; for (sal_Int32 i = 0; i < xColumns->getCount(); ++i) { ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet > xColSet; ::cppu::extractInterface(xColSet, xColumns->getByIndex(i)); pColumn = new DataColumn(xColSet); m_aVariants.Insert(pColumn, LIST_APPEND); } if (pCur->rowDeleted()) m_eStatus = GRS_DELETED; else { if (bPaintCursor) m_eStatus = (pCur->isAfterLast() || pCur->isBeforeFirst()) ? GRS_INVALID : GRS_CLEAN; else { ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet > xSet((::com::sun::star::uno::Reference< ::com::sun::star::uno::XInterface >)*pCur,::com::sun::star::uno::UNO_QUERY); if (xSet.is()) { m_bIsNew = ::comphelper::getBOOL(xSet->getPropertyValue(FM_PROP_ISNEW)); if (!m_bIsNew && (pCur->isAfterLast() || pCur->isBeforeFirst())) m_eStatus = GRS_INVALID; else if (::comphelper::getBOOL(xSet->getPropertyValue(FM_PROP_ISMODIFIED))) m_eStatus = GRS_MODIFIED; else m_eStatus = GRS_CLEAN; } else m_eStatus = GRS_INVALID; } } if (!m_bIsNew && IsValid()) m_aBookmark = pCur->getBookmark(); else m_aBookmark = ::com::sun::star::uno::Any(); } else m_eStatus = GRS_INVALID; } //------------------------------------------------------------------------------ DbGridRow::~DbGridRow() { sal_uInt32 nCount = m_aVariants.Count(); for (sal_uInt32 i = 0; i < nCount; i++) delete m_aVariants.GetObject(i); } //------------------------------------------------------------------------------ void DbGridRow::SetState(CursorWrapper* pCur, sal_Bool bPaintCursor) { if (pCur && pCur->Is()) { if (pCur->rowDeleted()) { m_eStatus = GRS_DELETED; m_bIsNew = sal_False; } else { m_eStatus = GRS_CLEAN; if (!bPaintCursor) { ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet > xSet((::com::sun::star::uno::Reference< ::com::sun::star::uno::XInterface >)*pCur, ::com::sun::star::uno::UNO_QUERY); DBG_ASSERT(xSet.is(), "DbGridRow::SetState : invalid cursor !"); if (::comphelper::getBOOL(xSet->getPropertyValue(FM_PROP_ISMODIFIED))) m_eStatus = GRS_MODIFIED; m_bIsNew = ::comphelper::getBOOL(xSet->getPropertyValue(FM_PROP_ISNEW)); } else m_bIsNew = sal_False; } if (!m_bIsNew && IsValid()) m_aBookmark = pCur->getBookmark(); else m_aBookmark = ::com::sun::star::uno::Any(); } else { m_aBookmark = ::com::sun::star::uno::Any(); m_eStatus = GRS_INVALID; m_bIsNew = sal_False; } } DBG_NAME(DbGridControl); //------------------------------------------------------------------------------ DbGridControl::DbGridControl( ::com::sun::star::uno::Reference< ::com::sun::star::lang::XMultiServiceFactory > _rxFactory, Window* pParent, WinBits nBits) :DbBrowseBox(pParent, DBBF_NONE, nBits, BROWSER_KEEPSELECTION | BROWSER_COLUMNSELECTION | BROWSER_MULTISELECTION | BROWSER_TRACKING_TIPS | BROWSER_HLINESFULL | BROWSER_VLINESFULL | BROWSER_AUTO_VSCROLL) #pragma warning (disable : 4355) ,m_aBar(this) #pragma warning (default : 4355) ,m_bSynchDisplay(sal_True) ,m_bForceROController(sal_False) ,m_bHandle(sal_False) ,m_aNullDate(DBTypeConversion::getStandardDate()) ,m_nAsynAdjustEvent(0) ,m_pDataSourcePropMultiplexer(NULL) ,m_pDataSourcePropListener(NULL) ,m_pFieldListeners(NULL) ,m_pCursorDisposeListener(NULL) ,m_bWantDestruction(sal_False) ,m_bInAdjustDataSource(sal_False) ,m_bPendingAdjustRows(sal_False) ,m_xServiceFactory(_rxFactory) ,m_pSelectionListener(NULL) { DBG_CTOR(DbGridControl,NULL); Construct(); } //------------------------------------------------------------------------------ DbGridControl::DbGridControl( ::com::sun::star::uno::Reference< ::com::sun::star::lang::XMultiServiceFactory > _rxFactory, Window* pParent, const ResId& rId) :DbBrowseBox(pParent, rId, DBBF_NONE, BROWSER_KEEPSELECTION | BROWSER_COLUMNSELECTION | BROWSER_MULTISELECTION | BROWSER_TRACKING_TIPS | BROWSER_HLINESFULL | BROWSER_VLINESFULL | BROWSER_AUTO_VSCROLL) #pragma warning (disable : 4355) ,m_aBar(this) #pragma warning (default : 4355) ,m_bSynchDisplay(sal_True) ,m_bForceROController(sal_False) ,m_bHandle(sal_False) ,m_aNullDate(DBTypeConversion::getStandardDate()) ,m_pDataSourcePropMultiplexer(NULL) ,m_pDataSourcePropListener(NULL) ,m_pFieldListeners(NULL) ,m_bWantDestruction(sal_False) ,m_bInAdjustDataSource(sal_False) ,m_bPendingAdjustRows(sal_False) ,m_xServiceFactory(_rxFactory) ,m_pSelectionListener(NULL) { DBG_CTOR(DbGridControl,NULL); Construct(); } //------------------------------------------------------------------------------ void DbGridControl::Construct() { m_nTotalCount = m_nSeekPos = m_nCurrentPos = -1; m_bDesignMode = m_bUpdating = m_bRecordCountFinal = sal_False; m_bFilterMode = sal_False; m_nMode = BROWSER_KEEPSELECTION | BROWSER_COLUMNSELECTION | BROWSER_MULTISELECTION | BROWSER_TRACKING_TIPS | BROWSER_HLINESFULL | BROWSER_VLINESFULL | BROWSER_AUTO_VSCROLL | BROWSER_HEADERBAR_NEW; m_bNavigationBar = m_bMultiSelection = sal_True; m_nOptions = OPT_READONLY; m_nOptionMask = OPT_INSERT | OPT_UPDATE | OPT_DELETE; m_nDeleteEvent = 0; m_pSeekCursor = m_pDataCursor = NULL; m_bHandle = sal_True; m_aBar.Show(); EnableDrop(sal_True); ImplInitSettings(sal_True,sal_True,sal_True); } //------------------------------------------------------------------------------ void DbGridControl::InsertHandleColumn() { // Handle Column einfuegen // Da die BrowseBox ohne handleColums Paintprobleme hat // wird diese versteckt if (HasHandle()) BrowseBox::InsertHandleColumn(GetDefaultColumnWidth(String()), sal_True); else BrowseBox::InsertHandleColumn(0, sal_False); } //------------------------------------------------------------------------------ void DbGridControl::Init() { BrowserHeader* pHeader = CreateHeaderBar(this); pHeader->SetMouseTransparent(sal_False); SetHeaderBar(pHeader); SetMode(m_nMode); SetCursorColor(Color(0xFF, 0, 0)); InsertHandleColumn(); } //------------------------------------------------------------------------------ DbGridControl::~DbGridControl() { RemoveColumns(); { m_bWantDestruction = sal_True; osl::MutexGuard aGuard(m_aDestructionSafety); if (m_pFieldListeners) DisconnectFromFields(); if (m_pCursorDisposeListener) { delete m_pCursorDisposeListener; m_pCursorDisposeListener = NULL; } } if (m_nDeleteEvent) Application::RemoveUserEvent(m_nDeleteEvent); if (m_pDataSourcePropMultiplexer) { m_pDataSourcePropMultiplexer->dispose(); m_pDataSourcePropMultiplexer->release(); // this should delete the multiplexer delete m_pDataSourcePropListener; m_pDataSourcePropMultiplexer = NULL; m_pDataSourcePropListener = NULL; } delete m_pDataCursor; delete m_pSeekCursor; DBG_DTOR(DbGridControl,NULL); } //------------------------------------------------------------------------------ void DbGridControl::StateChanged( StateChangedType nType ) { DbBrowseBox::StateChanged( nType ); switch (nType) { case STATE_CHANGE_ZOOM: { ImplInitSettings( sal_True, sal_False, sal_False ); // forward the zoom factor to the navigation bar if (m_bNavigationBar) m_aBar.SetZoom(GetZoom()); // and give it a chance to rearrange Point aPoint = GetControlArea().TopLeft(); sal_uInt16 nX = (sal_uInt16)aPoint.X(); ArrangeControls(nX, (sal_uInt16)aPoint.Y()); ReserveControlArea((sal_uInt16)nX); } break; case STATE_CHANGE_CONTROLFONT: ImplInitSettings( sal_True, sal_False, sal_False ); Invalidate(); break; case STATE_CHANGE_CONTROLFOREGROUND: ImplInitSettings( sal_False, sal_True, sal_False ); Invalidate(); break; case STATE_CHANGE_CONTROLBACKGROUND: ImplInitSettings( sal_False, sal_False, sal_True ); Invalidate(); break; } } //------------------------------------------------------------------------------ void DbGridControl::DataChanged( const DataChangedEvent& rDCEvt ) { DbBrowseBox::DataChanged( rDCEvt ); if ( (rDCEvt.GetType() == DATACHANGED_SETTINGS ) && (rDCEvt.GetFlags() & SETTINGS_STYLE) ) { ImplInitSettings( sal_True, sal_True, sal_True ); Invalidate(); } } //------------------------------------------------------------------------------ void DbGridControl::Select() { DbBrowseBox::Select(); // as the selected rows may have changed, udate the according display in our navigation bar m_aBar.InvalidateState(NavigationBar::RECORD_COUNT); if (m_pSelectionListener) m_pSelectionListener->selectionChanged(); } //------------------------------------------------------------------------------ void DbGridControl::ImplInitSettings( sal_Bool bFont, sal_Bool bForeground, sal_Bool bBackground ) { for (sal_uInt32 i = 0; i < m_aColumns.Count(); i++) { DbGridColumn* pCol = m_aColumns.GetObject(i); if (pCol) pCol->ImplInitSettings(&GetDataWindow(), bFont, bForeground, bBackground); } if (bBackground) { if (IsControlBackground()) { GetDataWindow().SetBackground(GetControlBackground()); GetDataWindow().SetControlBackground(GetControlBackground()); GetDataWindow().SetFillColor(GetControlBackground()); } else { GetDataWindow().SetControlBackground(); GetDataWindow().SetFillColor(GetFillColor()); } } } //------------------------------------------------------------------------------ void DbGridControl::RemoveRows(sal_Bool bNewCursor) { // Hat sich der DatenCursor verandert ? if (!bNewCursor) { DELETEZ(m_pSeekCursor); m_xPaintRow = m_xDataRow = m_xEmptyRow = m_xCurrentRow = m_xSeekRow = NULL; m_nCurrentPos = m_nSeekPos = -1; m_nOptions = OPT_READONLY; RowRemoved(0, GetRowCount(), sal_False); m_nTotalCount = -1; } else { RemoveRows(); } } //------------------------------------------------------------------------------ void DbGridControl::RemoveRows() { // we're going to remove all columns and all row, so deactivate the current cell if (IsEditing()) DeactivateCell(); // alle Columns deinitialisieren // existieren Spalten, dann alle Controller freigeben for (sal_uInt32 i = 0; i < m_aColumns.Count(); i++) m_aColumns.GetObject(i)->Clear(); DELETEZ(m_pSeekCursor); DELETEZ(m_pDataCursor); m_xPaintRow = m_xDataRow = m_xEmptyRow = m_xCurrentRow = m_xSeekRow = NULL; m_nCurrentPos = m_nSeekPos = m_nTotalCount = -1; m_nOptions = OPT_READONLY; // Anzahl Saetze im Browser auf 0 zuruecksetzen DbBrowseBox::RemoveRows(); m_aBar.InvalidateAll(m_nCurrentPos, sal_True); } //------------------------------------------------------------------------------ void DbGridControl::ArrangeControls(sal_uInt16& nX, sal_uInt16 nY) { // Positionierung der Controls if (m_bNavigationBar) { nX = m_aBar.GetDefaultWidth(); Rectangle aRect(GetControlArea()); m_aBar.SetPosSizePixel(Point(0,nY + 1), Size(nX, aRect.GetSize().Height() - 1)); } } //------------------------------------------------------------------------------ void DbGridControl::EnableHandle(sal_Bool bEnable) { if (m_bHandle == bEnable) return; // HandleColumn wird nur ausgeblendet, // da es sonst etliche Probleme mit dem Zeichnen gibt RemoveColumn(0); m_bHandle = bEnable; InsertHandleColumn(); } //------------------------------------------------------------------------------ void DbGridControl::EnableNavigationBar(sal_Bool bEnable) { if (m_bNavigationBar == bEnable) return; if (bEnable) { m_aBar.Show(); m_aBar.Enable(); m_aBar.InvalidateAll(m_nCurrentPos, sal_True); m_nMode &= ~BROWSER_AUTO_HSCROLL; SetMode(m_nMode); // liefert die Groeße der Reserved ControlArea Point aPoint = GetControlArea().TopLeft(); sal_uInt16 nX = (sal_uInt16)aPoint.X(); m_bNavigationBar = bEnable; ArrangeControls(nX, (sal_uInt16)aPoint.Y()); ReserveControlArea((sal_uInt16)nX); } else { m_aBar.Hide(); m_aBar.Disable(); m_nMode |= BROWSER_AUTO_HSCROLL; SetMode(m_nMode); ReserveControlArea(); m_bNavigationBar = bEnable; } } //------------------------------------------------------------------------------ sal_uInt16 DbGridControl::SetOptions(sal_uInt16 nOpt) { DBG_ASSERT(!m_xCurrentRow || !m_xCurrentRow->IsModified(), "DbGridControl::SetOptions : please do not call when editing a record (things are much easier this way ;) !"); // for the next setDataSource (which is triggered by a refresh, for instance) m_nOptionMask = nOpt; // normalize the new options ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet > xDataSourceSet((::com::sun::star::uno::Reference< ::com::sun::star::uno::XInterface >)*m_pDataCursor, ::com::sun::star::uno::UNO_QUERY); if (xDataSourceSet.is()) { // feststellen welche Updatemöglichkeiten bestehen sal_Int32 nPrivileges; xDataSourceSet->getPropertyValue(FM_PROP_PRIVILEGES) >>= nPrivileges; if ((nPrivileges & ::com::sun::star::sdbcx::Privilege::INSERT) == 0) nOpt &= ~OPT_INSERT; if ((nPrivileges & ::com::sun::star::sdbcx::Privilege::UPDATE) == 0) nOpt &= ~OPT_UPDATE; if ((nPrivileges & ::com::sun::star::sdbcx::Privilege::DELETE) == 0) nOpt &= ~OPT_DELETE; } else nOpt = OPT_READONLY; // need to do something after that ? if (nOpt == m_nOptions) return m_nOptions; // the 'update' option only affects our BrowserMode (with or w/o focus rect) BrowserMode nNewMode = m_nMode; if ((m_nMode & BROWSER_CURSOR_WO_FOCUS) == 0) { if (nOpt & OPT_UPDATE) nNewMode |= BROWSER_HIDECURSOR; else nNewMode &= ~BROWSER_HIDECURSOR; } else nNewMode &= ~BROWSER_HIDECURSOR; // should not be neccessary if EnablePermanentCursor is used to change the cursor behaviour, but to be sure ... if (nNewMode != m_nMode) { SetMode(nNewMode); m_nMode = nNewMode; } // _after_ setting the mode because this results in an ActivateCell DeactivateCell(); sal_Bool bInsertChanged = (nOpt & OPT_INSERT) != (m_nOptions & OPT_INSERT); m_nOptions = nOpt; // we need to set this before the code below because it indirectly uses m_nOptions // the 'insert' option affects our empty row if (bInsertChanged) if (m_nOptions & OPT_INSERT) { // the insert option is to be set m_xEmptyRow = new DbGridRow(); RowInserted(GetRowCount(), 1, sal_True); } else { // the insert option is to be reset m_xEmptyRow = NULL; if ((GetCurRow() == GetRowCount() - 1) && (GetCurRow() > 0)) GoToRowColumnId(GetCurRow() - 1, GetCurColumnId()); RowRemoved(GetRowCount(), 1, sal_True); } // the 'delete' options has no immediate consequences ActivateCell(); Invalidate(); return m_nOptions; } //------------------------------------------------------------------------------ void DbGridControl::EnablePermanentCursor(sal_Bool bEnable) { if (IsPermanentCursorEnabled() == bEnable) return; if (bEnable) { m_nMode &= ~BROWSER_HIDECURSOR; // without this BROWSER_CURSOR_WO_FOCUS won't have any affect m_nMode |= BROWSER_CURSOR_WO_FOCUS; } else { if (m_nOptions & OPT_UPDATE) m_nMode |= BROWSER_HIDECURSOR; // no cursor at all else m_nMode &= ~BROWSER_HIDECURSOR; // at least the "non-permanent" cursor m_nMode &= ~BROWSER_CURSOR_WO_FOCUS; } SetMode(m_nMode); sal_Bool bWasEditing = IsEditing(); DeactivateCell(); if (bWasEditing) ActivateCell(); } //------------------------------------------------------------------------------ sal_Bool DbGridControl::IsPermanentCursorEnabled() const { return ((m_nMode & BROWSER_CURSOR_WO_FOCUS) != 0) && ((m_nMode & BROWSER_HIDECURSOR) == 0); } //------------------------------------------------------------------------------ void DbGridControl::refreshController(sal_uInt16 _nColId, GrantCellControlAccess _aAccess) { if ((GetCurColumnId() == _nColId) && IsEditing()) { // the controller which is currently active needs to be refreshed DeactivateCell(); ActivateCell(); } } //------------------------------------------------------------------------------ void DbGridControl::SetMultiSelection(sal_Bool bMulti) { m_bMultiSelection = bMulti; if (m_bMultiSelection) m_nMode |= BROWSER_MULTISELECTION; else m_nMode &= ~BROWSER_MULTISELECTION; SetMode(m_nMode); } //------------------------------------------------------------------------------ void DbGridControl::setDataSource(const ::com::sun::star::uno::Reference< ::com::sun::star::sdbc::XRowSet >& _xCursor, sal_uInt16 nOpts) { if (!_xCursor.is() && !m_pDataCursor) return; if (m_pDataSourcePropMultiplexer) { m_pDataSourcePropMultiplexer->dispose(); m_pDataSourcePropMultiplexer->release(); // this should delete the multiplexer delete m_pDataSourcePropListener; m_pDataSourcePropMultiplexer = NULL; m_pDataSourcePropListener = NULL; } // is the new cursor valid ? // the cursor is only valid if it contains some columns // if there is no cursor or the cursor is not valid we have to clean up an leave if (!_xCursor.is() || !::com::sun::star::uno::Reference< ::com::sun::star::sdbcx::XColumnsSupplier > (_xCursor, ::com::sun::star::uno::UNO_QUERY)->getColumns()->hasElements()) { RemoveRows(); return; } // Hat sich der DatenCursor verandert ? sal_uInt16 nCurPos = GetColumnPos(GetCurColumnId()); SetUpdateMode(sal_False); RemoveRows(); DisconnectFromFields(); DELETEZ(m_pCursorDisposeListener); { ::osl::MutexGuard aGuard(m_aAdjustSafety); if (m_nAsynAdjustEvent) { // the adjust was thought to work with the old cursor which we don't have anymore RemoveUserEvent(m_nAsynAdjustEvent); m_nAsynAdjustEvent = 0; } } // get a new formatter and data cursor m_xFormatter = NULL; ::com::sun::star::uno::Reference< ::com::sun::star::util::XNumberFormatsSupplier > xSupplier = ::dbtools::getNumberFormats(::dbtools::getConnection(_xCursor), sal_True); if (xSupplier.is() && m_xServiceFactory.is()) { m_xFormatter = ::com::sun::star::uno::Reference< ::com::sun::star::util::XNumberFormatter >( m_xServiceFactory->createInstance(FM_NUMBER_FORMATTER), ::com::sun::star::uno::UNO_QUERY); if (m_xFormatter.is()) { m_xFormatter->attachNumberFormatsSupplier(xSupplier); // retrieve the datebase of the Numberformatter try { xSupplier->getNumberFormatSettings()->getPropertyValue(rtl::OUString::createFromAscii("NullDate")) >>= m_aNullDate; } catch(::com::sun::star::uno::Exception&) { } } } m_pDataCursor = new CursorWrapper(_xCursor); // now create a cursor for painting rows // we need that cursor only if we are not in insert only mode ::com::sun::star::uno::Reference< ::com::sun::star::sdbc::XResultSet > xClone; ::com::sun::star::uno::Reference< ::com::sun::star::sdb::XResultSetAccess > xAccess(_xCursor, ::com::sun::star::uno::UNO_QUERY); try { xClone = xAccess.is() ? xAccess->createResultSet() : ::com::sun::star::uno::Reference< ::com::sun::star::sdbc::XResultSet > (); } catch(Exception&) { } if (xClone.is()) m_pSeekCursor = new CursorWrapper(xClone); // property listening on the data source // (Normally one class would be sufficient : the multiplexer which could forward the property change to us. // But for that we would have been derived from ::comphelper::OPropertyChangeListener, which isn't exported. // So we introduce a second class, which is a ::comphelper::OPropertyChangeListener (in the implementation file we know this class) // and forwards the property changes to a our special method "DataSourcePropertyChanged".) if (m_pDataCursor) { m_pDataSourcePropListener = new FmXGridSourcePropListener(this); m_pDataSourcePropMultiplexer = new ::comphelper::OPropertyChangeMultiplexer(m_pDataSourcePropListener, ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet > ((::com::sun::star::uno::Reference< ::com::sun::star::uno::XInterface >)*m_pDataCursor, ::com::sun::star::uno::UNO_QUERY)); m_pDataSourcePropMultiplexer->acquire(); m_pDataSourcePropMultiplexer->addProperty(FM_PROP_ISMODIFIED); m_pDataSourcePropMultiplexer->addProperty(FM_PROP_ISNEW); } BrowserMode nOldMode = m_nMode; if (m_pSeekCursor) { ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet > xSet(_xCursor, ::com::sun::star::uno::UNO_QUERY); if (xSet.is()) { // feststellen welche Updatemöglichkeiten bestehen sal_Int32 nPrivileges; xSet->getPropertyValue(FM_PROP_PRIVILEGES) >>= nPrivileges; // Insert Option should be set if insert only otherwise you won't see any rows // and no insertion is possible if ((m_nOptionMask & OPT_INSERT) && ((nPrivileges & ::com::sun::star::sdbcx::Privilege::INSERT) == ::com::sun::star::sdbcx::Privilege::INSERT) && (nOpts & OPT_INSERT)) m_nOptions |= OPT_INSERT; if ((m_nOptionMask & OPT_UPDATE) && ((nPrivileges & ::com::sun::star::sdbcx::Privilege::UPDATE) == ::com::sun::star::sdbcx::Privilege::UPDATE) && (nOpts & OPT_UPDATE)) m_nOptions |= OPT_UPDATE; if ((m_nOptionMask & OPT_DELETE) && ((nPrivileges & ::com::sun::star::sdbcx::Privilege::DELETE) == ::com::sun::star::sdbcx::Privilege::DELETE) && (nOpts & OPT_DELETE)) m_nOptions |= OPT_DELETE; } sal_Bool bPermanentCursor = IsPermanentCursorEnabled(); m_nMode = BROWSER_COLUMNSELECTION | BROWSER_KEEPSELECTION | BROWSER_TRACKING_TIPS | BROWSER_HLINESFULL | BROWSER_VLINESFULL | BROWSER_HEADERBAR_NEW | BROWSER_AUTO_VSCROLL; if (bPermanentCursor) { m_nMode |= BROWSER_CURSOR_WO_FOCUS; m_nMode &= ~BROWSER_HIDECURSOR; } else // Duerfen Updates gemacht werden, kein Focus-RechtEck if (m_nOptions & OPT_UPDATE) m_nMode |= BROWSER_HIDECURSOR; if (m_bMultiSelection) m_nMode |= BROWSER_MULTISELECTION; if (!m_bNavigationBar) { m_nMode |= BROWSER_AUTO_HSCROLL; m_nMode |= BROWSER_NO_HSCROLL; ReserveControlArea(); } ::com::sun::star::uno::Reference< ::com::sun::star::sdbcx::XColumnsSupplier > xSupplyColumns(_xCursor, ::com::sun::star::uno::UNO_QUERY); if (xSupplyColumns.is()) InitColumnsByFields(::com::sun::star::uno::Reference< ::com::sun::star::container::XIndexAccess > (xSupplyColumns->getColumns(), ::com::sun::star::uno::UNO_QUERY)); ConnectToFields(); } sal_uInt32 nRecordCount(0); if (m_pSeekCursor) { ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet > xSet((::com::sun::star::uno::Reference< ::com::sun::star::uno::XInterface >)*m_pDataCursor, ::com::sun::star::uno::UNO_QUERY); xSet->getPropertyValue(FM_PROP_ROWCOUNT) >>= nRecordCount; m_bRecordCountFinal = ::comphelper::getBOOL(xSet->getPropertyValue(FM_PROP_ROWCOUNTFINAL)); // insert the currently known rows // and one row if we are able to insert rows if (m_nOptions & OPT_INSERT) { // insert the empty row for insertion m_xEmptyRow = new DbGridRow(); nRecordCount++; } if (nRecordCount) { m_xPaintRow = m_xSeekRow = new DbGridRow(m_pSeekCursor, sal_True); m_xDataRow = new DbGridRow(m_pDataCursor, sal_False); RowInserted(0, nRecordCount, sal_False); if (m_xSeekRow->IsValid()) m_nSeekPos = m_pSeekCursor->getRow() - 1; } else { // no rows so we don't need a seekcursor DELETEZ(m_pSeekCursor); } } // Zur alten Spalte gehen if (!nCurPos || nCurPos >= ColCount()) nCurPos = 1; // there are rows so go to the selected current column if (nRecordCount) GoToRowColumnId(0, GetColumnId(nCurPos)); // else stop the editing if neccessary else if (IsEditing()) DeactivateCell(); // now reset the mode if (m_nMode != nOldMode) SetMode(m_nMode); // beim Resizen wird RecalcRows gerufen if (!IsResizing() && GetRowCount()) RecalcRows(GetTopRow(), GetVisibleRows(), sal_True); m_aBar.InvalidateAll(m_nCurrentPos, sal_True); SetUpdateMode(sal_True); // start listening on the seek cursor if (m_pSeekCursor) m_pCursorDisposeListener = new DisposeListenerGridBridge(*this, ::com::sun::star::uno::Reference< ::com::sun::star::lang::XComponent > ((::com::sun::star::uno::Reference< ::com::sun::star::uno::XInterface >)*m_pSeekCursor, ::com::sun::star::uno::UNO_QUERY), 0); } //------------------------------------------------------------------------------ void DbGridControl::RemoveColumns() { for (sal_uInt32 i = 0; i < m_aColumns.Count(); i++) delete m_aColumns.GetObject(i); m_aColumns.Clear(); DbBrowseBox::RemoveColumns(); } //------------------------------------------------------------------------------ DbGridColumn* DbGridControl::CreateColumn(sal_uInt16 nId, const DbGridRowRef& _xRow) const { return new DbGridColumn(nId, *(DbGridControl*)this); } //------------------------------------------------------------------------------ sal_uInt16 DbGridControl::AppendColumn(const XubString& rName, sal_uInt16 nWidth, sal_uInt16 nModelPos, sal_uInt16 nId) { DBG_ASSERT(nId == (sal_uInt16)-1, "DbGridControl::AppendColumn : I want to set the ID myself ..."); sal_uInt16 nRealPos = nModelPos; if (nModelPos != HEADERBAR_APPEND) { // calc the view pos. we can't use our converting functions because the new column // has no VCL-representation, yet. sal_Int16 nViewPos = nModelPos; while (nModelPos--) { if (m_aColumns.GetObject(nModelPos)->IsHidden()) --nViewPos; } // restore nModelPos, we need it later nModelPos = nRealPos; // the position the base class gets is the view pos + 1 (because of the handle column) nRealPos = nViewPos + 1; } // calculate the new id for (nId=1; (GetModelColumnPos(nId) != (sal_uInt16)-1) && (nId<=m_aColumns.Count()); ++nId) ; DBG_ASSERT(GetViewColumnPos(nId) == (sal_uInt16)-1, "DbGridControl::AppendColumn : inconsistent internal state !"); // my column's models say "there is no column with id nId", but the view (the base class) says "there is a column ..." DbBrowseBox::AppendColumn(rName, nWidth, nRealPos, nId); if (nModelPos == HEADERBAR_APPEND) m_aColumns.Insert(CreateColumn(nId, m_xCurrentRow), LIST_APPEND); else m_aColumns.Insert(CreateColumn(nId, m_xCurrentRow), nModelPos); return nId; } //------------------------------------------------------------------------------ void DbGridControl::RemoveColumn(sal_uInt16 nId) { sal_uInt16 nIndex = GetModelColumnPos(nId); DbBrowseBox::RemoveColumn(nId); delete m_aColumns.Remove(nIndex); } //------------------------------------------------------------------------------ void DbGridControl::ColumnMoved(sal_uInt16 nId) { DbBrowseBox::ColumnMoved(nId); // remove the col from the model sal_uInt16 nOldModelPos = GetModelColumnPos(nId); #ifdef DBG_UTIL DbGridColumn* pCol = m_aColumns.GetObject((sal_uInt32)nOldModelPos); DBG_ASSERT(!pCol->IsHidden(), "DbGridControl::ColumnMoved : moved a hidden col ? how this ?"); #endif // for the new model pos we can't use GetModelColumnPos because we are altering the model at the moment // so the method won't work (in fact it would return the old model pos) // the new view pos is calculated easily sal_uInt16 nNewViewPos = GetViewColumnPos(nId); // from that we can compute the new model pos sal_uInt16 nNewModelPos; for (nNewModelPos = 0; nNewModelPos < m_aColumns.Count(); ++nNewModelPos) { if (!m_aColumns.GetObject(nNewModelPos)->IsHidden()) if (!nNewViewPos) break; else --nNewViewPos; } DBG_ASSERT(nNewModelPos