diff options
Diffstat (limited to 'kdeui/keditcl1.cpp')
-rw-r--r-- | kdeui/keditcl1.cpp | 718 |
1 files changed, 718 insertions, 0 deletions
diff --git a/kdeui/keditcl1.cpp b/kdeui/keditcl1.cpp new file mode 100644 index 000000000..56ab68c60 --- /dev/null +++ b/kdeui/keditcl1.cpp @@ -0,0 +1,718 @@ +/* This file is part of the KDE libraries + + Copyright (C) 1997 Bernd Johannes Wuebben <wuebben@math.cornell.edu> + Copyright (C) 2000 Waldo Bastian <bastian@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 as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + 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 <qdragobject.h> +#include <qpopupmenu.h> +#include <qtextstream.h> +#include <qtimer.h> + +#include <kapplication.h> +#include <kcursor.h> +#include <kdebug.h> +#include <kcmenumngr.h> +#include <kfontdialog.h> +#include <klocale.h> +#include <kmessagebox.h> +#include <kstdaccel.h> +#include <kurldrag.h> + +#include "keditcl.h" +#include "keditcl.moc" + +class KEdit::KEditPrivate +{ +public: + bool overwriteEnabled:1; + bool posDirty:1; + bool autoUpdate:1; +}; + + +KEdit::KEdit(QWidget *_parent, const char *name) + : QMultiLineEdit(_parent, name) +{ + d = new KEditPrivate; + d->overwriteEnabled = false; + d->posDirty = true; + d->autoUpdate = true; + + parent = _parent; + + // set some defaults + + line_pos = col_pos = 0; + + srchdialog = NULL; + replace_dialog= NULL; + gotodialog = NULL; + + setAcceptDrops(true); + KCursor::setAutoHideCursor( this, true ); + + connect(this, SIGNAL(cursorPositionChanged(int,int)), + this, SLOT(slotCursorPositionChanged())); +} + + +KEdit::~KEdit() +{ + delete d; +} + +void +KEdit::setAutoUpdate(bool b) +{ + d->autoUpdate = b; +} + +void +KEdit::insertText(QTextStream *stream) +{ +// setAutoUpdate(false); + int line, col; + getCursorPosition(&line, &col); + int saveline = line; + int savecol = col; + QString textLine; + + // MS: Patch by Martin Schenk <martin@schenk.com> + // MS: disable UNDO, or QMultiLineEdit remembers every textLine !!! + // memory usage is: + // textLine: 2*size rounded up to nearest power of 2 (520Kb -> 1024Kb) + // widget: about (2*size + 60bytes*lines) + // -> without disabling undo, it often needs almost 8*size + int oldUndoDepth = undoDepth(); + setUndoDepth( 0 ); // ### -1? + + // MS: read everything at once if file <= 1MB, + // else read in 5000-line chunks to keep memory usage acceptable. + QIODevice *dev=stream->device(); + if (dev && dev->size()>(1024*1024)) { + while(1) { + int i; + textLine=""; + for (i=0; i<5000; i++) { + QString line=stream->readLine(); + if (line.isNull()) break; // EOF + textLine+=line+'\n'; + } + insertAt(textLine, line, col); + line+=i; col=0; + if (i!=5000) break; + } + } + else { + textLine = stream->read(); // Read all ! + insertAt( textLine, line, col); + } + setUndoDepth( oldUndoDepth ); + + setCursorPosition(saveline, savecol); +// setAutoUpdate(true); + +// repaint(); + + setModified(true); + setFocus(); + + // Bernd: Please don't leave debug message like that lying around + // they cause ENORMOUSE performance hits. Once upon a day + // kedit used to be really really fast using memmap etc ..... + // oh well .... + + // QString str = text(); + // for (int i = 0; i < (int) str.length(); i++) + // printf("KEdit: U+%04X\n", str[i].unicode()); + +} + +void +KEdit::cleanWhiteSpace() +{ + d->autoUpdate = false; + if (!hasMarkedText()) + selectAll(); + QString oldText = markedText(); + QString newText; + QStringList lines = QStringList::split('\n', oldText, true); + bool addSpace = false; + bool firstLine = true; + QChar lastChar = oldText[oldText.length()-1]; + QChar firstChar = oldText[0]; + for(QStringList::Iterator it = lines.begin(); + it != lines.end();) + { + QString line = (*it).simplifyWhiteSpace(); + if (line.isEmpty()) + { + if (addSpace) + newText += QString::fromLatin1("\n\n"); + if (firstLine) + { + if (firstChar.isSpace()) + newText += '\n'; + firstLine = false; + } + addSpace = false; + } + else + { + if (addSpace) + newText += ' '; + if (firstLine) + { + if (firstChar.isSpace()) + newText += ' '; + firstLine = false; + } + newText += line; + addSpace = true; + } + it = lines.remove(it); + } + if (addSpace) + { + if (lastChar == '\n') + newText += '\n'; + else if (lastChar.isSpace()) + newText += ' '; + } + + if (oldText == newText) + { + deselect(); + d->autoUpdate = true; + repaint(); + return; + } + if (wordWrap() == NoWrap) + { + // If wordwrap is off, we have to do some line-wrapping ourselves now + // We use another QMultiLineEdit for this, so that we get nice undo + // behavior. + QMultiLineEdit *we = new QMultiLineEdit(); + we->setWordWrap(FixedColumnWidth); + we->setWrapColumnOrWidth(78); + we->setText(newText); + newText = QString::null; + for(int i = 0; i < we->numLines(); i++) + { + QString line = we->textLine(i); + if (line.right(1) != "\n") + line += '\n'; + newText += line; + } + delete we; + } + + insert(newText); + d->autoUpdate = true; + repaint(); + + setModified(true); + setFocus(); +} + + +void +KEdit::saveText(QTextStream *stream) +{ + saveText(stream, false); +} + +void +KEdit::saveText(QTextStream *stream, bool softWrap) +{ + int line_count = numLines()-1; + if (line_count < 0) + return; + + if (softWrap || (wordWrap() == NoWrap)) + { + for(int i = 0; i < line_count; i++) + { + (*stream) << textLine(i) << '\n'; + } + (*stream) << textLine(line_count); + } + else + { + for(int i = 0; i <= line_count; i++) + { + int lines_in_parag = linesOfParagraph(i); + if (lines_in_parag == 1) + { + (*stream) << textLine(i); + } + else + { + QString parag_text = textLine(i); + int pos = 0; + int first_pos = 0; + int current_line = 0; + while(true) { + while(lineOfChar(i, pos) == current_line) pos++; + (*stream) << parag_text.mid(first_pos, pos - first_pos - 1) << '\n'; + current_line++; + first_pos = pos; + if (current_line+1 == lines_in_parag) + { + // Last line + (*stream) << parag_text.mid(pos); + break; + } + } + } + if (i < line_count) + (*stream) << '\n'; + } + } +} + +int KEdit::currentLine(){ + + computePosition(); + return line_pos; + +} + +int KEdit::currentColumn(){ + + computePosition(); + return col_pos; +} + +void KEdit::slotCursorPositionChanged() +{ + d->posDirty = true; + emit CursorPositionChanged(); +} + +void KEdit::computePosition() +{ + if (!d->posDirty) return; + d->posDirty = false; + + int line, col; + + getCursorPosition(&line,&col); + + // line is expressed in paragraphs, we now need to convert to lines + line_pos = 0; + if (wordWrap() == NoWrap) + { + line_pos = line; + } + else + { + for(int i = 0; i < line; i++) + line_pos += linesOfParagraph(i); + } + + int line_offset = lineOfChar(line, col); + line_pos += line_offset; + + // We now calculate where the current line starts in the paragraph. + QString linetext = textLine(line); + int start_of_line = 0; + if (line_offset > 0) + { + start_of_line = col; + while(lineOfChar(line, --start_of_line) == line_offset); + start_of_line++; + } + + + // O.K here is the deal: The function getCursorPositoin returns the character + // position of the cursor, not the screenposition. I.e,. assume the line + // consists of ab\tc then the character c will be on the screen on position 8 + // whereas getCursorPosition will return 3 if the cursors is on the character c. + // Therefore we need to compute the screen position from the character position. + // That's what all the following trouble is all about: + + int coltemp = col-start_of_line; + int pos = 0; + int find = 0; + int mem = 0; + bool found_one = false; + + // if you understand the following algorithm you are worthy to look at the + // kedit+ sources -- if not, go away ;-) + + + while(find >=0 && find <= coltemp- 1 ){ + find = linetext.find('\t', find+start_of_line, true )-start_of_line; + if( find >=0 && find <= coltemp - 1 ){ + found_one = true; + pos = pos + find - mem; + pos = pos + 8 - pos % 8; + mem = find; + find ++; + } + } + + pos = pos + coltemp - mem; // add the number of characters behind the + // last tab on the line. + + if (found_one){ + pos = pos - 1; + } + + col_pos = pos; +} + + +void KEdit::keyPressEvent ( QKeyEvent *e) +{ + // ignore Ctrl-Return so that KDialogBase can catch them + if ( e->key() == Key_Return && e->state() == ControlButton ) { + e->ignore(); + return; + } + + KKey key(e); + int keyQt = key.keyCodeQt(); + + if ( keyQt == CTRL+Key_K ){ + + int line = 0; + int col = 0; + QString killstring; + + if(!killing){ + killbufferstring = ""; + killtrue = false; + lastwasanewline = false; + } + + if(!atEnd()){ + + getCursorPosition(&line,&col); + killstring = textLine(line); + killstring = killstring.mid(col,killstring.length()); + + + if(!killbufferstring.isEmpty() && !killtrue && !lastwasanewline){ + killbufferstring += '\n'; + } + + if( (killstring.length() == 0) && !killtrue){ + killbufferstring += '\n'; + lastwasanewline = true; + } + + if(killstring.length() > 0){ + + killbufferstring += killstring; + lastwasanewline = false; + killtrue = true; + + }else{ + + lastwasanewline = false; + killtrue = !killtrue; + + } + + }else{ + + if(killbufferstring.isEmpty() && !killtrue && !lastwasanewline){ + killtrue = true; + } + + } + + killing = true; + + QMultiLineEdit::keyPressEvent(e); + setModified(true); + return; + } + else if ( keyQt == CTRL+Key_Y ){ + + int line = 0; + int col = 0; + + getCursorPosition(&line,&col); + + QString tmpstring = killbufferstring; + if(!killtrue) + tmpstring += '\n'; + + insertAt(tmpstring,line,col); + + killing = false; + setModified(true); + return; + } + + killing = false; + + if ( KStdAccel::copy().contains( key ) ) + copy(); + else if ( isReadOnly() ) + QMultiLineEdit::keyPressEvent( e ); + // If this is an unmodified printable key, send it directly to QMultiLineEdit. + else if ( !(key.keyCodeQt() & (CTRL | ALT)) && !e->text().isEmpty() && e->text().unicode()->isPrint() ) + QMultiLineEdit::keyPressEvent( e ); + else if ( KStdAccel::paste().contains( key ) ) { + paste(); + setModified(true); + slotCursorPositionChanged(); + } + else if ( KStdAccel::cut().contains( key ) ) { + cut(); + setModified(true); + slotCursorPositionChanged(); + } + else if ( KStdAccel::undo().contains( key ) ) { + undo(); + setModified(true); + slotCursorPositionChanged(); + } + else if ( KStdAccel::redo().contains( key ) ) { + redo(); + setModified(true); + slotCursorPositionChanged(); + } + else if ( KStdAccel::deleteWordBack().contains( key ) ) { + moveCursor(MoveWordBackward, true); + if (hasSelectedText()) + del(); + setModified(true); + slotCursorPositionChanged(); + } + else if ( KStdAccel::deleteWordForward().contains( key ) ) { + moveCursor(MoveWordForward, true); + if (hasSelectedText()) + del(); + setModified(true); + slotCursorPositionChanged(); + } + else if ( KStdAccel::backwardWord().contains( key ) ) { + CursorAction action = MoveWordBackward; + int para, index; + getCursorPosition( ¶, & index ); + if (text(para).isRightToLeft()) + action = MoveWordForward; + moveCursor(action, false ); + slotCursorPositionChanged(); + } + else if ( KStdAccel::forwardWord().contains( key ) ) { + CursorAction action = MoveWordForward; + int para, index; + getCursorPosition( ¶, & index ); + if (text(para).isRightToLeft()) + action = MoveWordBackward; + moveCursor( action, false ); + slotCursorPositionChanged(); + } + else if ( KStdAccel::next().contains( key ) ) { + moveCursor( MovePgDown, false ); + slotCursorPositionChanged(); + } + else if ( KStdAccel::prior().contains( key ) ) { + moveCursor( MovePgUp, false ); + slotCursorPositionChanged(); + } + else if ( KStdAccel::home().contains( key ) ) { + moveCursor( MoveHome, false ); + slotCursorPositionChanged(); + } + else if ( KStdAccel::end().contains( key ) ) { + moveCursor( MoveEnd, false ); + slotCursorPositionChanged(); + } + else if ( KStdAccel::beginningOfLine().contains( key ) ) { + moveCursor( MoveLineStart, false); + slotCursorPositionChanged(); + } + else if ( KStdAccel::endOfLine().contains( key ) ) { + moveCursor( MoveLineEnd, false); + slotCursorPositionChanged(); + } + else if ( key == Key_Insert ) { + if (d->overwriteEnabled) + { + this->setOverwriteMode(!this->isOverwriteMode()); + emit toggle_overwrite_signal(); + } + } + else + QMultiLineEdit::keyPressEvent(e); +} + +void KEdit::installRBPopup(QPopupMenu *p) { + KContextMenuManager::insert( this, p ); +} + +void KEdit::selectFont(){ + + QFont font = this->font(); + KFontDialog::getFont(font); + this->setFont(font); + +} + +void KEdit::doGotoLine() { + + if( !gotodialog ) + gotodialog = new KEdGotoLine( parent, "gotodialog" ); + + this->clearFocus(); + + gotodialog->exec(); + // this seems to be not necessary + // gotodialog->setFocus(); + if( gotodialog->result() != KEdGotoLine::Accepted) + return; + int target_line = gotodialog->getLineNumber()-1; + if (wordWrap() == NoWrap) + { + setCursorPosition( target_line, 0 ); + setFocus(); + return; + } + + int max_parag = paragraphs(); + + int line = 0; + int parag = -1; + int lines_in_parag = 0; + while ((++parag < max_parag) && (line + lines_in_parag < target_line)) + { + line += lines_in_parag; + lines_in_parag = linesOfParagraph(parag); + } + + int col = 0; + if (parag >= max_parag) + { + target_line = line + lines_in_parag - 1; + parag = max_parag-1; + } + + while(1+line+lineOfChar(parag,col) < target_line) col++; + setCursorPosition( parag, col ); + setFocus(); +} + + +void KEdit::dragMoveEvent(QDragMoveEvent* e) { + + if(KURLDrag::canDecode(e)) + e->accept(); + else if(QTextDrag::canDecode(e)) + QMultiLineEdit::dragMoveEvent(e); +} + +void KEdit::contentsDragMoveEvent(QDragMoveEvent* e) { + + if(KURLDrag::canDecode(e)) + e->accept(); + else if(QTextDrag::canDecode(e)) + QMultiLineEdit::contentsDragMoveEvent(e); +} + +void KEdit::dragEnterEvent(QDragEnterEvent* e) { + + kdDebug() << "KEdit::dragEnterEvent()" << endl; + e->accept(KURLDrag::canDecode(e) || QTextDrag::canDecode(e)); +} + +void KEdit::contentsDragEnterEvent(QDragEnterEvent* e) { + + kdDebug() << "KEdit::contentsDragEnterEvent()" << endl; + e->accept(KURLDrag::canDecode(e) || QTextDrag::canDecode(e)); +} + + +void KEdit::dropEvent(QDropEvent* e) { + + kdDebug() << "KEdit::dropEvent()" << endl; + + if(KURLDrag::canDecode(e)) { + emit gotUrlDrop(e); + } + else if(QTextDrag::canDecode(e)) + QMultiLineEdit::dropEvent(e); +} + +void KEdit::contentsDropEvent(QDropEvent* e) { + + kdDebug() << "KEdit::contentsDropEvent()" << endl; + + if(KURLDrag::canDecode(e)) { + emit gotUrlDrop(e); + } + else if(QTextDrag::canDecode(e)) + QMultiLineEdit::contentsDropEvent(e); +} + +void KEdit::setOverwriteEnabled(bool b) +{ + d->overwriteEnabled = b; +} + +// QWidget::create() turns off mouse-Tracking which would break auto-hiding +void KEdit::create( WId id, bool initializeWindow, bool destroyOldWindow ) +{ + QMultiLineEdit::create( id, initializeWindow, destroyOldWindow ); + KCursor::setAutoHideCursor( this, true ); +} + +void KEdit::ensureCursorVisible() +{ + if (!d->autoUpdate) + return; + + QMultiLineEdit::ensureCursorVisible(); +} + +void KEdit::setCursor( const QCursor &c ) +{ + if (!d->autoUpdate) + return; + + QMultiLineEdit::setCursor(c); +} + +void KEdit::viewportPaintEvent( QPaintEvent*pe ) +{ + if (!d->autoUpdate) + return; + + QMultiLineEdit::viewportPaintEvent(pe); +} + + +void KEdGotoLine::virtual_hook( int id, void* data ) +{ KDialogBase::virtual_hook( id, data ); } + +void KEdFind::virtual_hook( int id, void* data ) +{ KDialogBase::virtual_hook( id, data ); } + +void KEdReplace::virtual_hook( int id, void* data ) +{ KDialogBase::virtual_hook( id, data ); } + +void KEdit::virtual_hook( int, void* ) +{ /*BASE::virtual_hook( id, data );*/ } + |