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.
tdelibs/kate/part/katedialogs.cpp

1741 lines
60 KiB

/* This file is part of the KDE libraries
Copyright (C) 2002, 2003 Anders Lund <anders.lund@lund.tdcadsl.dk>
Copyright (C) 2003 Christoph Cullmann <cullmann@kde.org>
Copyright (C) 2001 Joseph Wenninger <jowenn@kde.org>
Based on work of:
Copyright (C) 1999 Jochen Wilhelmy <digisnap@cs.tu-berlin.de>
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.
*/
//BEGIN Includes
#include "katedialogs.h"
#include "katedialogs.moc"
#include "kateautoindent.h"
#include "katebuffer.h"
#include "kateconfig.h"
#include "katedocument.h"
#include "katefactory.h"
#include "kateschema.h"
#include "katesyntaxdocument.h"
#include "kateview.h"
#include <ktexteditor/configinterfaceextension.h>
#include <ktexteditor/plugin.h>
#include <kio/job.h>
#include <kio/jobclasses.h>
#include <kio/netaccess.h>
#include <kaccel.h>
#include <kapplication.h>
#include <kbuttonbox.h>
#include <kcharsets.h>
#include <kcolorbutton.h>
#include <kcolorcombo.h>
#include <kcolordialog.h>
#include <kcombobox.h>
#include <kconfig.h>
#include <kdebug.h>
#include <kfontdialog.h>
#include <kglobal.h>
#include <kglobalsettings.h>
#include <kiconloader.h>
#include <kkeybutton.h>
#include <kkeydialog.h>
#include <klineedit.h>
#include <klistview.h>
#include <klocale.h>
#include <kmessagebox.h>
#include <kmimetypechooser.h>
#include <knuminput.h>
#include <kparts/componentfactory.h>
#include <kpopupmenu.h>
#include <kprocess.h>
#include <kprocio.h>
#include <kregexpeditorinterface.h>
#include <krun.h>
#include <kseparator.h>
#include <kstandarddirs.h>
#include <ktempfile.h>
#include <qbuttongroup.h>
#include <qcheckbox.h>
#include <qcombobox.h>
#include <qdialog.h>
#include <qdom.h>
#include <qfile.h>
#include <qgrid.h>
#include <qgroupbox.h>
#include <qhbox.h>
#include <qheader.h>
#include <qhgroupbox.h>
#include <qlabel.h>
#include <qlayout.h>
#include <qlineedit.h>
#include <qlistbox.h>
#include <qlistview.h>
#include <qmap.h>
#include <qobjectlist.h>
#include <qpainter.h>
#include <qpointarray.h>
#include <qptrcollection.h>
#include <qpushbutton.h>
#include <qradiobutton.h>
#include <qslider.h>
#include <qspinbox.h>
#include <qstringlist.h>
#include <qtabwidget.h>
#include <qtextcodec.h>
#include <qtoolbutton.h>
#include <qvbox.h>
#include <qvgroupbox.h>
#include <qwhatsthis.h>
#include <qwidgetstack.h>
// trailing slash is important
#define HLDOWNLOADPATH "http://kate.kde.org/syntax/"
//END
//BEGIN KateConfigPage
KateConfigPage::KateConfigPage ( QWidget *parent, const char *name )
: Kate::ConfigPage (parent, name)
, m_changed (false)
{
connect (this, SIGNAL(changed()), this, SLOT(somethingHasChanged ()));
}
KateConfigPage::~KateConfigPage ()
{
}
void KateConfigPage::somethingHasChanged ()
{
m_changed = true;
kdDebug (13000) << "TEST: something changed on the config page: " << this << endl;
}
//END KateConfigPage
//BEGIN KateIndentConfigTab
const int KateIndentConfigTab::flags[] = {
KateDocument::cfSpaceIndent,
KateDocument::cfKeepIndentProfile,
KateDocument::cfKeepExtraSpaces,
KateDocument::cfTabIndents,
KateDocument::cfBackspaceIndents,
KateDocumentConfig::cfDoxygenAutoTyping,
KateDocumentConfig::cfMixedIndent,
KateDocumentConfig::cfIndentPastedText
};
KateIndentConfigTab::KateIndentConfigTab(QWidget *parent)
: KateConfigPage(parent)
{
QVBoxLayout *layout = new QVBoxLayout(this, 0, KDialog::spacingHint() );
int configFlags = KateDocumentConfig::global()->configFlags();
QVGroupBox *gbAuto = new QVGroupBox(i18n("Automatic Indentation"), this);
QHBox *indentLayout = new QHBox(gbAuto);
indentLayout->setSpacing(KDialog::spacingHint());
QLabel *indentLabel = new QLabel(i18n("&Indentation mode:"), indentLayout);
m_indentMode = new KComboBox (indentLayout);
m_indentMode->insertStringList (KateAutoIndent::listModes());
indentLabel->setBuddy(m_indentMode);
m_configPage = new QPushButton(SmallIconSet("configure"), i18n("Configure..."), indentLayout);
opt[5] = new QCheckBox(i18n("Insert leading Doxygen \"*\" when typing"), gbAuto);
opt[7] = new QCheckBox(i18n("Adjust indentation of code pasted from the clipboard"), gbAuto);
QVGroupBox *gbSpaces = new QVGroupBox(i18n("Indentation with Spaces"), this);
QVBox *spaceLayout = new QVBox(gbSpaces);
opt[0] = new QCheckBox(i18n("Use &spaces instead of tabs to indent"), spaceLayout );
opt[6] = new QCheckBox(i18n("Emacs style mixed mode"), spaceLayout);
indentationWidth = new KIntNumInput(KateDocumentConfig::global()->indentationWidth(), spaceLayout);
indentationWidth->setRange(1, 16, 1, false);
indentationWidth->setLabel(i18n("Number of spaces:"), AlignVCenter);
opt[1] = new QCheckBox(i18n("Keep indent &profile"), this);
opt[2] = new QCheckBox(i18n("&Keep extra spaces"), this);
QVGroupBox *keys = new QVGroupBox(i18n("Keys to Use"), this);
opt[3] = new QCheckBox(i18n("&Tab key indents"), keys);
opt[4] = new QCheckBox(i18n("&Backspace key indents"), keys);
QRadioButton *rb1, *rb2, *rb3;
m_tabs = new QButtonGroup( 1, Qt::Horizontal, i18n("Tab Key Mode if Nothing Selected"), this );
m_tabs->setRadioButtonExclusive( true );
m_tabs->insert( rb1=new QRadioButton( i18n("Insert indent &characters"), m_tabs ), 0 );
m_tabs->insert( rb2=new QRadioButton( i18n("I&nsert tab character"), m_tabs ), 1 );
m_tabs->insert( rb3=new QRadioButton( i18n("Indent current &line"), m_tabs ), 2 );
opt[0]->setChecked(configFlags & flags[0]);
opt[1]->setChecked(configFlags & flags[1]);
opt[2]->setChecked(configFlags & flags[2]);
opt[3]->setChecked(configFlags & flags[3]);
opt[4]->setChecked(configFlags & flags[4]);
opt[5]->setChecked(configFlags & flags[5]);
opt[6]->setChecked(configFlags & flags[6]);
opt[7]->setChecked(configFlags & flags[7]);
layout->addWidget(gbAuto);
layout->addWidget(gbSpaces);
layout->addWidget(opt[1]);
layout->addWidget(opt[2]);
layout->addWidget(keys);
layout->addWidget(m_tabs, 0);
layout->addStretch();
// What is this? help
QWhatsThis::add(opt[0], i18n(
"Check this if you want to indent with spaces rather than tabs."));
QWhatsThis::add(opt[2], i18n(
"Indentations of more than the selected number of spaces will not be "
"shortened."));
QWhatsThis::add(opt[3], i18n(
"This allows the <b>Tab</b> key to be used to increase the indentation "
"level."));
QWhatsThis::add(opt[4], i18n(
"This allows the <b>Backspace</b> key to be used to decrease the "
"indentation level."));
QWhatsThis::add(opt[5], i18n(
"Automatically inserts a leading \"*\" while typing within a Doxygen "
"style comment."));
QWhatsThis::add( opt[6], i18n(
"Use a mix of tab and space characters for indentation.") );
QWhatsThis::add( opt[7], i18n(
"If this option is selected, pasted code from the clipboard is indented. "
"Triggering the <b>undo</b>-action removes the indentation.") );
QWhatsThis::add(indentationWidth, i18n("The number of spaces to indent with."));
QWhatsThis::add(m_configPage, i18n(
"If this button is enabled, additional indenter specific options are "
"available and can be configured in an extra dialog.") );
reload ();
//
// after initial reload, connect the stuff for the changed () signal
//
connect(m_indentMode, SIGNAL(activated(int)), this, SLOT(slotChanged()));
connect(m_indentMode, SIGNAL(activated(int)), this, SLOT(indenterSelected(int)));
connect( opt[0], SIGNAL(toggled(bool)), this, SLOT(somethingToggled()));
connect( opt[0], SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect( opt[1], SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect( opt[2], SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect( opt[3], SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect( opt[4], SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect( opt[5], SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect( opt[6], SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect( opt[7], SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect(indentationWidth, SIGNAL(valueChanged(int)), this, SLOT(slotChanged()));
connect(rb1, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(rb2, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(rb3, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(m_configPage, SIGNAL(clicked()), this, SLOT(configPage()));
}
void KateIndentConfigTab::somethingToggled() {
indentationWidth->setEnabled(opt[0]->isChecked());
opt[6]->setEnabled(opt[0]->isChecked());
}
void KateIndentConfigTab::indenterSelected (int index)
{
if (index == KateDocumentConfig::imCStyle || index == KateDocumentConfig::imCSAndS)
opt[5]->setEnabled(true);
else
opt[5]->setEnabled(false);
m_configPage->setEnabled( KateAutoIndent::hasConfigPage(index) );
}
void KateIndentConfigTab::configPage()
{
uint index = m_indentMode->currentItem();
if ( KateAutoIndent::hasConfigPage(index) )
{
KDialogBase dlg(this, "indenter_config_dialog", true, i18n("Configure Indenter"),
KDialogBase::Ok|KDialogBase::Cancel, KDialogBase::Cancel, true);
QVBox *box = new QVBox(&dlg);
box->setSpacing( KDialog::spacingHint() );
dlg.setMainWidget(box);
new QLabel("<qt><b>" + KateAutoIndent::modeDescription(index) + "</b></qt>", box);
new KSeparator(KSeparator::HLine, box);
IndenterConfigPage* page = KateAutoIndent::configPage(box, index);
if (!page) return;
box->setStretchFactor(page, 1);
connect( &dlg, SIGNAL(okClicked()), page, SLOT(apply()) );
dlg.resize(400, 300);
dlg.exec();
}
}
void KateIndentConfigTab::apply ()
{
// nothing changed, no need to apply stuff
if (!changed())
return;
m_changed = false;
KateDocumentConfig::global()->configStart ();
int configFlags, z;
configFlags = KateDocumentConfig::global()->configFlags();
for (z = 0; z < numFlags; z++) {
configFlags &= ~flags[z];
if (opt[z]->isChecked()) configFlags |= flags[z];
}
KateDocumentConfig::global()->setConfigFlags(configFlags);
KateDocumentConfig::global()->setIndentationWidth(indentationWidth->value());
KateDocumentConfig::global()->setIndentationMode(m_indentMode->currentItem());
KateDocumentConfig::global()->setConfigFlags (KateDocumentConfig::cfTabIndentsMode, 2 == m_tabs->id (m_tabs->selected()));
KateDocumentConfig::global()->setConfigFlags (KateDocumentConfig::cfTabInsertsTab, 1 == m_tabs->id (m_tabs->selected()));
KateDocumentConfig::global()->configEnd ();
}
void KateIndentConfigTab::reload ()
{
if (KateDocumentConfig::global()->configFlags() & KateDocumentConfig::cfTabIndentsMode)
m_tabs->setButton (2);
else if (KateDocumentConfig::global()->configFlags() & KateDocumentConfig::cfTabInsertsTab)
m_tabs->setButton (1);
else
m_tabs->setButton (0);
m_indentMode->setCurrentItem (KateDocumentConfig::global()->indentationMode());
somethingToggled ();
indenterSelected (m_indentMode->currentItem());
}
//END KateIndentConfigTab
//BEGIN KateSelectConfigTab
const int KateSelectConfigTab::flags[] = {};
KateSelectConfigTab::KateSelectConfigTab(QWidget *parent)
: KateConfigPage(parent)
{
int configFlags = KateDocumentConfig::global()->configFlags();
QVBoxLayout *layout = new QVBoxLayout(this, 0, KDialog::spacingHint() );
QVGroupBox *gbCursor = new QVGroupBox(i18n("Text Cursor Movement"), this);
opt[0] = new QCheckBox(i18n("Smart ho&me and smart end"), gbCursor);
opt[0]->setChecked(configFlags & KateDocumentConfig::cfSmartHome);
connect(opt[0], SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
opt[1] = new QCheckBox(i18n("Wrap c&ursor"), gbCursor);
opt[1]->setChecked(configFlags & KateDocumentConfig::cfWrapCursor);
connect(opt[1], SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
e6 = new QCheckBox(i18n("&PageUp/PageDown moves cursor"), gbCursor);
e6->setChecked(KateDocumentConfig::global()->pageUpDownMovesCursor());
connect(e6, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
e4 = new KIntNumInput(KateViewConfig::global()->autoCenterLines(), gbCursor);
e4->setRange(0, 1000000, 1, false);
e4->setLabel(i18n("Autocenter cursor (lines):"), AlignVCenter);
connect(e4, SIGNAL(valueChanged(int)), this, SLOT(slotChanged()));
layout->addWidget(gbCursor);
QRadioButton *rb1, *rb2;
m_tabs = new QButtonGroup( 1, Qt::Horizontal, i18n("Selection Mode"), this );
layout->add (m_tabs);
m_tabs->setRadioButtonExclusive( true );
m_tabs->insert( rb1=new QRadioButton( i18n("&Normal"), m_tabs ), 0 );
m_tabs->insert( rb2=new QRadioButton( i18n("&Persistent"), m_tabs ), 1 );
layout->addStretch();
QWhatsThis::add(rb1, i18n(
"Selections will be overwritten by typed text and will be lost on "
"cursor movement."));
QWhatsThis::add(rb2, i18n(
"Selections will stay even after cursor movement and typing."));
QWhatsThis::add(e4, i18n(
"Sets the number of lines to maintain visible above and below the "
"cursor when possible."));
QWhatsThis::add(opt[0], i18n(
"When selected, pressing the home key will cause the cursor to skip "
"whitespace and go to the start of a line's text. "
"The same applies for the end key."));
QWhatsThis::add(opt[1], i18n(
"When on, moving the insertion cursor using the <b>Left</b> and "
"<b>Right</b> keys will go on to previous/next line at beginning/end of "
"the line, similar to most editors.<p>When off, the insertion cursor "
"cannot be moved left of the line start, but it can be moved off the "
"line end, which can be very handy for programmers."));
QWhatsThis::add(e6, i18n("Selects whether the PageUp and PageDown keys should alter the vertical position of the cursor relative to the top of the view."));
reload ();
//
// after initial reload, connect the stuff for the changed () signal
//
connect(rb1, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(rb2, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
}
void KateSelectConfigTab::apply ()
{
// nothing changed, no need to apply stuff
if (!changed())
return;
m_changed = false;
KateViewConfig::global()->configStart ();
KateDocumentConfig::global()->configStart ();
int configFlags = KateDocumentConfig::global()->configFlags();
configFlags &= ~KateDocumentConfig::cfSmartHome;
configFlags &= ~KateDocumentConfig::cfWrapCursor;
if (opt[0]->isChecked()) configFlags |= KateDocumentConfig::cfSmartHome;
if (opt[1]->isChecked()) configFlags |= KateDocumentConfig::cfWrapCursor;
KateDocumentConfig::global()->setConfigFlags(configFlags);
KateViewConfig::global()->setAutoCenterLines(kMax(0, e4->value()));
KateDocumentConfig::global()->setPageUpDownMovesCursor(e6->isChecked());
KateViewConfig::global()->setPersistentSelection (m_tabs->id (m_tabs->selected()) == 1);
KateDocumentConfig::global()->configEnd ();
KateViewConfig::global()->configEnd ();
}
void KateSelectConfigTab::reload ()
{
if (KateViewConfig::global()->persistentSelection())
m_tabs->setButton (1);
else
m_tabs->setButton (0);
}
//END KateSelectConfigTab
//BEGIN KateEditConfigTab
const int KateEditConfigTab::flags[] = {KateDocument::cfWordWrap,
KateDocument::cfAutoBrackets, KateDocument::cfShowTabs,
KateDocumentConfig::cfReplaceTabsDyn, KateDocumentConfig::cfRemoveTrailingDyn};
KateEditConfigTab::KateEditConfigTab(QWidget *parent)
: KateConfigPage(parent)
{
QVBoxLayout *mainLayout = new QVBoxLayout(this, 0, KDialog::spacingHint() );
int configFlags = KateDocumentConfig::global()->configFlags();
QVGroupBox *gbWhiteSpace = new QVGroupBox(i18n("Tabulators"), this);
opt[3] = new QCheckBox( i18n("&Insert spaces instead of tabulators"), gbWhiteSpace );
opt[3]->setChecked( configFlags & KateDocumentConfig::cfReplaceTabsDyn );
connect( opt[3], SIGNAL(toggled(bool)), this, SLOT(slotChanged()) );
opt[2] = new QCheckBox(i18n("&Show tabulators"), gbWhiteSpace);
opt[2]->setChecked(configFlags & flags[2]);
connect(opt[2], SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
e2 = new KIntNumInput(KateDocumentConfig::global()->tabWidth(), gbWhiteSpace);
e2->setRange(1, 16, 1, false);
e2->setLabel(i18n("Tab width:"), AlignVCenter);
connect(e2, SIGNAL(valueChanged(int)), this, SLOT(slotChanged()));
mainLayout->addWidget(gbWhiteSpace);
QVGroupBox *gbWordWrap = new QVGroupBox(i18n("Static Word Wrap"), this);
opt[0] = new QCheckBox(i18n("Enable static &word wrap"), gbWordWrap);
opt[0]->setChecked(KateDocumentConfig::global()->wordWrap());
connect(opt[0], SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
m_wwmarker = new QCheckBox( i18n("&Show static word wrap marker (if applicable)"), gbWordWrap );
m_wwmarker->setChecked( KateRendererConfig::global()->wordWrapMarker() );
connect(m_wwmarker, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
e1 = new KIntNumInput(KateDocumentConfig::global()->wordWrapAt(), gbWordWrap);
e1->setRange(20, 200, 1, false);
e1->setLabel(i18n("Wrap words at:"), AlignVCenter);
connect(e1, SIGNAL(valueChanged(int)), this, SLOT(slotChanged()));
mainLayout->addWidget(gbWordWrap);
opt[4] = new QCheckBox( i18n("Remove &trailing spaces"), this );
mainLayout->addWidget( opt[4] );
opt[4]->setChecked( configFlags & KateDocumentConfig::cfRemoveTrailingDyn );
connect( opt[4], SIGNAL(toggled(bool)), this, SLOT(slotChanged()) );
opt[1] = new QCheckBox(i18n("Auto &brackets"), this);
mainLayout->addWidget(opt[1]);
opt[1]->setChecked(configFlags & flags[1]);
connect(opt[1], SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
e3 = new KIntNumInput(e2, KateDocumentConfig::global()->undoSteps(), this);
e3->setRange(0, 1000000, 1, false);
e3->setSpecialValueText( i18n("Unlimited") );
e3->setLabel(i18n("Maximum undo steps:"), AlignVCenter);
mainLayout->addWidget(e3);
connect(e3, SIGNAL(valueChanged(int)), this, SLOT(slotChanged()));
QHBoxLayout *e5Layout = new QHBoxLayout(mainLayout);
QLabel *e5Label = new QLabel(i18n("Smart search t&ext from:"), this);
e5Layout->addWidget(e5Label);
e5 = new KComboBox (this);
e5->insertItem( i18n("Nowhere") );
e5->insertItem( i18n("Selection Only") );
e5->insertItem( i18n("Selection, then Current Word") );
e5->insertItem( i18n("Current Word Only") );
e5->insertItem( i18n("Current Word, then Selection") );
e5->setCurrentItem(KateViewConfig::global()->textToSearchMode());
e5Layout->addWidget(e5);
e5Label->setBuddy(e5);
connect(e5, SIGNAL(activated(int)), this, SLOT(slotChanged()));
mainLayout->addStretch();
// What is this? help
QWhatsThis::add(opt[0], i18n(
"Automatically start a new line of text when the current line exceeds "
"the length specified by the <b>Wrap words at:</b> option."
"<p>This option does not wrap existing lines of text - use the <b>Apply "
"Static Word Wrap</b> option in the <b>Tools</b> menu for that purpose."
"<p>If you want lines to be <i>visually wrapped</i> instead, according "
"to the width of the view, enable <b>Dynamic Word Wrap</b> in the "
"<b>View Defaults</b> config page."));
QWhatsThis::add(e1, i18n(
"If the Word Wrap option is selected this entry determines the length "
"(in characters) at which the editor will automatically start a new line."));
QWhatsThis::add(opt[1], i18n(
"When the user types a left bracket ([,(, or {) KateView automatically "
"enters the right bracket (}, ), or ]) to the right of the cursor."));
QWhatsThis::add(opt[2], i18n(
"The editor will display a symbol to indicate the presence of a tab in "
"the text."));
QWhatsThis::add(e3, i18n(
"Sets the number of undo/redo steps to record. More steps uses more memory."));
QString gstfwt = i18n(
"This determines where KateView will get the search text from "
"(this will be automatically entered into the Find Text dialog): "
"<br>"
"<ul>"
"<li><b>Nowhere:</b> Don't guess the search text."
"</li>"
"<li><b>Selection Only:</b> Use the current text selection, "
"if available."
"</li>"
"<li><b>Selection, then Current Word:</b> Use the current "
"selection if available, otherwise use the current word."
"</li>"
"<li><b>Current Word Only:</b> Use the word that the cursor "
"is currently resting on, if available."
"</li>"
"<li><b>Current Word, then Selection:</b> Use the current "
"word if available, otherwise use the current selection."
"</li>"
"</ul>"
"Note that, in all the above modes, if a search string has "
"not been or cannot be determined, then the Find Text Dialog "
"will fall back to the last search text.");
QWhatsThis::add(e5Label, gstfwt);
QWhatsThis::add(e5, gstfwt);
QWhatsThis::add( opt[3], i18n(
"If this is enabled, the editor will calculate the number of spaces up to "
"the next tab position as defined by the tab width, and insert that number "
"of spaces instead of a TAB character." ) );
QWhatsThis::add( opt[4], i18n(
"If this is enabled, the editor will remove any trailing whitespace on "
"lines when they are left by the insertion cursor.") );
QWhatsThis::add( m_wwmarker, i18n(
"<p>If this option is checked, a vertical line will be drawn at the word "
"wrap column as defined in the <strong>Editing</strong> properties."
"<p>Note that the word wrap marker is only drawn if you use a fixed "
"pitch font." ));
}
void KateEditConfigTab::apply ()
{
// nothing changed, no need to apply stuff
if (!changed())
return;
m_changed = false;
KateViewConfig::global()->configStart ();
KateDocumentConfig::global()->configStart ();
int configFlags, z;
configFlags = KateDocumentConfig::global()->configFlags();
for (z = 1; z < numFlags; z++) {
configFlags &= ~flags[z];
if (opt[z]->isChecked()) configFlags |= flags[z];
}
KateDocumentConfig::global()->setConfigFlags(configFlags);
KateDocumentConfig::global()->setWordWrapAt(e1->value());
KateDocumentConfig::global()->setWordWrap (opt[0]->isChecked());
KateDocumentConfig::global()->setTabWidth(e2->value());
if (e3->value() <= 0)
KateDocumentConfig::global()->setUndoSteps(0);
else
KateDocumentConfig::global()->setUndoSteps(e3->value());
KateViewConfig::global()->setTextToSearchMode(e5->currentItem());
KateRendererConfig::global()->setWordWrapMarker (m_wwmarker->isChecked());
KateDocumentConfig::global()->configEnd ();
KateViewConfig::global()->configEnd ();
}
void KateEditConfigTab::reload ()
{
}
//END KateEditConfigTab
//BEGIN KateViewDefaultsConfig
KateViewDefaultsConfig::KateViewDefaultsConfig(QWidget *parent)
:KateConfigPage(parent)
{
QRadioButton *rb1;
QRadioButton *rb2;
QVBoxLayout *blay=new QVBoxLayout(this,0,KDialog::spacingHint());
QVGroupBox *gbWordWrap = new QVGroupBox(i18n("Word Wrap"), this);
m_dynwrap=new QCheckBox(i18n("&Dynamic word wrap"),gbWordWrap);
QHBox *m_dynwrapIndicatorsLay = new QHBox (gbWordWrap);
m_dynwrapIndicatorsLabel = new QLabel( i18n("Dynamic word wrap indicators (if applicable):"), m_dynwrapIndicatorsLay );
m_dynwrapIndicatorsCombo = new KComboBox( m_dynwrapIndicatorsLay );
m_dynwrapIndicatorsCombo->insertItem( i18n("Off") );
m_dynwrapIndicatorsCombo->insertItem( i18n("Follow Line Numbers") );
m_dynwrapIndicatorsCombo->insertItem( i18n("Always On") );
m_dynwrapIndicatorsLabel->setBuddy(m_dynwrapIndicatorsCombo);
m_dynwrapAlignLevel = new KIntNumInput(gbWordWrap);
m_dynwrapAlignLevel->setLabel(i18n("Vertically align dynamically wrapped lines to indentation depth:"));
m_dynwrapAlignLevel->setRange(0, 80, 10);
// xgettext:no-c-format
m_dynwrapAlignLevel->setSuffix(i18n("% of View Width"));
m_dynwrapAlignLevel->setSpecialValueText(i18n("Disabled"));
blay->addWidget(gbWordWrap);
QVGroupBox *gbFold = new QVGroupBox(i18n("Code Folding"), this);
m_folding=new QCheckBox(i18n("Show &folding markers (if available)"), gbFold );
m_collapseTopLevel = new QCheckBox( i18n("Collapse toplevel folding nodes"), gbFold );
m_collapseTopLevel->hide ();
blay->addWidget(gbFold);
QVGroupBox *gbBar = new QVGroupBox(i18n("Borders"), this);
m_icons=new QCheckBox(i18n("Show &icon border"),gbBar);
m_line=new QCheckBox(i18n("Show &line numbers"),gbBar);
m_scrollBarMarks=new QCheckBox(i18n("Show &scrollbar marks"),gbBar);
blay->addWidget(gbBar);
m_bmSort = new QButtonGroup( 1, Qt::Horizontal, i18n("Sort Bookmarks Menu"), this );
m_bmSort->setRadioButtonExclusive( true );
m_bmSort->insert( rb1=new QRadioButton( i18n("By &position"), m_bmSort ), 0 );
m_bmSort->insert( rb2=new QRadioButton( i18n("By c&reation"), m_bmSort ), 1 );
blay->addWidget(m_bmSort, 0 );
m_showIndentLines = new QCheckBox(i18n("Show indentation lines"), this);
m_showIndentLines->setChecked(KateRendererConfig::global()->showIndentationLines());
blay->addWidget(m_showIndentLines);
blay->addStretch(1000);
QWhatsThis::add(m_dynwrap,i18n(
"If this option is checked, the text lines will be wrapped at the view "
"border on the screen."));
QString wtstr = i18n("Choose when the Dynamic Word Wrap Indicators should be displayed");
QWhatsThis::add(m_dynwrapIndicatorsLabel, wtstr);
QWhatsThis::add(m_dynwrapIndicatorsCombo, wtstr);
// xgettext:no-c-format
QWhatsThis::add(m_dynwrapAlignLevel, i18n(
"<p>Enables the start of dynamically wrapped lines to be aligned "
"vertically to the indentation level of the first line. This can help "
"to make code and markup more readable.</p><p>Additionally, this allows "
"you to set a maximum width of the screen, as a percentage, after which "
"dynamically wrapped lines will no longer be vertically aligned. For "
"example, at 50%, lines whose indentation levels are deeper than 50% of "
"the width of the screen will not have vertical alignment applied to "
"subsequent wrapped lines.</p>"));
QWhatsThis::add(m_line,i18n(
"If this option is checked, every new view will display line numbers "
"on the left hand side."));
QWhatsThis::add(m_icons,i18n(
"If this option is checked, every new view will display an icon border "
"on the left hand side.<br><br>The icon border shows bookmark signs, "
"for instance."));
QWhatsThis::add(m_scrollBarMarks,i18n(
"If this option is checked, every new view will show marks on the "
"vertical scrollbar.<br><br>These marks will, for instance, show "
"bookmarks."));
QWhatsThis::add(m_folding,i18n(
"If this option is checked, every new view will display marks for code "
"folding, if code folding is available."));
QWhatsThis::add(m_bmSort,i18n(
"Choose how the bookmarks should be ordered in the <b>Bookmarks</b> menu."));
QWhatsThis::add(rb1,i18n(
"The bookmarks will be ordered by the line numbers they are placed at."));
QWhatsThis::add(rb2,i18n(
"Each new bookmark will be added to the bottom, independently from "
"where it is placed in the document."));
QWhatsThis::add(m_showIndentLines, i18n(
"If this is enabled, the editor will display vertical lines to help "
"identify indent lines.") );
reload();
//
// after initial reload, connect the stuff for the changed () signal
//
connect(m_dynwrap, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(m_dynwrapIndicatorsCombo, SIGNAL(activated(int)), this, SLOT(slotChanged()));
connect(m_dynwrapAlignLevel, SIGNAL(valueChanged(int)), this, SLOT(slotChanged()));
connect(m_icons, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(m_scrollBarMarks, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(m_line, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(m_folding, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(m_collapseTopLevel, SIGNAL(toggled(bool)), this, SLOT(slotChanged()) );
connect(rb1, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(rb2, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect(m_showIndentLines, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
}
KateViewDefaultsConfig::~KateViewDefaultsConfig()
{
}
void KateViewDefaultsConfig::apply ()
{
// nothing changed, no need to apply stuff
if (!changed())
return;
m_changed = false;
KateViewConfig::global()->configStart ();
KateRendererConfig::global()->configStart ();
KateViewConfig::global()->setDynWordWrap (m_dynwrap->isChecked());
KateViewConfig::global()->setDynWordWrapIndicators (m_dynwrapIndicatorsCombo->currentItem ());
KateViewConfig::global()->setDynWordWrapAlignIndent(m_dynwrapAlignLevel->value());
KateViewConfig::global()->setLineNumbers (m_line->isChecked());
KateViewConfig::global()->setIconBar (m_icons->isChecked());
KateViewConfig::global()->setScrollBarMarks (m_scrollBarMarks->isChecked());
KateViewConfig::global()->setFoldingBar (m_folding->isChecked());
KateViewConfig::global()->setBookmarkSort (m_bmSort->id (m_bmSort->selected()));
KateRendererConfig::global()->setShowIndentationLines(m_showIndentLines->isChecked());
KateRendererConfig::global()->configEnd ();
KateViewConfig::global()->configEnd ();
}
void KateViewDefaultsConfig::reload ()
{
m_dynwrap->setChecked(KateViewConfig::global()->dynWordWrap());
m_dynwrapIndicatorsCombo->setCurrentItem( KateViewConfig::global()->dynWordWrapIndicators() );
m_dynwrapAlignLevel->setValue(KateViewConfig::global()->dynWordWrapAlignIndent());
m_line->setChecked(KateViewConfig::global()->lineNumbers());
m_icons->setChecked(KateViewConfig::global()->iconBar());
m_scrollBarMarks->setChecked(KateViewConfig::global()->scrollBarMarks());
m_folding->setChecked(KateViewConfig::global()->foldingBar());
m_bmSort->setButton( KateViewConfig::global()->bookmarkSort() );
m_showIndentLines->setChecked(KateRendererConfig::global()->showIndentationLines());
}
void KateViewDefaultsConfig::reset () {;}
void KateViewDefaultsConfig::defaults (){;}
//END KateViewDefaultsConfig
//BEGIN KateEditKeyConfiguration
KateEditKeyConfiguration::KateEditKeyConfiguration( QWidget* parent, KateDocument* doc )
: KateConfigPage( parent )
{
m_doc = doc;
m_ready = false;
}
void KateEditKeyConfiguration::showEvent ( QShowEvent * )
{
if (!m_ready)
{
(new QVBoxLayout(this))->setAutoAdd(true);
KateView* view = (KateView*)m_doc->views().at(0);
m_ac = view->editActionCollection();
m_keyChooser = new KKeyChooser( m_ac, this, false );
connect( m_keyChooser, SIGNAL( keyChange() ), this, SLOT( slotChanged() ) );
m_keyChooser->show ();
m_ready = true;
}
QWidget::show ();
}
void KateEditKeyConfiguration::apply()
{
if ( ! changed() )
return;
m_changed = false;
if (m_ready)
{
m_keyChooser->commitChanges();
m_ac->writeShortcutSettings( "Katepart Shortcuts" );
}
}
//END KateEditKeyConfiguration
//BEGIN KateSaveConfigTab
KateSaveConfigTab::KateSaveConfigTab( QWidget *parent )
: KateConfigPage( parent )
{
int configFlags = KateDocumentConfig::global()->configFlags();
QVBoxLayout *layout = new QVBoxLayout(this, 0, KDialog::spacingHint() );
QVGroupBox *gbEnc = new QVGroupBox(i18n("File Format"), this);
layout->addWidget( gbEnc );
QHBox *e5Layout = new QHBox(gbEnc);
QLabel *e5Label = new QLabel(i18n("&Encoding:"), e5Layout);
m_encoding = new KComboBox (e5Layout);
e5Label->setBuddy(m_encoding);
e5Layout = new QHBox(gbEnc);
e5Label = new QLabel(i18n("End &of line:"), e5Layout);
m_eol = new KComboBox (e5Layout);
e5Label->setBuddy(m_eol);
allowEolDetection = new QCheckBox(i18n("&Automatic end of line detection"), gbEnc);
m_eol->insertItem (i18n("UNIX"));
m_eol->insertItem (i18n("DOS/Windows"));
m_eol->insertItem (i18n("Macintosh"));
QVGroupBox *gbMem = new QVGroupBox(i18n("Memory Usage"), this);
layout->addWidget( gbMem );
e5Layout = new QHBox(gbMem);
e5Layout->setSpacing (32);
blockCountLabel = new QLabel(i18n("Maximum loaded &blocks per file:"), e5Layout);
blockCount = new QSpinBox (4, 512, 4, e5Layout);
blockCount->setValue (KateBuffer::maxLoadedBlocks());
blockCountLabel->setBuddy(blockCount);
QVGroupBox *gbWhiteSpace = new QVGroupBox(i18n("Automatic Cleanups on Load/Save"), this);
layout->addWidget( gbWhiteSpace );
removeSpaces = new QCheckBox(i18n("Re&move trailing spaces"), gbWhiteSpace);
removeSpaces->setChecked(configFlags & KateDocument::cfRemoveSpaces);
QVGroupBox *dirConfigBox = new QVGroupBox(i18n("Folder Config File"), this);
layout->addWidget( dirConfigBox );
dirSearchDepth = new KIntNumInput(KateDocumentConfig::global()->searchDirConfigDepth(), dirConfigBox);
dirSearchDepth->setRange(-1, 64, 1, false);
dirSearchDepth->setSpecialValueText( i18n("Do not use config file") );
dirSearchDepth->setLabel(i18n("Se&arch depth for config file:"), AlignVCenter);
QGroupBox *gb = new QGroupBox( 1, Qt::Horizontal, i18n("Backup on Save"), this );
layout->addWidget( gb );
cbLocalFiles = new QCheckBox( i18n("&Local files"), gb );
cbRemoteFiles = new QCheckBox( i18n("&Remote files"), gb );
QHBox *hbBuPrefix = new QHBox( gb );
QLabel *lBuPrefix = new QLabel( i18n("&Prefix:"), hbBuPrefix );
leBuPrefix = new QLineEdit( hbBuPrefix );
lBuPrefix->setBuddy( leBuPrefix );
QHBox *hbBuSuffix = new QHBox( gb );
QLabel *lBuSuffix = new QLabel( i18n("&Suffix:"), hbBuSuffix );
leBuSuffix = new QLineEdit( hbBuSuffix );
lBuSuffix->setBuddy( leBuSuffix );
layout->addStretch();
QWhatsThis::add(removeSpaces, i18n(
"The editor will automatically eliminate extra spaces at the ends of "
"lines of text while loading/saving the file."));
QWhatsThis::add( gb, i18n(
"<p>Backing up on save will cause Kate to copy the disk file to "
"'&lt;prefix&gt;&lt;filename&gt;&lt;suffix&gt;' before saving changes."
"<p>The suffix defaults to <strong>~</strong> and prefix is empty by default" ) );
QWhatsThis::add( allowEolDetection, i18n(
"Check this if you want the editor to autodetect the end of line type."
"The first found end of line type will be used for the whole file.") );
QWhatsThis::add( cbLocalFiles, i18n(
"Check this if you want backups of local files when saving") );
QWhatsThis::add( cbRemoteFiles, i18n(
"Check this if you want backups of remote files when saving") );
QWhatsThis::add( leBuPrefix, i18n(
"Enter the prefix to prepend to the backup file names" ) );
QWhatsThis::add( leBuSuffix, i18n(
"Enter the suffix to add to the backup file names" ) );
QWhatsThis::add(dirSearchDepth, i18n(
"The editor will search the given number of folder levels upwards for .kateconfig file"
" and load the settings line from it." ));
QWhatsThis::add(blockCount, i18n(
"The editor will load given number of blocks (of around 2048 lines) of text into memory;"
" if the filesize is bigger than this the other blocks are swapped "
" to disk and loaded transparently as-needed.<br>"
" This can cause little delays while navigating in the document; a larger block count"
" increases the editing speed at the cost of memory. <br>For normal usage, just choose the highest possible"
" block count: limit it only if you have problems with the memory usage."));
reload();
//
// after initial reload, connect the stuff for the changed () signal
//
connect(m_encoding, SIGNAL(activated(int)), this, SLOT(slotChanged()));
connect(m_eol, SIGNAL(activated(int)), this, SLOT(slotChanged()));
connect( allowEolDetection, SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect(blockCount, SIGNAL(valueChanged(int)), this, SLOT(slotChanged()));
connect(removeSpaces, SIGNAL(toggled(bool)), this, SLOT(slotChanged()));
connect( cbLocalFiles, SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect( cbRemoteFiles, SIGNAL( toggled(bool) ), this, SLOT( slotChanged() ) );
connect(dirSearchDepth, SIGNAL(valueChanged(int)), this, SLOT(slotChanged()));
connect( leBuPrefix, SIGNAL( textChanged ( const QString & ) ), this, SLOT( slotChanged() ) );
connect( leBuSuffix, SIGNAL( textChanged ( const QString & ) ), this, SLOT( slotChanged() ) );
}
void KateSaveConfigTab::apply()
{
// nothing changed, no need to apply stuff
if (!changed())
return;
m_changed = false;
KateBuffer::setMaxLoadedBlocks (blockCount->value());
KateDocumentConfig::global()->configStart ();
if ( leBuSuffix->text().isEmpty() && leBuPrefix->text().isEmpty() ) {
KMessageBox::information(
this,
i18n("You did not provide a backup suffix or prefix. Using default suffix: '~'"),
i18n("No Backup Suffix or Prefix")
);
leBuSuffix->setText( "~" );
}
uint f( 0 );
if ( cbLocalFiles->isChecked() )
f |= KateDocumentConfig::LocalFiles;
if ( cbRemoteFiles->isChecked() )
f |= KateDocumentConfig::RemoteFiles;
KateDocumentConfig::global()->setBackupFlags(f);
KateDocumentConfig::global()->setBackupPrefix(leBuPrefix->text());
KateDocumentConfig::global()->setBackupSuffix(leBuSuffix->text());
KateDocumentConfig::global()->setSearchDirConfigDepth(dirSearchDepth->value());
int configFlags = KateDocumentConfig::global()->configFlags();
configFlags &= ~KateDocument::cfRemoveSpaces; // clear flag
if (removeSpaces->isChecked()) configFlags |= KateDocument::cfRemoveSpaces; // set flag if checked
KateDocumentConfig::global()->setConfigFlags(configFlags);
KateDocumentConfig::global()->setEncoding((m_encoding->currentItem() == 0) ? "" : KGlobal::charsets()->encodingForName(m_encoding->currentText()));
KateDocumentConfig::global()->setEol(m_eol->currentItem());
KateDocumentConfig::global()->setAllowEolDetection(allowEolDetection->isChecked());
KateDocumentConfig::global()->configEnd ();
}
void KateSaveConfigTab::reload()
{
// encoding
m_encoding->clear ();
m_encoding->insertItem (i18n("KDE Default"));
m_encoding->setCurrentItem(0);
QStringList encodings (KGlobal::charsets()->descriptiveEncodingNames());
int insert = 1;
for (uint i=0; i < encodings.count(); i++)
{
bool found = false;
QTextCodec *codecForEnc = KGlobal::charsets()->codecForName(KGlobal::charsets()->encodingForName(encodings[i]), found);
if (found)
{
m_encoding->insertItem (encodings[i]);
if ( codecForEnc->name() == KateDocumentConfig::global()->encoding() )
{
m_encoding->setCurrentItem(insert);
}
insert++;
}
}
// eol
m_eol->setCurrentItem(KateDocumentConfig::global()->eol());
allowEolDetection->setChecked(KateDocumentConfig::global()->allowEolDetection());
dirSearchDepth->setValue(KateDocumentConfig::global()->searchDirConfigDepth());
// other stuff
uint f ( KateDocumentConfig::global()->backupFlags() );
cbLocalFiles->setChecked( f & KateDocumentConfig::LocalFiles );
cbRemoteFiles->setChecked( f & KateDocumentConfig::RemoteFiles );
leBuPrefix->setText( KateDocumentConfig::global()->backupPrefix() );
leBuSuffix->setText( KateDocumentConfig::global()->backupSuffix() );
}
void KateSaveConfigTab::reset()
{
}
void KateSaveConfigTab::defaults()
{
cbLocalFiles->setChecked( true );
cbRemoteFiles->setChecked( false );
leBuPrefix->setText( "" );
leBuSuffix->setText( "~" );
}
//END KateSaveConfigTab
//BEGIN PluginListItem
class KatePartPluginListItem : public QCheckListItem
{
public:
KatePartPluginListItem(bool checked, uint i, const QString &name, QListView *parent);
uint pluginIndex () const { return index; }
protected:
void stateChange(bool);
private:
uint index;
bool silentStateChange;
};
KatePartPluginListItem::KatePartPluginListItem(bool checked, uint i, const QString &name, QListView *parent)
: QCheckListItem(parent, name, CheckBox)
, index(i)
, silentStateChange(false)
{
silentStateChange = true;
setOn(checked);
silentStateChange = false;
}
void KatePartPluginListItem::stateChange(bool b)
{
if(!silentStateChange)
static_cast<KatePartPluginListView *>(listView())->stateChanged(this, b);
}
//END
//BEGIN PluginListView
KatePartPluginListView::KatePartPluginListView(QWidget *parent, const char *name)
: KListView(parent, name)
{
}
void KatePartPluginListView::stateChanged(KatePartPluginListItem *item, bool b)
{
emit stateChange(item, b);
}
//END
//BEGIN KatePartPluginConfigPage
KatePartPluginConfigPage::KatePartPluginConfigPage (QWidget *parent) : KateConfigPage (parent, "")
{
// sizemanagment
QGridLayout *grid = new QGridLayout( this, 1, 1 );
grid->setSpacing( KDialogBase::spacingHint() );
listView = new KatePartPluginListView(this);
listView->addColumn(i18n("Name"));
listView->addColumn(i18n("Comment"));
grid->addWidget( listView, 0, 0);
for (uint i=0; i<KateFactory::self()->plugins().count(); i++)
{
KatePartPluginListItem *item = new KatePartPluginListItem(KateDocumentConfig::global()->plugin(i), i, (KateFactory::self()->plugins())[i]->name(), listView);
item->setText(0, (KateFactory::self()->plugins())[i]->name());
item->setText(1, (KateFactory::self()->plugins())[i]->comment());
m_items.append (item);
}
// configure button
btnConfigure = new QPushButton( i18n("Configure..."), this );
btnConfigure->setEnabled( false );
grid->addWidget( btnConfigure, 1, 0, Qt::AlignRight );
connect( btnConfigure, SIGNAL(clicked()), this, SLOT(slotConfigure()) );
connect( listView, SIGNAL(selectionChanged(QListViewItem*)), this, SLOT(slotCurrentChanged(QListViewItem*)) );
connect( listView, SIGNAL(stateChange(KatePartPluginListItem *, bool)),
this, SLOT(slotStateChanged(KatePartPluginListItem *, bool)));
connect(listView, SIGNAL(stateChange(KatePartPluginListItem *, bool)), this, SLOT(slotChanged()));
}
KatePartPluginConfigPage::~KatePartPluginConfigPage ()
{
}
void KatePartPluginConfigPage::apply ()
{
// nothing changed, no need to apply stuff
if (!changed())
return;
m_changed = false;
KateDocumentConfig::global()->configStart ();
for (uint i=0; i < m_items.count(); i++)
KateDocumentConfig::global()->setPlugin (m_items.at(i)->pluginIndex(), m_items.at(i)->isOn());
KateDocumentConfig::global()->configEnd ();
}
void KatePartPluginConfigPage::slotStateChanged( KatePartPluginListItem *item, bool b )
{
if ( b )
slotCurrentChanged( (QListViewItem*)item );
}
void KatePartPluginConfigPage::slotCurrentChanged( QListViewItem* i )
{
KatePartPluginListItem *item = static_cast<KatePartPluginListItem *>(i);
if ( ! item ) return;
bool b = false;
if ( item->isOn() )
{
// load this plugin, and see if it has config pages
KTextEditor::Plugin *plugin = KTextEditor::createPlugin(QFile::encodeName((KateFactory::self()->plugins())[item->pluginIndex()]->library()));
if ( plugin ) {
KTextEditor::ConfigInterfaceExtension *cie = KTextEditor::configInterfaceExtension( plugin );
b = ( cie && cie->configPages() );
}
}
btnConfigure->setEnabled( b );
}
void KatePartPluginConfigPage::slotConfigure()
{
KatePartPluginListItem *item = static_cast<KatePartPluginListItem*>(listView->currentItem());
KTextEditor::Plugin *plugin =
KTextEditor::createPlugin(QFile::encodeName((KateFactory::self()->plugins())[item->pluginIndex()]->library()));
if ( ! plugin ) return;
KTextEditor::ConfigInterfaceExtension *cife =
KTextEditor::configInterfaceExtension( plugin );
if ( ! cife )
return;
if ( ! cife->configPages() )
return;
// If we have only one page, we use a simple dialog, else an icon list type
KDialogBase::DialogType dt =
cife->configPages() > 1 ?
KDialogBase::IconList : // still untested
KDialogBase::Plain;
QString name = (KateFactory::self()->plugins())[item->pluginIndex()]->name();
KDialogBase *kd = new KDialogBase ( dt,
i18n("Configure %1").arg( name ),
KDialogBase::Ok | KDialogBase::Cancel | KDialogBase::Help,
KDialogBase::Ok,
this );
QPtrList<KTextEditor::ConfigPage> editorPages;
for (uint i = 0; i < cife->configPages (); i++)
{
QWidget *page;
if ( dt == KDialogBase::IconList )
{
QStringList path;
path.clear();
path << cife->configPageName( i );
page = kd->addVBoxPage( path, cife->configPageFullName (i),
cife->configPagePixmap(i, KIcon::SizeMedium) );
}
else
{
page = kd->plainPage();
QVBoxLayout *_l = new QVBoxLayout( page );
_l->setAutoAdd( true );
}
editorPages.append( cife->configPage( i, page ) );
}
if (kd->exec())
{
for( uint i=0; i<editorPages.count(); i++ )
{
editorPages.at( i )->apply();
}
}
delete kd;
}
//END KatePartPluginConfigPage
//BEGIN KateHlConfigPage
KateHlConfigPage::KateHlConfigPage (QWidget *parent, KateDocument *doc)
: KateConfigPage (parent, "")
, hlData (0)
, m_doc (doc)
{
QVBoxLayout *layout = new QVBoxLayout(this, 0, KDialog::spacingHint() );
// hl chooser
QHBox *hbHl = new QHBox( this );
layout->add (hbHl);
hbHl->setSpacing( KDialog::spacingHint() );
QLabel *lHl = new QLabel( i18n("H&ighlight:"), hbHl );
hlCombo = new QComboBox( false, hbHl );
lHl->setBuddy( hlCombo );
connect( hlCombo, SIGNAL(activated(int)),
this, SLOT(hlChanged(int)) );
for( int i = 0; i < KateHlManager::self()->highlights(); i++) {
if (KateHlManager::self()->hlSection(i).length() > 0)
hlCombo->insertItem(KateHlManager::self()->hlSection(i) + QString ("/") + KateHlManager::self()->hlNameTranslated(i));
else
hlCombo->insertItem(KateHlManager::self()->hlNameTranslated(i));
}
QGroupBox *gbInfo = new QGroupBox( 1, Qt::Horizontal, i18n("Information"), this );
layout->add (gbInfo);
// author
QHBox *hb1 = new QHBox( gbInfo);
new QLabel( i18n("Author:"), hb1 );
author = new QLabel (hb1);
author->setTextFormat (Qt::RichText);
// license
QHBox *hb2 = new QHBox( gbInfo);
new QLabel( i18n("License:"), hb2 );
license = new QLabel (hb2);
QGroupBox *gbProps = new QGroupBox( 1, Qt::Horizontal, i18n("Properties"), this );
layout->add (gbProps);
// file & mime types
QHBox *hbFE = new QHBox( gbProps);
QLabel *lFileExts = new QLabel( i18n("File e&xtensions:"), hbFE );
wildcards = new QLineEdit( hbFE );
lFileExts->setBuddy( wildcards );
QHBox *hbMT = new QHBox( gbProps );
QLabel *lMimeTypes = new QLabel( i18n("MIME &types:"), hbMT);
mimetypes = new QLineEdit( hbMT );
lMimeTypes->setBuddy( mimetypes );
QHBox *hbMT2 = new QHBox( gbProps );
QLabel *lprio = new QLabel( i18n("Prio&rity:"), hbMT2);
priority = new KIntNumInput( hbMT2 );
lprio->setBuddy( priority );
QToolButton *btnMTW = new QToolButton(hbMT);
btnMTW->setIconSet(QIconSet(SmallIcon("wizard")));
connect(btnMTW, SIGNAL(clicked()), this, SLOT(showMTDlg()));
// download/new buttons
QHBox *hbBtns = new QHBox( this );
layout->add (hbBtns);
((QBoxLayout*)hbBtns->layout())->addStretch(1); // hmm.
hbBtns->setSpacing( KDialog::spacingHint() );
QPushButton *btnDl = new QPushButton(i18n("Do&wnload..."), hbBtns);
connect( btnDl, SIGNAL(clicked()), this, SLOT(hlDownload()) );
int currentHl = m_doc ? m_doc->hlMode() : 0;
hlCombo->setCurrentItem( currentHl );
hlChanged( currentHl );
QWhatsThis::add( hlCombo, i18n(
"Choose a <em>Syntax Highlight mode</em> from this list to view its "
"properties below.") );
QWhatsThis::add( wildcards, i18n(
"The list of file extensions used to determine which files to highlight "
"using the current syntax highlight mode.") );
QWhatsThis::add( mimetypes, i18n(
"The list of Mime Types used to determine which files to highlight "
"using the current highlight mode.<p>Click the wizard button on the "
"left of the entry field to display the MimeType selection dialog.") );
QWhatsThis::add( btnMTW, i18n(
"Display a dialog with a list of all available mime types to choose from."
"<p>The <strong>File Extensions</strong> entry will automatically be "
"edited as well.") );
QWhatsThis::add( btnDl, i18n(
"Click this button to download new or updated syntax highlight "
"descriptions from the Kate website.") );
layout->addStretch ();
connect( wildcards, SIGNAL( textChanged ( const QString & ) ), this, SLOT( slotChanged() ) );
connect( mimetypes, SIGNAL( textChanged ( const QString & ) ), this, SLOT( slotChanged() ) );
connect( priority, SIGNAL( valueChanged ( int ) ), this, SLOT( slotChanged() ) );
}
KateHlConfigPage::~KateHlConfigPage ()
{
}
void KateHlConfigPage::apply ()
{
// nothing changed, no need to apply stuff
if (!changed())
return;
m_changed = false;
writeback();
for ( QIntDictIterator<KateHlData> it( hlDataDict ); it.current(); ++it )
KateHlManager::self()->getHl( it.currentKey() )->setData( it.current() );
KateHlManager::self()->getKConfig()->sync ();
}
void KateHlConfigPage::reload ()
{
}
void KateHlConfigPage::hlChanged(int z)
{
writeback();
KateHighlighting *hl = KateHlManager::self()->getHl( z );
if (!hl)
{
hlData = 0;
return;
}
if ( !hlDataDict.find( z ) )
hlDataDict.insert( z, hl->getData() );
hlData = hlDataDict.find( z );
wildcards->setText(hlData->wildcards);
mimetypes->setText(hlData->mimetypes);
priority->setValue(hlData->priority);
// split author string if needed into multiple lines !
QStringList l= QStringList::split (QRegExp("[,;]"), hl->author());
author->setText (l.join ("<br>"));
license->setText (hl->license());
}
void KateHlConfigPage::writeback()
{
if (hlData)
{
hlData->wildcards = wildcards->text();
hlData->mimetypes = mimetypes->text();
hlData->priority = priority->value();
}
}
void KateHlConfigPage::hlDownload()
{
KateHlDownloadDialog diag(this,"hlDownload",true);
diag.exec();
}
void KateHlConfigPage::showMTDlg()
{
QString text = i18n("Select the MimeTypes you want highlighted using the '%1' syntax highlight rules.\nPlease note that this will automatically edit the associated file extensions as well.").arg( hlCombo->currentText() );
QStringList list = QStringList::split( QRegExp("\\s*;\\s*"), mimetypes->text() );
KMimeTypeChooserDialog d( i18n("Select Mime Types"), text, list, "text", this );
if ( d.exec() == KDialogBase::Accepted ) {
// do some checking, warn user if mime types or patterns are removed.
// if the lists are empty, and the fields not, warn.
wildcards->setText(d.chooser()->patterns().join(";"));
mimetypes->setText(d.chooser()->mimeTypes().join(";"));
}
}
//END KateHlConfigPage
//BEGIN KateHlDownloadDialog
KateHlDownloadDialog::KateHlDownloadDialog(QWidget *parent, const char *name, bool modal)
:KDialogBase(KDialogBase::Swallow, i18n("Highlight Download"), User1|Close, User1, parent, name, modal, true, i18n("&Install"))
{
QVBox* vbox = new QVBox(this);
setMainWidget(vbox);
vbox->setSpacing(spacingHint());
new QLabel(i18n("Select the syntax highlighting files you want to update:"), vbox);
list = new QListView(vbox);
list->addColumn("");
list->addColumn(i18n("Name"));
list->addColumn(i18n("Installed"));
list->addColumn(i18n("Latest"));
list->setSelectionMode(QListView::Multi);
list->setAllColumnsShowFocus(true);
new QLabel(i18n("<b>Note:</b> New versions are selected automatically."), vbox);
actionButton (User1)->setIconSet(SmallIconSet("ok"));
transferJob = KIO::get(
KURL(QString(HLDOWNLOADPATH)
+ QString("update-")
+ QString(KATEPART_VERSION)
+ QString(".xml")), true, true );
connect(transferJob, SIGNAL(data(KIO::Job *, const QByteArray &)),
this, SLOT(listDataReceived(KIO::Job *, const QByteArray &)));
// void data( KIO::Job *, const QByteArray &data);
resize(450, 400);
}
KateHlDownloadDialog::~KateHlDownloadDialog(){}
void KateHlDownloadDialog::listDataReceived(KIO::Job *, const QByteArray &data)
{
if (!transferJob || transferJob->isErrorPage())
{
actionButton(User1)->setEnabled(false);
return;
}
listData+=QString(data);
kdDebug(13000)<<QString("CurrentListData: ")<<listData<<endl<<endl;
kdDebug(13000)<<QString("Data length: %1").arg(data.size())<<endl;
kdDebug(13000)<<QString("listData length: %1").arg(listData.length())<<endl;
if (data.size()==0)
{
if (listData.length()>0)
{
QString installedVersion;
KateHlManager *hlm=KateHlManager::self();
QDomDocument doc;
doc.setContent(listData);
QDomElement DocElem=doc.documentElement();
QDomNode n=DocElem.firstChild();
KateHighlighting *hl = 0;
if (n.isNull()) kdDebug(13000)<<"There is no usable childnode"<<endl;
while (!n.isNull())
{
installedVersion=" --";
QDomElement e=n.toElement();
if (!e.isNull())
kdDebug(13000)<<QString("NAME: ")<<e.tagName()<<QString(" - ")<<e.attribute("name")<<endl;
n=n.nextSibling();
QString Name=e.attribute("name");
for (int i=0;i<hlm->highlights();i++)
{
hl=hlm->getHl(i);
if (hl && hl->name()==Name)
{
installedVersion=" "+hl->version();
break;
}
else hl = 0;
}
// autoselect entry if new or updated.
QListViewItem* entry = new QListViewItem(
list, "", e.attribute("name"), installedVersion,
e.attribute("version"),e.attribute("url"));
if (!hl || hl->version() < e.attribute("version"))
{
entry->setSelected(true);
entry->setPixmap(0, SmallIcon(("knewstuff")));
}
}
}
}
}
void KateHlDownloadDialog::slotUser1()
{
QString destdir=KGlobal::dirs()->saveLocation("data","katepart/syntax/");
for (QListViewItem *it=list->firstChild();it;it=it->nextSibling())
{
if (list->isSelected(it))
{
KURL src(it->text(4));
QString filename=src.fileName(false);
QString dest = destdir+filename;
KIO::NetAccess::download(src,dest, this);
}
}
// update Config !!
KateSyntaxDocument doc (true);
}
//END KateHlDownloadDialog
//BEGIN KateGotoLineDialog
KateGotoLineDialog::KateGotoLineDialog(QWidget *parent, int line, int max)
: KDialogBase(parent, 0L, true, i18n("Go to Line"), Ok | Cancel, Ok) {
QWidget *page = new QWidget(this);
setMainWidget(page);
QVBoxLayout *topLayout = new QVBoxLayout( page, 0, spacingHint() );
e1 = new KIntNumInput(line, page);
e1->setRange(1, max);
e1->setEditFocus(true);
QLabel *label = new QLabel( e1,i18n("&Go to line:"), page );
topLayout->addWidget(label);
topLayout->addWidget(e1);
topLayout->addSpacing(spacingHint()); // A little bit extra space
topLayout->addStretch(10);
e1->setFocus();
}
int KateGotoLineDialog::getLine() {
return e1->value();
}
//END KateGotoLineDialog
//BEGIN KateModOnHdPrompt
KateModOnHdPrompt::KateModOnHdPrompt( KateDocument *doc,
int modtype,
const QString &reason,
QWidget *parent )
: KDialogBase( parent, "", true, "", Ok|Apply|Cancel|User1 ),
m_doc( doc ),
m_modtype ( modtype ),
m_tmpfile( 0 )
{
QString title, btnOK, whatisok;
if ( modtype == 3 ) // deleted
{
title = i18n("File Was Deleted on Disk");
btnOK = i18n("&Save File As...");
whatisok = i18n("Lets you select a location and save the file again.");
} else {
title = i18n("File Changed on Disk");
btnOK = i18n("&Reload File");
whatisok = i18n("Reload the file from disk. If you have unsaved changes, "
"they will be lost.");
}
setButtonText( Ok, btnOK);
setButtonText( Apply, i18n("&Ignore") );
setButtonWhatsThis( Ok, whatisok );
setButtonWhatsThis( Apply, i18n("Ignore the changes. You will not be prompted again.") );
setButtonWhatsThis( Cancel, i18n("Do nothing. Next time you focus the file, "
"or try to save it or close it, you will be prompted again.") );
enableButtonSeparator( true );
setCaption( title );
QFrame *w = makeMainWidget();
QVBoxLayout *lo = new QVBoxLayout( w );
QHBoxLayout *lo1 = new QHBoxLayout( lo );
QLabel *icon = new QLabel( w );
icon->setPixmap( DesktopIcon("messagebox_warning" ) );
lo1->addWidget( icon );
lo1->addWidget( new QLabel( reason + "\n\n" + i18n("What do you want to do?"), w ) );
// If the file isn't deleted, present a diff button, and a overwrite action.
if ( modtype != 3 )
{
QHBoxLayout *lo2 = new QHBoxLayout( lo );
QPushButton *btnDiff = new QPushButton( i18n("&View Difference"), w );
lo2->addStretch( 1 );
lo2->addWidget( btnDiff );
connect( btnDiff, SIGNAL(clicked()), this, SLOT(slotDiff()) );
QWhatsThis::add( btnDiff, i18n(
"Calculates the difference between the editor contents and the disk "
"file using diff(1) and opens the diff file with the default application "
"for that.") );
setButtonText( User1, i18n("Overwrite") );
setButtonWhatsThis( User1, i18n("Overwrite the disk file with the editor content.") );
}
else
showButton( User1, false );
}
KateModOnHdPrompt::~KateModOnHdPrompt()
{
}
void KateModOnHdPrompt::slotDiff()
{
// Start a KProcess that creates a diff
KProcIO *p = new KProcIO();
p->setComm( KProcess::All );
*p << "diff" << "-u" << "-" << m_doc->url().path();
connect( p, SIGNAL(processExited(KProcess*)), this, SLOT(slotPDone(KProcess*)) );
connect( p, SIGNAL(readReady(KProcIO*)), this, SLOT(slotPRead(KProcIO*)) );
setCursor( WaitCursor );
p->start( KProcess::NotifyOnExit, true );
uint lastln = m_doc->numLines();
for ( uint l = 0; l < lastln; l++ )
p->writeStdin( m_doc->textLine( l ) );
p->closeWhenDone();
}
void KateModOnHdPrompt::slotPRead( KProcIO *p)
{
// create a file for the diff if we haven't one allready
if ( ! m_tmpfile )
m_tmpfile = new KTempFile();
// put all the data we have in it
QString stmp;
bool dataRead = false;
while ( p->readln( stmp, false ) > -1 )
{
*m_tmpfile->textStream() << stmp << endl;
dataRead = true;
}
// dominik: only ackRead(), when we *really* read data, otherwise, this slot
// is called initity times, which leads to a crash
if( dataRead )
p->ackRead();
}
void KateModOnHdPrompt::slotPDone( KProcess *p )
{
setCursor( ArrowCursor );
if( ! m_tmpfile )
{
// dominik: there were only whitespace changes, so that the diff returned by
// diff(1) has 0 bytes. So slotPRead() is never called, as there is
// no data, so that m_tmpfile was never created and thus is NULL.
// NOTE: would be nice, if we could produce a fake-diff, so that kompare
// tells us "The files are identical". Right now, we get an ugly
// "Could not parse diff output".
m_tmpfile = new KTempFile();
}
m_tmpfile->close();
if ( ! p->normalExit() /*|| p->exitStatus()*/ )
{
KMessageBox::sorry( this,
i18n("The diff command failed. Please make sure that "
"diff(1) is installed and in your PATH."),
i18n("Error Creating Diff") );
delete m_tmpfile;
m_tmpfile = 0;
return;
}
KRun::runURL( m_tmpfile->name(), "text/x-diff", true );
delete m_tmpfile;
m_tmpfile = 0;
}
void KateModOnHdPrompt::slotApply()
{
if ( KMessageBox::warningContinueCancel(
this,
i18n("Ignoring means that you will not be warned again (unless "
"the disk file changes once more): if you save the document, you "
"will overwrite the file on disk; if you do not save then the disk file "
"(if present) is what you have."),
i18n("You Are on Your Own"),
KStdGuiItem::cont(),
"kate_ignore_modonhd" ) != KMessageBox::Continue )
return;
done(Ignore);
}
void KateModOnHdPrompt::slotOk()
{
done( m_modtype == 3 ? Save : Reload );
}
void KateModOnHdPrompt::slotUser1()
{
done( Overwrite );
}
//END KateModOnHdPrompt
// kate: space-indent on; indent-width 2; replace-tabs on;