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.
tdebase/kate/app/kategrepdialog.cpp

545 lines
17 KiB

/* This file is part of the KDE project
Copyright (C) 2001 Christoph Cullmann <cullmann@kde.org>
Copyright (C) 2001 Joseph Wenninger <jowenn@kde.org>
Copyright (C) 2001, 2004 Anders Lund <anders.lund@lund.tdcadsl.dk>
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 "kategrepdialog.h"
#include "katemainwindow.h"
#include <tqobject.h>
#include <tqlayout.h>
#include <tqlabel.h>
#include <tqcheckbox.h>
#include <tqevent.h>
#include <tqlistbox.h>
#include <tqregexp.h>
#include <tqwhatsthis.h>
#include <tqcursor.h>
#include <kapplication.h>
#include <kaccelmanager.h>
#include <kbuttonbox.h>
#include <kfiledialog.h>
#include <kprocess.h>
#include <kapplication.h>
#include <klocale.h>
#include <kiconloader.h>
#include <kmessagebox.h>
#include <kpushbutton.h>
#include <kurlrequester.h>
#include <kurlcompletion.h>
#include <kcombobox.h>
#include <klineedit.h>
const char *template_desc[] = {
"normal",
"assignment",
"->MEMBER(",
"class::MEMBER(",
"OBJECT->member(",
0
};
const char *strTemplate[] = {
"%s",
"\\<%s\\>[\t ]*=[^=]",
"\\->[\\t ]*\\<%s\\>[\\t ]*(",
"[a-z0-9_$]\\+[\\t ]*::[\\t ]*\\<%s\\>[\\t ]*(",
"\\<%s\\>[\\t ]*\\->[\\t ]*[a-z0-9_$]\\+[\\t ]*(",
0
};
GrepTool::GrepTool(TQWidget *parent, const char *name)
: TQWidget(parent, name/*, false*/), m_fixFocus(true), childproc(0)
{
setCaption(i18n("Find in Files"));
config = KGlobal::config();
config->setGroup("GrepTool");
lastSearchItems = config->readListEntry("LastSearchItems");
lastSearchPaths = config->readListEntry("LastSearchPaths");
lastSearchFiles = config->readListEntry("LastSearchFiles");
if( lastSearchFiles.isEmpty() )
{
// if there are no entries, most probably the first Kate start.
// Initialize with default values.
lastSearchFiles << "*.h,*.hxx,*.cpp,*.cc,*.C,*.cxx,*.idl,*.c"
<< "*.cpp,*.cc,*.C,*.cxx,*.c"
<< "*.h,*.hxx,*.idl"
<< "*";
}
TQGridLayout *layout = new TQGridLayout(this, 6, 3, 4, 4);
layout->setColStretch(0, 10);
layout->addColSpacing(1, 10);
layout->setColStretch(1, 0);
layout->setColStretch(2, 1);
layout->setRowStretch(1, 0);
layout->setRowStretch(2, 10);
layout->setRowStretch(4, 0);
TQGridLayout *loInput = new TQGridLayout(4, 2, 4);
layout->addLayout(loInput, 0, 0);
loInput->setColStretch(0, 0);
loInput->setColStretch(1, 20);
TQLabel *lPattern = new TQLabel(i18n("Pattern:"), this);
lPattern->setFixedSize(lPattern->tqsizeHint());
loInput->addWidget(lPattern, 0, 0, Qt::AlignRight | Qt::AlignVCenter);
TQBoxLayout *loPattern = new TQHBoxLayout( 4 );
loInput->addLayout( loPattern, 0, 1 );
cmbPattern = new KComboBox(true, this);
cmbPattern->setDuplicatesEnabled(false);
cmbPattern->insertStringList(lastSearchItems);
cmbPattern->setEditText(TQString::null);
cmbPattern->setInsertionPolicy(TQComboBox::NoInsertion);
lPattern->setBuddy(cmbPattern);
cmbPattern->setFocus();
cmbPattern->setMinimumSize(cmbPattern->tqsizeHint());
loPattern->addWidget( cmbPattern );
cbCasesensitive = new TQCheckBox(i18n("Case sensitive"), this);
cbCasesensitive->setMinimumWidth(cbCasesensitive->tqsizeHint().width());
cbCasesensitive->setChecked(config->readBoolEntry("CaseSensitive", true));
loPattern->addWidget(cbCasesensitive);
cbRegex = new TQCheckBox( i18n("Regular expression"), this );
cbRegex->setMinimumWidth( cbRegex->tqsizeHint().width() );
cbRegex->setChecked( config->readBoolEntry( "Regex", true ) );
loPattern->addWidget( cbRegex );
loPattern->setStretchFactor( cmbPattern, 100 );
TQLabel *lTemplate = new TQLabel(i18n("Template:"), this);
lTemplate->setFixedSize(lTemplate->tqsizeHint());
loInput->addWidget(lTemplate, 1, 0, Qt::AlignRight | Qt::AlignVCenter);
TQBoxLayout *loTemplate = new TQHBoxLayout(4);
loInput->addLayout(loTemplate, 1, 1);
leTemplate = new KLineEdit(this);
lTemplate->setBuddy(leTemplate);
leTemplate->setText(strTemplate[0]);
leTemplate->setMinimumSize(leTemplate->tqsizeHint());
loTemplate->addWidget(leTemplate);
KComboBox *cmbTemplate = new KComboBox(false, this);
cmbTemplate->insertStrList(template_desc);
cmbTemplate->adjustSize();
cmbTemplate->setFixedSize(cmbTemplate->size());
loTemplate->addWidget(cmbTemplate);
TQLabel *lFiles = new TQLabel(i18n("Files:"), this);
lFiles->setFixedSize(lFiles->tqsizeHint());
loInput->addWidget(lFiles, 2, 0, Qt::AlignRight | Qt::AlignVCenter);
cmbFiles = new KComboBox(true, this);
lFiles->setBuddy(TQT_TQWIDGET(cmbFiles->focusProxy()));
cmbFiles->setMinimumSize(cmbFiles->tqsizeHint());
cmbFiles->setInsertionPolicy(TQComboBox::NoInsertion);
cmbFiles->setDuplicatesEnabled(false);
cmbFiles->insertStringList(lastSearchFiles);
loInput->addWidget(cmbFiles, 2, 1);
TQLabel *lDir = new TQLabel(i18n("Folder:"), this);
lDir->setFixedSize(lDir->tqsizeHint());
loInput->addWidget(lDir, 3, 0, Qt::AlignRight | Qt::AlignVCenter);
TQBoxLayout *loDir = new TQHBoxLayout(3);
loInput->addLayout(loDir, 3, 1);
KComboBox* cmbUrl = new KComboBox(true, this);
cmbUrl->setMinimumWidth(80); // make sure that 800x600 res works
cmbUrl->setDuplicatesEnabled(false);
cmbUrl->setInsertionPolicy(TQComboBox::NoInsertion);
cmbDir = new KURLRequester( cmbUrl, this, "dir combo" );
cmbDir->completionObject()->setMode(KURLCompletion::DirCompletion);
cmbDir->comboBox()->insertStringList(lastSearchPaths);
cmbDir->setMode( KFile::Directory|KFile::LocalOnly );
loDir->addWidget(cmbDir, 1);
lDir->setBuddy(cmbDir);
cbRecursive = new TQCheckBox(i18n("Recursive"), this);
cbRecursive->setMinimumWidth(cbRecursive->tqsizeHint().width());
cbRecursive->setChecked(config->readBoolEntry("Recursive", true));
loDir->addWidget(cbRecursive);
KButtonBox *actionbox = new KButtonBox(this, Qt::Vertical);
layout->addWidget(actionbox, 0, 2);
actionbox->addStretch();
btnSearch = static_cast<KPushButton*>(actionbox->addButton(KGuiItem(i18n("Find"),"find")));
btnSearch->setDefault(true);
btnClear = static_cast<KPushButton*>(actionbox->addButton( KStdGuiItem::clear() ));
actionbox->addStretch();
actionbox->layout();
lbResult = new TQListBox(this);
TQFontMetrics rb_fm(lbResult->fontMetrics());
layout->addMultiCellWidget(lbResult, 2, 2, 0, 2);
layout->activate();
KAcceleratorManager::manage( this );
TQWhatsThis::add(lPattern,
i18n("<p>Enter the expression you want to search for here."
"<p>If 'regular expression' is unchecked, any non-space letters in your "
"expression will be escaped with a backslash character."
"<p>Possible meta characters are:<br>"
"<b>.</b> - Matches any character<br>"
"<b>^</b> - Matches the beginning of a line<br>"
"<b>$</b> - Matches the end of a line<br>"
"<b>\\&lt;</b> - Matches the beginning of a word<br>"
"<b>\\&gt;</b> - Matches the end of a word"
"<p>The following repetition operators exist:<br>"
"<b>?</b> - The preceding item is matched at most once<br>"
"<b>*</b> - The preceding item is matched zero or more times<br>"
"<b>+</b> - The preceding item is matched one or more times<br>"
"<b>{<i>n</i>}</b> - The preceding item is matched exactly <i>n</i> times<br>"
"<b>{<i>n</i>,}</b> - The preceding item is matched <i>n</i> or more times<br>"
"<b>{,<i>n</i>}</b> - The preceding item is matched at most <i>n</i> times<br>"
"<b>{<i>n</i>,<i>m</i>}</b> - The preceding item is matched at least <i>n</i>, "
"but at most <i>m</i> times."
"<p>Furthermore, backreferences to bracketed subexpressions are available "
"via the notation <code>\\#</code>."
"<p>See the grep(1) documentation for the full documentation."
));
TQWhatsThis::add(lFiles,
i18n("Enter the file name pattern of the files to search here.\n"
"You may give several patterns separated by commas."));
TQWhatsThis::add(lTemplate,
i18n("You can choose a template for the pattern from the combo box\n"
"and edit it here. The string %s in the template is replaced\n"
"by the pattern input field, resulting in the regular expression\n"
"to search for."));
TQWhatsThis::add(lDir,
i18n("Enter the folder which contains the files in which you want to search."));
TQWhatsThis::add(cbRecursive,
i18n("Check this box to search in all subfolders."));
TQWhatsThis::add(cbCasesensitive,
i18n("If this option is enabled (the default), the search will be case sensitive."));
TQWhatsThis::add( cbRegex, i18n(
"<p>If this is enabled, your pattern will be passed unmodified to "
"<em>grep(1)</em>. Otherwise, all characters that are not letters will be "
"escaped using a backslash character to prevent grep from interpreting "
"them as part of the expression.") );
TQWhatsThis::add(lbResult,
i18n("The results of the grep run are listed here. Select a\n"
"filename/line number combination and press Enter or doubleclick\n"
"on the item to show the respective line in the editor."));
// event filter, do something relevant for RETURN
cmbPattern->installEventFilter( this );
leTemplate->installEventFilter( this );
cmbFiles->installEventFilter( this );
cmbDir->comboBox()->installEventFilter( this );
connect( cmbTemplate, TQT_SIGNAL(activated(int)),
TQT_SLOT(templateActivated(int)) );
connect( lbResult, TQT_SIGNAL(selected(const TQString&)),
TQT_SLOT(itemSelected(const TQString&)) );
connect( btnSearch, TQT_SIGNAL(clicked()),
TQT_SLOT(slotSearch()) );
connect( btnClear, TQT_SIGNAL(clicked()),
TQT_SLOT(slotClear()) );
connect( cmbPattern->lineEdit(), TQT_SIGNAL(textChanged ( const TQString & )),
TQT_SLOT( patternTextChanged( const TQString & )));
patternTextChanged( cmbPattern->lineEdit()->text());
}
GrepTool::~GrepTool()
{
delete childproc;
}
void GrepTool::patternTextChanged( const TQString & _text)
{
btnSearch->setEnabled( !_text.isEmpty() );
}
void GrepTool::templateActivated(int index)
{
leTemplate->setText(strTemplate[index]);
}
void GrepTool::itemSelected(const TQString& item)
{
int pos;
TQString filename, linenumber;
TQString str = item;
if ( (pos = str.tqfind(':')) != -1)
{
filename = str.left(pos);
str = str.mid(pos+1);
if ( (pos = str.tqfind(':')) != -1)
{
filename = m_workingDir + TQDir::separator() + filename;
linenumber = str.left(pos);
emit itemSelected(filename,linenumber.toInt()-1);
}
}
}
void GrepTool::processOutput()
{
int pos;
while ( (pos = buf.tqfind('\n')) != -1)
{
TQString item = buf.mid(2,pos-2);
if (!item.isEmpty())
lbResult->insertItem(item);
buf = buf.mid(pos+1);
}
kapp->processEvents();
}
void GrepTool::slotSearch()
{
if ( cmbPattern->currentText().isEmpty() )
{
cmbPattern->setFocus();
return;
}
if ( cmbDir->url().isEmpty() || ! TQDir(cmbDir->url()).exists() )
{
cmbDir->setFocus();
KMessageBox::information( this, i18n(
"You must enter an existing local folder in the 'Folder' entry."),
i18n("Invalid Folder"), "Kate grep tool: invalid folder" );
return;
}
if ( ! leTemplate->text().tqcontains("%s") )
{
leTemplate->setFocus();
return;
}
if ( childproc && childproc->isRunning() )
{
childproc->kill();
return;
}
slotClear ();
m_workingDir = cmbDir->url();
TQString s = cmbPattern->currentText();
if ( ! cbRegex->isChecked() )
s.tqreplace( TQRegExp( "([^\\w'()<>])" ), "\\\\1" );
TQString pattern = leTemplate->text();
pattern.tqreplace( "%s", s );
childproc = new KProcess();
childproc->setWorkingDirectory( m_workingDir );
*childproc << "find" << ".";
if (!cbRecursive->isChecked())
*childproc << "-maxdepth" << "1";
if (!cmbFiles->currentText().isEmpty() )
{
TQStringList files = TQStringList::split ( ",", cmbFiles->currentText(), FALSE );
*childproc << "(";
bool first = true;
for ( TQStringList::Iterator it = files.begin(); it != files.end(); ++it )
{
if (!first)
*childproc << "-o";
*childproc << "-name" << (*it);
first = false;
}
*childproc << ")";
}
*childproc << "-exec" << "grep";
if (!cbCasesensitive->isChecked())
*childproc << "-i";
*childproc << "-n" << "-e" << pattern << "{}";
*childproc << "/dev/null"; //trick to have grep always display the filename
*childproc << ";";
connect( childproc, TQT_SIGNAL(processExited(KProcess *)),
TQT_SLOT(childExited()) );
connect( childproc, TQT_SIGNAL(receivedStdout(KProcess *, char *, int)),
TQT_SLOT(receivedOutput(KProcess *, char *, int)) );
connect( childproc, TQT_SIGNAL(receivedStderr(KProcess *, char *, int)),
TQT_SLOT(receivedErrOutput(KProcess *, char *, int)) );
// actually it should be checked whether the process was started successfully
lbResult->setCursor( TQCursor(Qt::WaitCursor) );
btnClear->setEnabled( false );
btnSearch->setGuiItem( KGuiItem(i18n("Cancel"), "button_cancel"));
childproc->start(KProcess::NotifyOnExit, KProcess::AllOutput);
}
void GrepTool::slotSearchFor(const TQString &pattern)
{
slotClear();
cmbPattern->setEditText(pattern);
slotSearch();
}
void GrepTool::finish()
{
btnSearch->setEnabled( !cmbPattern->lineEdit()->text().isEmpty() );
buf += '\n';
processOutput();
delete childproc;
childproc = 0;
config->setGroup("GrepTool");
TQString cmbText = cmbPattern->currentText();
bool itemsRemoved = lastSearchItems.remove(cmbText) > 0;
lastSearchItems.prepend(cmbText);
if (itemsRemoved)
{
cmbPattern->removeItem(cmbPattern->currentItem());
}
cmbPattern->insertItem(cmbText, 0);
cmbPattern->setCurrentItem(0);
if (lastSearchItems.count() > 10) {
lastSearchItems.pop_back();
cmbPattern->removeItem(cmbPattern->count() - 1);
}
config->writeEntry("LastSearchItems", lastSearchItems);
cmbText = cmbDir->url();
itemsRemoved = lastSearchPaths.remove(cmbText) > 0;
lastSearchPaths.prepend(cmbText);
if (itemsRemoved)
{
cmbDir->comboBox()->removeItem(cmbDir->comboBox()->currentItem());
}
cmbDir->comboBox()->insertItem(cmbText, 0);
cmbDir->comboBox()->setCurrentItem(0);
if (lastSearchPaths.count() > 10)
{
lastSearchPaths.pop_back();
cmbDir->comboBox()->removeItem(cmbDir->comboBox()->count() - 1);
}
config->writeEntry("LastSearchPaths", lastSearchPaths);
cmbText = cmbFiles->currentText();
itemsRemoved = lastSearchFiles.remove(cmbText) > 0;
lastSearchFiles.prepend(cmbText);
if (itemsRemoved)
{
cmbFiles->removeItem(cmbFiles->currentItem());
}
cmbFiles->insertItem(cmbText, 0);
cmbFiles->setCurrentItem(0);
if (lastSearchFiles.count() > 10) {
lastSearchFiles.pop_back();
cmbFiles->removeItem(cmbFiles->count() - 1);
}
config->writeEntry("LastSearchFiles", lastSearchFiles);
config->writeEntry("Recursive", cbRecursive->isChecked());
config->writeEntry("CaseSensitive", cbCasesensitive->isChecked());
config->writeEntry("Regex", cbRegex->isChecked());
}
void GrepTool::slotCancel()
{
finish();
}
void GrepTool::childExited()
{
// int status = childproc->exitStatus();
lbResult->unsetCursor();
btnClear->setEnabled( true );
btnSearch->setGuiItem( KGuiItem(i18n("Find"), "find") );
if ( ! errbuf.isEmpty() )
{
KMessageBox::information( parentWidget(), i18n("<strong>Error:</strong><p>") + errbuf, i18n("Grep Tool Error") );
errbuf.truncate(0);
}
else
finish();
}
void GrepTool::receivedOutput(KProcess */*proc*/, char *buffer, int buflen)
{
buf += TQCString(buffer, buflen+1);
processOutput();
}
void GrepTool::receivedErrOutput(KProcess */*proc*/, char *buffer, int buflen)
{
errbuf += TQCString( buffer, buflen + 1 );
}
void GrepTool::slotClear()
{
finish();
lbResult->clear();
}
void GrepTool::updateDirName(const TQString &dir)
{
if (m_lastUpdatedDir != dir)
{
setDirName (dir);
m_lastUpdatedDir = dir;
}
}
void GrepTool::setDirName(const TQString &dir){
cmbDir->setURL(dir);
}
bool GrepTool::eventFilter( TQObject *o, TQEvent *e )
{
if ( e->type() == TQEvent::KeyPress && (
((TQKeyEvent*)e)->key() == Qt::Key_Return ||
((TQKeyEvent*)e)->key() == Qt::Key_Enter ) )
{
slotSearch();
return true;
}
return TQWidget::eventFilter( o, e );
}
void GrepTool::focusInEvent ( TQFocusEvent * ev )
{
TQWidget::focusInEvent(ev);
if (m_fixFocus) {
m_fixFocus = false;
cmbPattern->setFocus();
}
}
void GrepTool::showEvent( TQShowEvent * ev )
{
TQWidget::showEvent(ev);
m_fixFocus = true;
}
#include "kategrepdialog.moc"