summaryrefslogtreecommitdiffstats
path: root/kate/part/katesupercursor.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kate/part/katesupercursor.cpp')
-rw-r--r--kate/part/katesupercursor.cpp746
1 files changed, 746 insertions, 0 deletions
diff --git a/kate/part/katesupercursor.cpp b/kate/part/katesupercursor.cpp
new file mode 100644
index 000000000..774b695db
--- /dev/null
+++ b/kate/part/katesupercursor.cpp
@@ -0,0 +1,746 @@
+/* This file is part of the KDE libraries
+ Copyright (C) 2003 Hamish Rodda <rodda@kde.org>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Library General Public
+ License version 2 as published by the Free Software Foundation.
+
+ This library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Library General Public License for more details.
+
+ You should have received a copy of the GNU Library General Public License
+ along with this library; see the file COPYING.LIB. If not, write to
+ the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ Boston, MA 02110-1301, USA.
+*/
+
+#include "katesupercursor.h"
+#include "katesupercursor.moc"
+
+#include "katedocument.h"
+
+#include <kdebug.h>
+
+#include <qobjectlist.h>
+
+KateSuperCursor::KateSuperCursor(KateDocument* doc, bool privateC, const KateTextCursor& cursor, QObject* parent, const char* name)
+ : QObject(parent, name)
+ , KateDocCursor(cursor.line(), cursor.col(), doc)
+ , Kate::Cursor ()
+ , m_doc (doc)
+{
+ m_moveOnInsert = false;
+ m_lineRemoved = false;
+ m_privateCursor = privateC;
+
+ m_doc->addSuperCursor (this, privateC);
+}
+
+KateSuperCursor::KateSuperCursor(KateDocument* doc, bool privateC, int lineNum, int col, QObject* parent, const char* name)
+ : QObject(parent, name)
+ , KateDocCursor(lineNum, col, doc)
+ , Kate::Cursor ()
+ , m_doc (doc)
+{
+ m_moveOnInsert = false;
+ m_lineRemoved = false;
+ m_privateCursor = privateC;
+
+ m_doc->addSuperCursor (this, privateC);
+}
+
+KateSuperCursor::~KateSuperCursor ()
+{
+ m_doc->removeSuperCursor (this, m_privateCursor);
+}
+
+void KateSuperCursor::position(uint *pline, uint *pcol) const
+{
+ KateDocCursor::position(pline, pcol);
+}
+
+bool KateSuperCursor::setPosition(uint line, uint col)
+{
+ if (line == uint(-2) && col == uint(-2)) { delete this; return true; }
+ return KateDocCursor::setPosition(line, col);
+}
+
+bool KateSuperCursor::insertText(const QString& s)
+{
+ return KateDocCursor::insertText(s);
+}
+
+bool KateSuperCursor::removeText(uint nbChar)
+{
+ return KateDocCursor::removeText(nbChar);
+}
+
+QChar KateSuperCursor::currentChar() const
+{
+ return KateDocCursor::currentChar();
+}
+
+bool KateSuperCursor::atStartOfLine() const
+{
+ return col() == 0;
+}
+
+bool KateSuperCursor::atEndOfLine() const
+{
+ return col() >= (int)m_doc->kateTextLine(line())->length();
+}
+
+bool KateSuperCursor::moveOnInsert() const
+{
+ return m_moveOnInsert;
+}
+
+void KateSuperCursor::setMoveOnInsert(bool moveOnInsert)
+{
+ m_moveOnInsert = moveOnInsert;
+}
+
+void KateSuperCursor::setLine(int lineNum)
+{
+ int tempLine = line(), tempcol = col();
+ KateDocCursor::setLine(lineNum);
+
+ if (tempLine != line() || tempcol != col())
+ emit positionDirectlyChanged();
+}
+
+void KateSuperCursor::setCol(int colNum)
+{
+ KateDocCursor::setCol(colNum);
+}
+
+void KateSuperCursor::setPos(const KateTextCursor& pos)
+{
+ KateDocCursor::setPos(pos);
+}
+
+void KateSuperCursor::setPos(int lineNum, int colNum)
+{
+ KateDocCursor::setPos(lineNum, colNum);
+}
+
+void KateSuperCursor::editTextInserted(uint line, uint col, uint len)
+{
+ if (m_line == int(line))
+ {
+ if ((m_col > int(col)) || (m_moveOnInsert && (m_col == int(col))))
+ {
+ bool insertedAt = m_col == int(col);
+
+ m_col += len;
+
+ if (insertedAt)
+ emit charInsertedAt();
+
+ emit positionChanged();
+ return;
+ }
+ }
+
+ emit positionUnChanged();
+}
+
+void KateSuperCursor::editTextRemoved(uint line, uint col, uint len)
+{
+ if (m_line == int(line))
+ {
+ if (m_col > int(col))
+ {
+ if (m_col > int(col + len))
+ {
+ m_col -= len;
+ }
+ else
+ {
+ bool prevCharDeleted = m_col == int(col + len);
+
+ m_col = col;
+
+ if (prevCharDeleted)
+ emit charDeletedBefore();
+ else
+ emit positionDeleted();
+ }
+
+ emit positionChanged();
+ return;
+
+ }
+ else if (m_col == int(col))
+ {
+ emit charDeletedAfter();
+ }
+ }
+
+ emit positionUnChanged();
+}
+
+void KateSuperCursor::editLineWrapped(uint line, uint col, bool newLine)
+{
+ if (newLine)
+ {
+ if (m_line > int(line) || (m_line == int(line) && m_col >= int(col)))
+ {
+ if(m_line == int(line))
+ m_col -= col;
+ m_line++;
+
+ emit positionChanged();
+ return;
+ }
+ }
+ else if ( (m_line == int(line)) && (m_col > int(col)) || (m_moveOnInsert && (m_col == int(col))) )
+ {
+ m_line++;
+ m_col -= col;
+
+ emit positionChanged();
+ return;
+ }
+
+ emit positionUnChanged();
+}
+
+void KateSuperCursor::editLineUnWrapped(uint line, uint col, bool removeLine, uint length)
+{
+ if (removeLine && (m_line > int(line+1)))
+ {
+ m_line--;
+
+ emit positionChanged();
+ return;
+ }
+ else if ( (m_line == int(line+1)) && (removeLine || (m_col < int(length))) )
+ {
+ m_line = line;
+ m_col += col;
+
+ emit positionChanged();
+ return;
+ }
+ else if ( (m_line == int(line+1)) && (m_col >= int(length)) )
+ {
+ m_col -= length;
+
+ emit positionChanged();
+ return;
+ }
+
+ emit positionUnChanged();
+}
+
+void KateSuperCursor::editLineInserted (uint line)
+{
+ if (m_line >= int(line))
+ {
+ m_line++;
+
+ emit positionChanged();
+ return;
+ }
+
+ emit positionUnChanged();
+}
+
+void KateSuperCursor::editLineRemoved(uint line)
+{
+ if (m_line > int(line))
+ {
+ m_line--;
+
+ emit positionChanged();
+ return;
+ }
+ else if (m_line == int(line))
+ {
+ m_line = (line <= m_doc->lastLine()) ? line : (line - 1);
+ m_col = 0;
+
+ emit positionDeleted();
+
+ emit positionChanged();
+ return;
+ }
+
+ emit positionUnChanged();
+}
+
+KateSuperCursor::operator QString()
+{
+ return QString("[%1,%1]").arg(line()).arg(col());
+}
+
+KateSuperRange::KateSuperRange(KateSuperCursor* start, KateSuperCursor* end, QObject* parent, const char* name)
+ : QObject(parent, name)
+ , m_start(start)
+ , m_end(end)
+ , m_evaluate(false)
+ , m_startChanged(false)
+ , m_endChanged(false)
+ , m_deleteCursors(false)
+ , m_allowZeroLength(false)
+{
+ init();
+}
+
+KateSuperRange::KateSuperRange(KateDocument* doc, const KateRange& range, QObject* parent, const char* name)
+ : QObject(parent, name)
+ , m_start(new KateSuperCursor(doc, true, range.start()))
+ , m_end(new KateSuperCursor(doc, true, range.end()))
+ , m_evaluate(false)
+ , m_startChanged(false)
+ , m_endChanged(false)
+ , m_deleteCursors(true)
+ , m_allowZeroLength(false)
+{
+ init();
+}
+
+KateSuperRange::KateSuperRange(KateDocument* doc, const KateTextCursor& start, const KateTextCursor& end, QObject* parent, const char* name)
+ : QObject(parent, name)
+ , m_start(new KateSuperCursor(doc, true, start))
+ , m_end(new KateSuperCursor(doc, true, end))
+ , m_evaluate(false)
+ , m_startChanged(false)
+ , m_endChanged(false)
+ , m_deleteCursors(true)
+ , m_allowZeroLength(false)
+{
+ init();
+}
+
+void KateSuperRange::init()
+{
+ Q_ASSERT(isValid());
+ if (!isValid())
+ kdDebug(13020) << superStart() << " " << superEnd() << endl;
+
+ insertChild(m_start);
+ insertChild(m_end);
+
+ setBehaviour(DoNotExpand);
+
+ // Not necessarily the best implementation
+ connect(m_start, SIGNAL(positionDirectlyChanged()), SIGNAL(contentsChanged()));
+ connect(m_end, SIGNAL(positionDirectlyChanged()), SIGNAL(contentsChanged()));
+
+ connect(m_start, SIGNAL(positionChanged()), SLOT(slotEvaluateChanged()));
+ connect(m_end, SIGNAL(positionChanged()), SLOT(slotEvaluateChanged()));
+ connect(m_start, SIGNAL(positionUnChanged()), SLOT(slotEvaluateUnChanged()));
+ connect(m_end, SIGNAL(positionUnChanged()), SLOT(slotEvaluateUnChanged()));
+ connect(m_start, SIGNAL(positionDeleted()), SIGNAL(boundaryDeleted()));
+ connect(m_end, SIGNAL(positionDeleted()), SIGNAL(boundaryDeleted()));
+}
+
+KateSuperRange::~KateSuperRange()
+{
+ if (m_deleteCursors)
+ {
+ //insertChild(m_start);
+ //insertChild(m_end);
+ delete m_start;
+ delete m_end;
+ }
+}
+
+KateTextCursor& KateSuperRange::start()
+{
+ return *m_start;
+}
+
+const KateTextCursor& KateSuperRange::start() const
+{
+ return *m_start;
+}
+
+KateTextCursor& KateSuperRange::end()
+{
+ return *m_end;
+}
+
+const KateTextCursor& KateSuperRange::end() const
+{
+ return *m_end;
+}
+
+KateSuperCursor& KateSuperRange::superStart()
+{
+ return *m_start;
+}
+
+const KateSuperCursor& KateSuperRange::superStart() const
+{
+ return *m_start;
+}
+
+KateSuperCursor& KateSuperRange::superEnd()
+{
+ return *m_end;
+}
+
+const KateSuperCursor& KateSuperRange::superEnd() const
+{
+ return *m_end;
+}
+
+int KateSuperRange::behaviour() const
+{
+ return (m_start->moveOnInsert() ? DoNotExpand : ExpandLeft) | (m_end->moveOnInsert() ? ExpandRight : DoNotExpand);
+}
+
+void KateSuperRange::setBehaviour(int behaviour)
+{
+ m_start->setMoveOnInsert(behaviour & ExpandLeft);
+ m_end->setMoveOnInsert(!(behaviour & ExpandRight));
+}
+
+bool KateSuperRange::isValid() const
+{
+ return superStart() <= superEnd();
+}
+
+bool KateSuperRange::owns(const KateTextCursor& cursor) const
+{
+ if (!includes(cursor)) return false;
+
+ if (children())
+ for (QObjectListIt it(*children()); *it; ++it)
+ if ((*it)->inherits("KateSuperRange"))
+ if (static_cast<KateSuperRange*>(*it)->owns(cursor))
+ return false;
+
+ return true;
+}
+
+bool KateSuperRange::includes(const KateTextCursor& cursor) const
+{
+ return isValid() && cursor >= superStart() && cursor < superEnd();
+}
+
+bool KateSuperRange::includes(uint lineNum) const
+{
+ return isValid() && (int)lineNum >= superStart().line() && (int)lineNum <= superEnd().line();
+}
+
+bool KateSuperRange::includesWholeLine(uint lineNum) const
+{
+ return isValid() && ((int)lineNum > superStart().line() || ((int)lineNum == superStart().line() && superStart().atStartOfLine())) && ((int)lineNum < superEnd().line() || ((int)lineNum == superEnd().line() && superEnd().atEndOfLine()));
+}
+
+bool KateSuperRange::boundaryAt(const KateTextCursor& cursor) const
+{
+ return isValid() && (cursor == superStart() || cursor == superEnd());
+}
+
+bool KateSuperRange::boundaryOn(uint lineNum) const
+{
+ return isValid() && (superStart().line() == (int)lineNum || superEnd().line() == (int)lineNum);
+}
+
+void KateSuperRange::slotEvaluateChanged()
+{
+ if (sender() == static_cast<QObject*>(m_start)) {
+ if (m_evaluate) {
+ if (!m_endChanged) {
+ // Only one was changed
+ evaluateEliminated();
+
+ } else {
+ // Both were changed
+ evaluatePositionChanged();
+ m_endChanged = false;
+ }
+
+ } else {
+ m_startChanged = true;
+ }
+
+ } else {
+ if (m_evaluate) {
+ if (!m_startChanged) {
+ // Only one was changed
+ evaluateEliminated();
+
+ } else {
+ // Both were changed
+ evaluatePositionChanged();
+ m_startChanged = false;
+ }
+
+ } else {
+ m_endChanged = true;
+ }
+ }
+
+ m_evaluate = !m_evaluate;
+}
+
+void KateSuperRange::slotEvaluateUnChanged()
+{
+ if (sender() == static_cast<QObject*>(m_start)) {
+ if (m_evaluate) {
+ if (m_endChanged) {
+ // Only one changed
+ evaluateEliminated();
+ m_endChanged = false;
+
+ } else {
+ // Neither changed
+ emit positionUnChanged();
+ }
+ }
+
+ } else {
+ if (m_evaluate) {
+ if (m_startChanged) {
+ // Only one changed
+ evaluateEliminated();
+ m_startChanged = false;
+
+ } else {
+ // Neither changed
+ emit positionUnChanged();
+ }
+ }
+ }
+
+ m_evaluate = !m_evaluate;
+}
+
+void KateSuperRange::slotTagRange()
+{
+ emit tagRange(this);
+}
+
+void KateSuperRange::evaluateEliminated()
+{
+ if (superStart() == superEnd()) {
+ if (!m_allowZeroLength) emit eliminated();
+ }
+ else
+ emit contentsChanged();
+}
+
+void KateSuperRange::evaluatePositionChanged()
+{
+ if (superStart() == superEnd())
+ emit eliminated();
+ else
+ emit positionChanged();
+}
+
+int KateSuperCursorList::compareItems(QPtrCollection::Item item1, QPtrCollection::Item item2)
+{
+ if (*(static_cast<KateSuperCursor*>(item1)) == *(static_cast<KateSuperCursor*>(item2)))
+ return 0;
+
+ return *(static_cast<KateSuperCursor*>(item1)) < *(static_cast<KateSuperCursor*>(item2)) ? -1 : 1;
+}
+
+KateSuperRangeList::KateSuperRangeList(bool autoManage, QObject* parent, const char* name)
+ : QObject(parent, name)
+ , m_autoManage(autoManage)
+ , m_connect(true)
+ , m_trackingBoundaries(false)
+{
+ setAutoManage(autoManage);
+}
+
+KateSuperRangeList::KateSuperRangeList(const QPtrList<KateSuperRange>& rangeList, QObject* parent, const char* name)
+ : QObject(parent, name)
+ , m_autoManage(false)
+ , m_connect(false)
+ , m_trackingBoundaries(false)
+{
+ appendList(rangeList);
+}
+
+void KateSuperRangeList::appendList(const QPtrList<KateSuperRange>& rangeList)
+{
+ for (QPtrListIterator<KateSuperRange> it = rangeList; *it; ++it)
+ append(*it);
+}
+
+void KateSuperRangeList::clear()
+{
+ for (KateSuperRange* range = first(); range; range = next())
+ emit rangeEliminated(range);
+
+ QPtrList<KateSuperRange>::clear();
+}
+
+void KateSuperRangeList::connectAll()
+{
+ if (!m_connect) {
+ m_connect = true;
+ for (KateSuperRange* range = first(); range; range = next()) {
+ connect(range, SIGNAL(destroyed(QObject*)), SLOT(slotDeleted(QObject*)));
+ connect(range, SIGNAL(eliminated()), SLOT(slotEliminated()));
+ }
+ }
+}
+
+bool KateSuperRangeList::autoManage() const
+{
+ return m_autoManage;
+}
+
+void KateSuperRangeList::setAutoManage(bool autoManage)
+{
+ m_autoManage = autoManage;
+ setAutoDelete(m_autoManage);
+}
+
+QPtrList<KateSuperRange> KateSuperRangeList::rangesIncluding(const KateTextCursor& cursor)
+{
+ sort();
+
+ QPtrList<KateSuperRange> ret;
+
+ for (KateSuperRange* r = first(); r; r = next())
+ if (r->includes(cursor))
+ ret.append(r);
+
+ return ret;
+}
+
+QPtrList<KateSuperRange> KateSuperRangeList::rangesIncluding(uint line)
+{
+ sort();
+
+ QPtrList<KateSuperRange> ret;
+
+ for (KateSuperRange* r = first(); r; r = next())
+ if (r->includes(line))
+ ret.append(r);
+
+ return ret;
+}
+
+bool KateSuperRangeList::rangesInclude(const KateTextCursor& cursor)
+{
+ for (KateSuperRange* r = first(); r; r = next())
+ if (r->includes(cursor))
+ return true;
+
+ return false;
+}
+
+void KateSuperRangeList::slotEliminated()
+{
+ if (sender()) {
+ KateSuperRange* range = static_cast<KateSuperRange*>(const_cast<QObject*>(sender()));
+ emit rangeEliminated(range);
+
+ if (m_trackingBoundaries) {
+ m_columnBoundaries.removeRef(range->m_start);
+ m_columnBoundaries.removeRef(range->m_end);
+ }
+
+ if (m_autoManage)
+ removeRef(range);
+
+ if (!count())
+ emit listEmpty();
+ }
+}
+
+void KateSuperRangeList::slotDeleted(QObject* range)
+{
+ //kdDebug(13020)<<"KateSuperRangeList::slotDeleted"<<endl;
+ KateSuperRange* r = static_cast<KateSuperRange*>(range);
+
+ if (m_trackingBoundaries) {
+ m_columnBoundaries.removeRef(r->m_start);
+ m_columnBoundaries.removeRef(r->m_end);
+ }
+
+ int index = findRef(r);
+ if (index != -1)
+ take(index);
+ //else kdDebug(13020)<<"Range not found in list"<<endl;
+
+ if (!count())
+ emit listEmpty();
+}
+
+KateSuperCursor* KateSuperRangeList::firstBoundary(const KateTextCursor* start)
+{
+ if (!m_trackingBoundaries) {
+ m_trackingBoundaries = true;
+
+ for (KateSuperRange* r = first(); r; r = next()) {
+ m_columnBoundaries.append(&(r->superStart()));
+ m_columnBoundaries.append(&(r->superEnd()));
+ }
+ }
+
+ m_columnBoundaries.sort();
+
+ if (start)
+ // OPTIMISE: QMap with QPtrList for each line? (==> sorting issues :( )
+ for (KateSuperCursor* c = m_columnBoundaries.first(); c; c = m_columnBoundaries.next())
+ if (*start <= *c)
+ break;
+
+ return m_columnBoundaries.current();
+}
+
+KateSuperCursor* KateSuperRangeList::nextBoundary()
+{
+ KateSuperCursor* current = m_columnBoundaries.current();
+
+ // make sure the new cursor is after the current cursor; multiple cursors with the same position can be in the list.
+ if (current)
+ while (m_columnBoundaries.next())
+ if (*(m_columnBoundaries.current()) != *current)
+ break;
+
+ return m_columnBoundaries.current();
+}
+
+KateSuperCursor* KateSuperRangeList::currentBoundary()
+{
+ return m_columnBoundaries.current();
+}
+
+int KateSuperRangeList::compareItems(QPtrCollection::Item item1, QPtrCollection::Item item2)
+{
+ if (static_cast<KateSuperRange*>(item1)->superStart() == static_cast<KateSuperRange*>(item2)->superStart()) {
+ if (static_cast<KateSuperRange*>(item1)->superEnd() == static_cast<KateSuperRange*>(item2)->superEnd()) {
+ return 0;
+ } else {
+ return static_cast<KateSuperRange*>(item1)->superEnd() < static_cast<KateSuperRange*>(item2)->superEnd() ? -1 : 1;
+ }
+ }
+
+ return static_cast<KateSuperRange*>(item1)->superStart() < static_cast<KateSuperRange*>(item2)->superStart() ? -1 : 1;
+}
+
+QPtrCollection::Item KateSuperRangeList::newItem(QPtrCollection::Item d)
+{
+ if (m_connect) {
+ connect(static_cast<KateSuperRange*>(d), SIGNAL(destroyed(QObject*)), SLOT(slotDeleted(QObject*)));
+ connect(static_cast<KateSuperRange*>(d), SIGNAL(eliminated()), SLOT(slotEliminated()));
+ connect(static_cast<KateSuperRange*>(d), SIGNAL(tagRange(KateSuperRange*)), SIGNAL(tagRange(KateSuperRange*)));
+
+ // HACK HACK
+ static_cast<KateSuperRange*>(d)->slotTagRange();
+ }
+
+ if (m_trackingBoundaries) {
+ m_columnBoundaries.append(&(static_cast<KateSuperRange*>(d)->superStart()));
+ m_columnBoundaries.append(&(static_cast<KateSuperRange*>(d)->superEnd()));
+ }
+
+ return QPtrList<KateSuperRange>::newItem(d);
+}
+
+// kate: space-indent on; indent-width 2; replace-tabs on;