Move cursor to end of line when browsing history (KTextEdit).
[quassel.git] / src / uisupport / inputline.cpp
1 /***************************************************************************
2  *   Copyright (C) 2005/06 by the Quassel Project                          *
3  *   devel@quassel-irc.org                                                 *
4  *                                                                         *
5  *   This program is free software; you can redistribute it and/or modify  *
6  *   it under the terms of the GNU General Public License as published by  *
7  *   the Free Software Foundation; either version 2 of the License, or     *
8  *   (at your option) version 3.                                           *
9  *                                                                         *
10  *   This program is distributed in the hope that it will be useful,       *
11  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
12  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
13  *   GNU General Public License for more details.                          *
14  *                                                                         *
15  *   You should have received a copy of the GNU General Public License     *
16  *   along with this program; if not, write to the                         *
17  *   Free Software Foundation, Inc.,                                       *
18  *   59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.             *
19  ***************************************************************************/
20
21 #include "bufferview.h"
22 #include "graphicalui.h"
23 #include "inputline.h"
24 #include "tabcompleter.h"
25
26 InputLine::InputLine(QWidget *parent)
27   :
28 #ifdef HAVE_KDE
29     KTextEdit(parent),
30 #else
31     QLineEdit(parent),
32 #endif
33     idx(0),
34     tabCompleter(new TabCompleter(this))
35 {
36 #ifdef HAVE_KDE
37 //This is done to make the KTextEdit look like a lineedit
38   setMaximumHeight(document()->size().toSize().height());
39   setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
40   setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
41   setAcceptRichText(false);
42   setLineWrapMode(NoWrap);
43   enableFindReplace(false);
44   connect(this, SIGNAL(textChanged()), this, SLOT(on_textChanged()));
45 #endif
46
47   connect(this, SIGNAL(returnPressed()), this, SLOT(on_returnPressed()));
48   connect(this, SIGNAL(textChanged(QString)), this, SLOT(on_textChanged(QString)));
49 }
50
51 InputLine::~InputLine() {
52 }
53
54 void InputLine::setCustomFont(const QFont &font) {
55   setFont(font);
56 #ifdef HAVE_KDE
57   setMaximumHeight(document()->size().toSize().height());
58 #endif
59 }
60
61 bool InputLine::eventFilter(QObject *watched, QEvent *event) {
62   if(event->type() != QEvent::KeyPress)
63     return false;
64
65   // keys from BufferView should be sent to (and focus) the input line
66   BufferView *view = qobject_cast<BufferView *>(watched);
67   if(view) {
68     QKeyEvent *keyEvent = static_cast<QKeyEvent*>(event);
69     if(keyEvent->text().length() == 1 && !(keyEvent->modifiers() & (Qt::ControlModifier ^ Qt::AltModifier)) ) { // normal key press
70       QChar c = keyEvent->text().at(0);
71       if(c.isLetterOrNumber() || c.isSpace() || c.isPunct() || c.isSymbol()) {
72         setFocus();
73         keyPressEvent(keyEvent);
74         return true;
75       } else
76         return false;
77     }
78   }
79   return false;
80 }
81
82 void InputLine::keyPressEvent(QKeyEvent * event) {
83
84 #ifdef HAVE_KDE
85   if(event->matches(QKeySequence::Find)) {
86     QAction *act = GraphicalUi::actionCollection()->action("ToggleSearchBar");
87     if(act) {
88       act->toggle();
89       event->accept();
90       return;
91     }
92   }
93 #endif
94
95   switch(event->key()) {
96   case Qt::Key_Up:
97     event->accept();
98
99     addToHistory(text(), true);
100
101     if(idx > 0) {
102       idx--;
103       showHistoryEntry();
104     }
105
106     break;
107
108   case Qt::Key_Down:
109     event->accept();
110
111     addToHistory(text(), true);
112
113     if(idx < history.count()) {
114       idx++;
115       if(idx < history.count() || tempHistory.contains(idx)) // tempHistory might have an entry for idx == history.count() + 1
116         showHistoryEntry();
117       else
118         resetLine();              // equals clear() in this case
119     } else {
120       addToHistory(text());
121       resetLine();
122     }
123
124     break;
125
126   case Qt::Key_Select:          // for Qtopia
127     emit returnPressed();
128     break;
129
130 #ifdef HAVE_KDE
131 //Since this is a ktextedit, we don't have this signal "natively"
132   case Qt::Key_Return:
133   case Qt::Key_Enter:
134     event->accept();
135     emit returnPressed();
136     break;
137
138 #endif
139
140   default:
141 #ifdef HAVE_KDE
142     KTextEdit::keyPressEvent(event);
143 #else
144     QLineEdit::keyPressEvent(event);
145 #endif
146   }
147 }
148
149 bool InputLine::addToHistory(const QString &text, bool temporary) {
150   if(text.isEmpty())
151     return false;
152
153   Q_ASSERT(0 <= idx && idx <= history.count());
154
155   if(history.isEmpty() || text != history[idx - (int)(idx == history.count())]) {
156     // if an entry of the history is changed, we remember it and show it again at this
157     // position until a line was actually sent
158     // sent lines get appended to the history
159     if(temporary) {
160       tempHistory[idx] = text;
161     } else {
162       history << text;
163       tempHistory.clear();
164     }
165     return true;
166   } else {
167     return false;
168   }
169 }
170
171 void InputLine::on_returnPressed() {
172   if(!text().isEmpty()) {
173     addToHistory(text());
174     emit sendText(text());
175     resetLine();
176   }
177 }
178
179 void InputLine::on_textChanged(QString newText) {
180   QStringList lineSeparators;
181   lineSeparators << QString("\r\n")
182                  << QString('\n')
183                  << QString('\r');
184
185   QString lineSep;
186   foreach(QString separator, lineSeparators) {
187     if(newText.contains(separator)) {
188       lineSep = separator;
189       break;
190     }
191   }
192
193   if(lineSep.isEmpty())
194     return;
195
196   QStringList lines = newText.split(lineSep);
197   clear();
198
199   if(lines.count() >= 4) {
200     QString msg = tr("Do you really want to paste %n lines?", "", lines.count());
201     msg += "<p>";
202     for(int i = 0; i < 3; i++) {
203       msg += lines[i].left(40);
204       if(lines[i].count() > 40)
205         msg += "...";
206       msg += "<br />";
207     }
208     msg += "...</p>";
209     QMessageBox question(QMessageBox::NoIcon, tr("Paste Protection"), msg, QMessageBox::Yes|QMessageBox::No);
210     question.setDefaultButton(QMessageBox::No);
211 #ifdef Q_WS_MAC
212     question.setWindowFlags(question.windowFlags() | Qt::Sheet);
213 #endif
214     if(question.exec() == QMessageBox::No)
215       return;
216   }
217
218   foreach(QString line, lines) {
219     if(!line.isEmpty()) {
220       clear();
221       insert(line);
222       emit returnPressed();
223     }
224   }
225 //   if(newText.contains(lineSep)) {
226 //     clear();
227 //     QString line = newText.section(lineSep, 0, 0);
228 //     QString remainder = newText.section(lineSep, 1);
229 //     insert(line);
230 //     emit returnPressed();
231 //     insert(remainder);
232 //   }
233 }
234
235 void InputLine::resetLine() {
236   // every time the InputLine is cleared we also reset history index
237   idx = history.count();
238   clear();
239 }
240
241 void InputLine::showHistoryEntry() {
242   // if the user changed the history, display the changed line
243   tempHistory.contains(idx) ? setText(tempHistory[idx]) : setText(history[idx]);
244 #ifdef HAVE_KDE
245   QTextCursor cursor = textCursor();
246   cursor.movePosition(QTextCursor::End);
247   setTextCursor(cursor);
248 #endif
249 }