You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
496 lines
14 KiB
496 lines
14 KiB
/* This file is part of the KDE libraries
|
|
Copyright (C) 2002 by John Firebaugh <jfirebaugh@kde.org>
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Library General Public
|
|
License version 2 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
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public License
|
|
along with this library; see the file COPYING.LIB. If not, write to
|
|
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include <tqlabel.h>
|
|
#include <tqregexp.h>
|
|
#include <tqstyle.h>
|
|
#include <tqpopupmenu.h>
|
|
#include <kgenericfactory.h>
|
|
#include <klocale.h>
|
|
#include <kaction.h>
|
|
#include <kcombobox.h>
|
|
#include <kconfig.h>
|
|
#include <kdebug.h>
|
|
|
|
#include "ISearchPlugin.h"
|
|
#include "ISearchPlugin.moc"
|
|
|
|
K_EXPORT_COMPONENT_FACTORY( ktexteditor_isearch, KGenericFactory<ISearchPlugin>( "ktexteditor_isearch" ) )
|
|
|
|
ISearchPluginView::ISearchPluginView( KTextEditor::View *view )
|
|
: TQObject ( view ), KXMLGUIClient (view)
|
|
, m_view( 0L )
|
|
, m_doc( 0L )
|
|
, m_searchIF( 0L )
|
|
, m_cursorIF( 0L )
|
|
, m_selectIF( 0L )
|
|
// , m_toolBarAction( 0L )
|
|
, m_searchForwardAction( 0L )
|
|
, m_searchBackwardAction( 0L )
|
|
, m_label( 0L )
|
|
, m_combo( 0L )
|
|
, m_lastString( "" )
|
|
, m_searchBackward( false )
|
|
, m_caseSensitive( false )
|
|
, m_fromBeginning( false )
|
|
, m_regExp( false )
|
|
, m_autoWrap( false )
|
|
, m_wrapped( false )
|
|
, m_startLine( 0 )
|
|
, m_startCol( 0 )
|
|
, m_searchLine( 0 )
|
|
, m_searchCol( 0 )
|
|
, m_foundLine( 0 )
|
|
, m_foundCol( 0 )
|
|
, m_matchLen( 0 )
|
|
, m_toolBarWasHidden( false )
|
|
{
|
|
view->insertChildClient (this);
|
|
|
|
setInstance( KGenericFactory<ISearchPlugin>::instance() );
|
|
|
|
m_searchForwardAction = new KAction(
|
|
i18n("Search Incrementally"), CTRL+ALT+Key_F,
|
|
this, TQT_SLOT(slotSearchForwardAction()),
|
|
actionCollection(), "edit_isearch" );
|
|
m_searchBackwardAction = new KAction(
|
|
i18n("Search Incrementally Backwards"), CTRL+ALT+SHIFT+Key_F,
|
|
this, TQT_SLOT(slotSearchBackwardAction()),
|
|
actionCollection(), "edit_isearch_reverse" );
|
|
|
|
m_label = new TQLabel( i18n("I-Search:"), 0L, "kde toolbar widget" );
|
|
KWidgetAction* labelAction = new KWidgetAction(
|
|
m_label,
|
|
i18n("I-Search:"), 0, 0, 0,
|
|
actionCollection(), "isearch_label" );
|
|
labelAction->setShortcutConfigurable( false );
|
|
|
|
m_combo = new KHistoryCombo();
|
|
m_combo->setDuplicatesEnabled( false );
|
|
m_combo->setMaximumWidth( 300 );
|
|
m_combo->lineEdit()->installEventFilter( this );
|
|
connect( m_combo, TQT_SIGNAL(textChanged(const TQString&)),
|
|
this, TQT_SLOT(slotTextChanged(const TQString&)) );
|
|
connect( m_combo, TQT_SIGNAL(returnPressed(const TQString&)),
|
|
this, TQT_SLOT(slotReturnPressed(const TQString&)) );
|
|
connect( m_combo, TQT_SIGNAL(aboutToShowContextMenu(TQPopupMenu*)),
|
|
this, TQT_SLOT(slotAddContextMenuItems(TQPopupMenu*)) );
|
|
m_comboAction = new KWidgetAction(
|
|
m_combo,
|
|
i18n("Search"), 0, 0, 0,
|
|
actionCollection(), "isearch_combo" );
|
|
m_comboAction->setAutoSized( true );
|
|
m_comboAction->setShortcutConfigurable( false );
|
|
|
|
KActionMenu* optionMenu = new KActionMenu(
|
|
i18n("Search Options"), "configure",
|
|
actionCollection(), "isearch_options" );
|
|
optionMenu->setDelayed( false );
|
|
|
|
KToggleAction* action = new KToggleAction(
|
|
i18n("Case Sensitive"), KShortcut(),
|
|
actionCollection(), "isearch_case_sensitive" );
|
|
action->setShortcutConfigurable( false );
|
|
connect( action, TQT_SIGNAL(toggled(bool)),
|
|
this, TQT_SLOT(setCaseSensitive(bool)) );
|
|
action->setChecked( m_caseSensitive );
|
|
optionMenu->insert( action );
|
|
|
|
action = new KToggleAction(
|
|
i18n("From Beginning"), KShortcut(),
|
|
actionCollection(), "isearch_from_beginning" );
|
|
action->setShortcutConfigurable( false );
|
|
connect( action, TQT_SIGNAL(toggled(bool)),
|
|
this, TQT_SLOT(setFromBeginning(bool)) );
|
|
action->setChecked( m_fromBeginning );
|
|
optionMenu->insert( action );
|
|
|
|
action = new KToggleAction(
|
|
i18n("Regular Expression"), KShortcut(),
|
|
actionCollection(), "isearch_reg_exp" );
|
|
action->setShortcutConfigurable( false );
|
|
connect( action, TQT_SIGNAL(toggled(bool)),
|
|
this, TQT_SLOT(setRegExp(bool)) );
|
|
action->setChecked( m_regExp );
|
|
optionMenu->insert( action );
|
|
|
|
// optionMenu->insert( new KActionSeparator() );
|
|
//
|
|
// action = new KToggleAction(
|
|
// i18n("Auto-Wrap Search"), KShortcut(),
|
|
// actionCollection(), "isearch_auto_wrap" );
|
|
// connect( action, TQT_SIGNAL(toggled(bool)),
|
|
// this, TQT_SLOT(setAutoWrap(bool)) );
|
|
// action->setChecked( m_autoWrap );
|
|
// optionMenu->insert( action );
|
|
|
|
setXMLFile( "ktexteditor_isearchui.rc" );
|
|
}
|
|
|
|
ISearchPluginView::~ISearchPluginView()
|
|
{
|
|
writeConfig();
|
|
m_combo->lineEdit()->removeEventFilter( this );
|
|
delete m_combo;
|
|
delete m_label;
|
|
}
|
|
|
|
void ISearchPluginView::setView( KTextEditor::View* view )
|
|
{
|
|
m_view = view;
|
|
m_doc = m_view->document();
|
|
m_searchIF = KTextEditor::searchInterface ( m_doc );
|
|
m_cursorIF = KTextEditor::viewCursorInterface ( m_view );
|
|
m_selectIF = KTextEditor::selectionInterface ( m_doc );
|
|
if( !m_doc || !m_cursorIF || !m_selectIF ) {
|
|
m_view = 0L;
|
|
m_doc = 0L;
|
|
m_searchIF = 0L;
|
|
m_cursorIF = 0L;
|
|
m_selectIF = 0L;
|
|
}
|
|
|
|
readConfig();
|
|
}
|
|
|
|
void ISearchPluginView::readConfig()
|
|
{
|
|
// KConfig* config = instance()->config();
|
|
}
|
|
|
|
void ISearchPluginView::writeConfig()
|
|
{
|
|
// KConfig* config = instance()->config();
|
|
}
|
|
|
|
void ISearchPluginView::setCaseSensitive( bool caseSensitive )
|
|
{
|
|
m_caseSensitive = caseSensitive;
|
|
}
|
|
|
|
void ISearchPluginView::setFromBeginning( bool fromBeginning )
|
|
{
|
|
m_fromBeginning = fromBeginning;
|
|
|
|
if( m_fromBeginning ) {
|
|
m_searchLine = m_searchCol = 0;
|
|
}
|
|
}
|
|
|
|
void ISearchPluginView::setRegExp( bool regExp )
|
|
{
|
|
m_regExp = regExp;
|
|
}
|
|
|
|
void ISearchPluginView::setAutoWrap( bool autoWrap )
|
|
{
|
|
m_autoWrap = autoWrap;
|
|
}
|
|
|
|
bool ISearchPluginView::eventFilter( TQObject* o, TQEvent* e )
|
|
{
|
|
if( TQT_BASE_OBJECT(o) != TQT_BASE_OBJECT(m_combo->lineEdit()) )
|
|
return false;
|
|
|
|
if( e->type() == TQEvent::FocusIn ) {
|
|
TQFocusEvent* focusEvent = (TQFocusEvent*)e;
|
|
if( focusEvent->reason() == TQFocusEvent::ActiveWindow ||
|
|
focusEvent->reason() == TQFocusEvent::Popup )
|
|
return false;
|
|
startSearch();
|
|
}
|
|
|
|
if( e->type() == TQEvent::FocusOut ) {
|
|
TQFocusEvent* focusEvent = (TQFocusEvent*)e;
|
|
if( focusEvent->reason() == TQFocusEvent::ActiveWindow ||
|
|
focusEvent->reason() == TQFocusEvent::Popup )
|
|
return false;
|
|
endSearch();
|
|
}
|
|
|
|
if( e->type() == TQEvent::KeyPress ) {
|
|
TQKeyEvent *keyEvent = (TQKeyEvent*)e;
|
|
if( keyEvent->key() == Qt::Key_Escape )
|
|
quitToView( TQString::null );
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
// Sigh... i18n hell.
|
|
void ISearchPluginView::updateLabelText(
|
|
bool failing /* = false */, bool reverse /* = false */,
|
|
bool wrapped /* = false */, bool overwrapped /* = false */ )
|
|
{
|
|
TQString text;
|
|
// Reverse binary:
|
|
// 0000
|
|
if( !failing && !reverse && !wrapped && !overwrapped ) {
|
|
text = i18n("Incremental Search", "I-Search:");
|
|
// 1000
|
|
} else if ( failing && !reverse && !wrapped && !overwrapped ) {
|
|
text = i18n("Incremental Search found no match", "Failing I-Search:");
|
|
// 0100
|
|
} else if ( !failing && reverse && !wrapped && !overwrapped ) {
|
|
text = i18n("Incremental Search in the reverse direction", "I-Search Backward:");
|
|
// 1100
|
|
} else if ( failing && reverse && !wrapped && !overwrapped ) {
|
|
text = i18n("Failing I-Search Backward:");
|
|
// 0010
|
|
} else if ( !failing && !reverse && wrapped && !overwrapped ) {
|
|
text = i18n("Incremental Search has passed the end of the document", "Wrapped I-Search:");
|
|
// 1010
|
|
} else if ( failing && !reverse && wrapped && !overwrapped ) {
|
|
text = i18n("Failing Wrapped I-Search:");
|
|
// 0110
|
|
} else if ( !failing && reverse && wrapped && !overwrapped ) {
|
|
text = i18n("Wrapped I-Search Backward:");
|
|
// 1110
|
|
} else if ( failing && reverse && wrapped && !overwrapped ) {
|
|
text = i18n("Failing Wrapped I-Search Backward:");
|
|
// 0011
|
|
} else if ( !failing && !reverse && overwrapped ) {
|
|
text = i18n("Incremental Search has passed both the end of the document "
|
|
"and the original starting position", "Overwrapped I-Search:");
|
|
// 1011
|
|
} else if ( failing && !reverse && overwrapped ) {
|
|
text = i18n("Failing Overwrapped I-Search:");
|
|
// 0111
|
|
} else if ( !failing && reverse && overwrapped ) {
|
|
text = i18n("Overwrapped I-Search Backwards:");
|
|
// 1111
|
|
} else if ( failing && reverse && overwrapped ) {
|
|
text = i18n("Failing Overwrapped I-Search Backward:");
|
|
} else {
|
|
text = i18n("Error: unknown i-search state!");
|
|
}
|
|
m_label->setText( text );
|
|
}
|
|
|
|
void ISearchPluginView::slotSearchForwardAction()
|
|
{
|
|
slotSearchAction( false );
|
|
}
|
|
|
|
void ISearchPluginView::slotSearchBackwardAction()
|
|
{
|
|
slotSearchAction( true );
|
|
}
|
|
|
|
void ISearchPluginView::slotSearchAction( bool reverse )
|
|
{
|
|
if( !m_combo->hasFocus() ) {
|
|
if( m_comboAction->container(0) && m_comboAction->container(0)->isHidden() ) {
|
|
m_toolBarWasHidden = true;
|
|
m_comboAction->container(0)->setHidden( false );
|
|
} else {
|
|
m_toolBarWasHidden = false;
|
|
}
|
|
m_combo->setFocus(); // Will call startSearch()
|
|
} else {
|
|
nextMatch( reverse );
|
|
}
|
|
}
|
|
|
|
void ISearchPluginView::nextMatch( bool reverse )
|
|
{
|
|
TQString text = m_combo->currentText();
|
|
if( text.isEmpty() )
|
|
return;
|
|
if( state != MatchSearch ) {
|
|
// Last search was performed by typing, start from that match.
|
|
if( !reverse ) {
|
|
m_searchLine = m_foundLine;
|
|
m_searchCol = m_foundCol + m_matchLen;
|
|
} else {
|
|
m_searchLine = m_foundLine;
|
|
m_searchCol = m_foundCol;
|
|
}
|
|
state = MatchSearch;
|
|
}
|
|
|
|
bool found = iSearch( m_searchLine, m_searchCol, text, reverse, m_autoWrap );
|
|
if( found ) {
|
|
m_searchLine = m_foundLine;
|
|
m_searchCol = m_foundCol + m_matchLen;
|
|
} else {
|
|
m_wrapped = true;
|
|
m_searchLine = m_searchCol = 0;
|
|
}
|
|
}
|
|
|
|
void ISearchPluginView::startSearch()
|
|
{
|
|
if( !m_view ) return;
|
|
|
|
m_searchForwardAction->setText( i18n("Next Incremental Search Match") );
|
|
m_searchBackwardAction->setText( i18n("Previous Incremental Search Match") );
|
|
|
|
m_wrapped = false;
|
|
|
|
if( m_fromBeginning ) {
|
|
m_startLine = m_startCol = 0;
|
|
} else {
|
|
m_cursorIF->cursorPositionReal( &m_startLine, &m_startCol );
|
|
}
|
|
m_searchLine = m_startLine;
|
|
m_searchCol = m_startCol;
|
|
|
|
updateLabelText( false, m_searchBackward );
|
|
|
|
m_combo->blockSignals( true );
|
|
|
|
TQString text = m_selectIF->selection();
|
|
if( text.isEmpty() )
|
|
text = m_lastString;
|
|
m_combo->setCurrentText( text );
|
|
|
|
m_combo->blockSignals( false );
|
|
m_combo->lineEdit()->selectAll();
|
|
|
|
// kdDebug() << "Starting search at " << m_startLine << ", " << m_startCol << endl;
|
|
}
|
|
|
|
void ISearchPluginView::endSearch()
|
|
{
|
|
m_searchForwardAction->setText( i18n("Search Incrementally") );
|
|
m_searchBackwardAction->setText( i18n("Search Incrementally Backwards") );
|
|
|
|
updateLabelText();
|
|
|
|
if( m_toolBarWasHidden && m_comboAction->containerCount() > 0 ) {
|
|
m_comboAction->container(0)->setHidden( true );
|
|
}
|
|
}
|
|
|
|
void ISearchPluginView::quitToView( const TQString &text )
|
|
{
|
|
if( !text.isNull() && !text.isEmpty() ) {
|
|
m_combo->addToHistory( text );
|
|
m_lastString = text;
|
|
}
|
|
|
|
if( m_view ) {
|
|
m_view->setFocus(); // Will call endSearch()
|
|
}
|
|
}
|
|
|
|
void ISearchPluginView::slotTextChanged( const TQString& text )
|
|
{
|
|
state = TextSearch;
|
|
|
|
if( text.isEmpty() )
|
|
return;
|
|
|
|
iSearch( m_searchLine, m_searchCol, text, m_searchBackward, m_autoWrap );
|
|
}
|
|
|
|
void ISearchPluginView::slotReturnPressed( const TQString& text )
|
|
{
|
|
quitToView( text );
|
|
}
|
|
|
|
void ISearchPluginView::slotAddContextMenuItems( TQPopupMenu *menu )
|
|
{
|
|
if( menu ) {
|
|
menu->insertSeparator();
|
|
menu->insertItem( i18n("Case Sensitive"), this,
|
|
TQT_SLOT(setCaseSensitive(bool)));
|
|
menu->insertItem( i18n("From Beginning"), this,
|
|
TQT_SLOT(setFromBeginning(bool)));
|
|
menu->insertItem( i18n("Regular Expression"), this,
|
|
TQT_SLOT(setRegExp(bool)));
|
|
//menu->insertItem( i18n("Auto-Wrap Search"), this,
|
|
// TQT_SLOT(setAutoWrap(bool)));
|
|
}
|
|
}
|
|
|
|
bool ISearchPluginView::iSearch(
|
|
uint startLine, uint startCol,
|
|
const TQString& text, bool reverse,
|
|
bool autoWrap )
|
|
{
|
|
if( !m_view ) return false;
|
|
|
|
// kdDebug() << "Searching for " << text << " at " << startLine << ", " << startCol << endl;
|
|
bool found = false;
|
|
if( !m_regExp ) {
|
|
found = m_searchIF->searchText( startLine,
|
|
startCol,
|
|
text,
|
|
&m_foundLine,
|
|
&m_foundCol,
|
|
&m_matchLen,
|
|
m_caseSensitive,
|
|
reverse );
|
|
} else {
|
|
found = m_searchIF->searchText( startLine,
|
|
startCol,
|
|
TQRegExp( text ),
|
|
&m_foundLine,
|
|
&m_foundCol,
|
|
&m_matchLen,
|
|
reverse );
|
|
}
|
|
if( found ) {
|
|
// kdDebug() << "Found '" << text << "' at " << m_foundLine << ", " << m_foundCol << endl;
|
|
// v->gotoLineNumber( m_foundLine );
|
|
m_cursorIF->setCursorPositionReal( m_foundLine, m_foundCol + m_matchLen );
|
|
m_selectIF->setSelection( m_foundLine, m_foundCol, m_foundLine, m_foundCol + m_matchLen );
|
|
} else if ( autoWrap ) {
|
|
m_wrapped = true;
|
|
found = iSearch( 0, 0, text, reverse, false );
|
|
}
|
|
// FIXME
|
|
bool overwrapped = ( m_wrapped &&
|
|
((m_foundLine > m_startLine ) ||
|
|
(m_foundLine == m_startLine && m_foundCol >= m_startCol)) );
|
|
// kdDebug() << "Overwrap = " << overwrapped << ". Start was " << m_startLine << ", " << m_startCol << endl;
|
|
updateLabelText( !found, reverse, m_wrapped, overwrapped );
|
|
return found;
|
|
}
|
|
|
|
ISearchPlugin::ISearchPlugin( TQObject *parent, const char* name, const TQStringList& )
|
|
: KTextEditor::Plugin ( (KTextEditor::Document*) parent, name )
|
|
{
|
|
}
|
|
|
|
ISearchPlugin::~ISearchPlugin()
|
|
{
|
|
}
|
|
|
|
void ISearchPlugin::addView(KTextEditor::View *view)
|
|
{
|
|
ISearchPluginView *nview = new ISearchPluginView (view);
|
|
nview->setView (view);
|
|
m_views.append (nview);
|
|
}
|
|
|
|
void ISearchPlugin::removeView(KTextEditor::View *view)
|
|
{
|
|
for (uint z=0; z < m_views.count(); z++)
|
|
{
|
|
if (m_views.tqat(z)->parentClient() == view)
|
|
{
|
|
ISearchPluginView *nview = m_views.tqat(z);
|
|
m_views.remove (nview);
|
|
delete nview;
|
|
}
|
|
}
|
|
}
|