/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include "directsql.hxx" #include "directsql.hrc" #include "dbu_dlg.hrc" #include #include #include #include #include #include #include //........................................................................ namespace dbaui { //........................................................................ using namespace ::com::sun::star::uno; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::lang; using ::rtl::OStringBuffer; //==================================================================== //= LargeEntryListBox //==================================================================== class LargeEntryListBox : public ListBox { public: LargeEntryListBox( Window* _pParent, const ResId& _rId ); protected: virtual void UserDraw( const UserDrawEvent& rUDEvt ); }; //-------------------------------------------------------------------- LargeEntryListBox::LargeEntryListBox( Window* _pParent, const ResId& _rId ) :ListBox(_pParent, _rId ) { EnableUserDraw(sal_True); } //-------------------------------------------------------------------- void LargeEntryListBox::UserDraw( const UserDrawEvent& _rUDEvt ) { if (LISTBOX_ENTRY_NOTFOUND == _rUDEvt.GetItemId()) ListBox::UserDraw( _rUDEvt ); else _rUDEvt.GetDevice()->DrawText( _rUDEvt.GetRect(), GetEntry( _rUDEvt.GetItemId() ), TEXT_DRAW_LEFT | TEXT_DRAW_VCENTER | TEXT_DRAW_ENDELLIPSIS); } //==================================================================== //= DirectSQLDialog //==================================================================== DBG_NAME(DirectSQLDialog) //-------------------------------------------------------------------- DirectSQLDialog::DirectSQLDialog( Window* _pParent, const Reference< XConnection >& _rxConn ) :ModalDialog(_pParent, ModuleRes(DLG_DIRECTSQL)) ,m_aFrame (this, ModuleRes(FL_SQL)) ,m_aSQLLabel (this, ModuleRes(FT_SQL)) ,m_aSQL (this, ModuleRes(ME_SQL)) ,m_aExecute (this, ModuleRes(PB_EXECUTE)) ,m_aHistoryLabel (this, ModuleRes(FT_HISTORY)) ,m_pSQLHistory(new LargeEntryListBox(this, ModuleRes(LB_HISTORY))) ,m_aStatusFrame (this, ModuleRes(FL_STATUS)) ,m_aStatus (this, ModuleRes(ME_STATUS)) ,m_pShowOutput(new CheckBox(this, ModuleRes(CB_SHOWOUTPUT))) ,m_aOutputFrame (this, ModuleRes(FL_OUTPUT)) ,m_aOutput (this, ModuleRes(ME_OUTPUT)) ,m_aButtonSeparator (this, ModuleRes(FL_BUTTONS)) ,m_aHelp (this, ModuleRes(PB_HELP)) ,m_aClose (this, ModuleRes(PB_CLOSE)) ,m_nHistoryLimit(20) ,m_nStatusCount(1) ,m_xConnection(_rxConn) { DBG_CTOR(DirectSQLDialog,NULL); FreeResource(); m_aSQL.GrabFocus(); m_aExecute.SetClickHdl(LINK(this, DirectSQLDialog, OnExecute)); m_aClose.SetClickHdl(LINK(this, DirectSQLDialog, OnClose)); m_pSQLHistory->SetSelectHdl(LINK(this, DirectSQLDialog, OnListEntrySelected)); m_pSQLHistory->SetDropDownLineCount(10); // add a dispose listener to the connection Reference< XComponent > xConnComp(m_xConnection, UNO_QUERY); OSL_ENSURE(xConnComp.is(), "DirectSQLDialog::DirectSQLDialog: invalid connection!"); if (xConnComp.is()) startComponentListening(xConnComp); m_aSQL.SetModifyHdl(LINK(this, DirectSQLDialog, OnStatementModified)); OnStatementModified(&m_aSQL); } //-------------------------------------------------------------------- DirectSQLDialog::~DirectSQLDialog() { { ::osl::MutexGuard aGuard(m_aMutex); stopAllComponentListening(); } delete m_pSQLHistory; DBG_DTOR(DirectSQLDialog,NULL); } //-------------------------------------------------------------------- void DirectSQLDialog::_disposing( const EventObject& _rSource ) { SolarMutexGuard aSolarGuard; ::osl::MutexGuard aGuard(m_aMutex); OSL_ENSURE(Reference< XConnection >(_rSource.Source, UNO_QUERY).get() == m_xConnection.get(), "DirectSQLDialog::_disposing: where does this come from?"); (void)_rSource; { String sMessage(ModuleRes(STR_DIRECTSQL_CONNECTIONLOST)); ErrorBox aError(this, WB_OK, sMessage); aError.Execute(); } PostUserEvent(LINK(this, DirectSQLDialog, OnClose)); } //-------------------------------------------------------------------- sal_Int32 DirectSQLDialog::getHistorySize() const { CHECK_INVARIANTS("DirectSQLDialog::getHistorySize"); return m_aStatementHistory.size(); } //-------------------------------------------------------------------- void DirectSQLDialog::implEnsureHistoryLimit() { CHECK_INVARIANTS("DirectSQLDialog::implEnsureHistoryLimit"); if (getHistorySize() <= m_nHistoryLimit) // nothing to do return; sal_Int32 nRemoveEntries = getHistorySize() - m_nHistoryLimit; while (nRemoveEntries--) { m_aStatementHistory.pop_front(); m_aNormalizedHistory.pop_front(); m_pSQLHistory->RemoveEntry((sal_uInt16)0); } } //-------------------------------------------------------------------- void DirectSQLDialog::implAddToStatementHistory(const String& _rStatement) { CHECK_INVARIANTS("DirectSQLDialog::implAddToStatementHistory"); // add the statement to the history m_aStatementHistory.push_back(_rStatement); // normalize the statement, and remember the normalized form, too String sNormalized(_rStatement); sNormalized.SearchAndReplaceAll((sal_Unicode)'\n', ' '); m_aNormalizedHistory.push_back(sNormalized); // add the normalized version to the list box m_pSQLHistory->InsertEntry(sNormalized); // ensure that we don't exceed the history limit implEnsureHistoryLimit(); } #ifdef DBG_UTIL //-------------------------------------------------------------------- const sal_Char* DirectSQLDialog::impl_CheckInvariants() const { if (m_aStatementHistory.size() != m_aNormalizedHistory.size()) return "statement history is inconsistent!"; if (!m_pSQLHistory) return "invalid listbox!"; if (m_aStatementHistory.size() != m_pSQLHistory->GetEntryCount()) return "invalid listbox entry count!"; if (!m_xConnection.is()) return "have no connection!"; return NULL; } #endif //-------------------------------------------------------------------- void DirectSQLDialog::implExecuteStatement(const String& _rStatement) { CHECK_INVARIANTS("DirectSQLDialog::implExecuteStatement"); ::osl::MutexGuard aGuard(m_aMutex); String sStatus; ::com::sun::star::uno::Reference< ::com::sun::star::sdbc::XResultSet > xResultSet; try { // create a statement Reference< XStatement > xStatement = m_xConnection->createStatement(); OSL_ENSURE(xStatement.is(), "DirectSQLDialog::implExecuteStatement: no statement returned by the connection!"); // clear the output box m_aOutput.SetText(rtl::OUString()); if (xStatement.is()) { if (OUString(_rStatement).toAsciiUpperCase().startsWith("SELECT") && m_pShowOutput->IsChecked()) { // execute it as a query xResultSet = xStatement->executeQuery(_rStatement); // get a handle for the rows ::com::sun::star::uno::Reference< ::com::sun::star::sdbc::XRow > xRow( xResultSet, ::com::sun::star::uno::UNO_QUERY ); // work through each of the rows while (xResultSet->next()) { // initialise the output line for each row String out = ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("")); // work along the columns until that are none left int i = 1; try { for (;;) { // be dumb, treat everything as a string out += xRow->getString(i) + ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(",")); i++; } } // trap for when we fall off the end of the row catch (const SQLException&) { } // report the output addOutputText(::rtl::OUString(out)); } } else { // execute it xStatement->execute(_rStatement); } } // successful sStatus = String(ModuleRes(STR_COMMAND_EXECUTED_SUCCESSFULLY)); // dispose the statement ::comphelper::disposeComponent(xStatement); } catch(const SQLException& e) { sStatus = e.Message; } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } // add the status text addStatusText(sStatus); } //-------------------------------------------------------------------- void DirectSQLDialog::addStatusText(const String& _rMessage) { OUString sAppendMessage = OUString::number(m_nStatusCount++) + ": " + _rMessage + "\n\n"; OUString sCompleteMessage = m_aStatus.GetText() + sAppendMessage; m_aStatus.SetText(sCompleteMessage); m_aStatus.SetSelection(Selection(sCompleteMessage.getLength(), sCompleteMessage.getLength())); } //-------------------------------------------------------------------- void DirectSQLDialog::addOutputText(const String& _rMessage) { String sAppendMessage = _rMessage; sAppendMessage += rtl::OUString("\n"); String sCompleteMessage = m_aOutput.GetText(); sCompleteMessage += sAppendMessage; m_aOutput.SetText(sCompleteMessage); } //-------------------------------------------------------------------- void DirectSQLDialog::executeCurrent() { CHECK_INVARIANTS("DirectSQLDialog::executeCurrent"); String sStatement = m_aSQL.GetText(); // execute implExecuteStatement(sStatement); // add the statement to the history implAddToStatementHistory(sStatement); m_aSQL.SetSelection(Selection()); m_aSQL.GrabFocus(); } //-------------------------------------------------------------------- void DirectSQLDialog::switchToHistory(sal_Int32 _nHistoryPos, sal_Bool _bUpdateListBox) { CHECK_INVARIANTS("DirectSQLDialog::switchToHistory"); if ((_nHistoryPos >= 0) && (_nHistoryPos < getHistorySize())) { // set the text in the statement editor String sStatement = m_aStatementHistory[_nHistoryPos]; m_aSQL.SetText(sStatement); OnStatementModified(&m_aSQL); if (_bUpdateListBox) { // selecte the normalized statement in the list box m_pSQLHistory->SelectEntryPos((sal_uInt16)_nHistoryPos); OSL_ENSURE(m_pSQLHistory->GetSelectEntry() == m_aNormalizedHistory[_nHistoryPos], "DirectSQLDialog::switchToHistory: inconsistent listbox entries!"); } m_aSQL.GrabFocus(); m_aSQL.SetSelection(Selection(sStatement.Len(), sStatement.Len())); } else OSL_FAIL("DirectSQLDialog::switchToHistory: invalid position!"); } //-------------------------------------------------------------------- IMPL_LINK( DirectSQLDialog, OnStatementModified, void*, /*NOTINTERESTEDIN*/ ) { m_aExecute.Enable(!m_aSQL.GetText().isEmpty()); return 0L; } //-------------------------------------------------------------------- IMPL_LINK( DirectSQLDialog, OnClose, void*, /*NOTINTERESTEDIN*/ ) { EndDialog( RET_OK ); return 0L; } //-------------------------------------------------------------------- IMPL_LINK( DirectSQLDialog, OnExecute, void*, /*NOTINTERESTEDIN*/ ) { executeCurrent(); return 0L; } //-------------------------------------------------------------------- IMPL_LINK( DirectSQLDialog, OnListEntrySelected, void*, /*NOTINTERESTEDIN*/ ) { if (!m_pSQLHistory->IsTravelSelect()) { const sal_uInt16 nSelected = m_pSQLHistory->GetSelectEntryPos(); if (LISTBOX_ENTRY_NOTFOUND != nSelected) switchToHistory(nSelected, sal_False); } return 0L; } //........................................................................ } // namespace dbaui //........................................................................ /* vim:set shiftwidth=4 softtabstop=4 expandtab: */