forked from qt-creator/qt-creator
Add folding for files and chunks. Change-Id: I76476351e88f0b3e71e3cccbca0fa17b02c26226 Reviewed-by: Friedemann Kleint <Friedemann.Kleint@qt.io> Reviewed-by: David Schulz <david.schulz@qt.io>
226 lines
8.2 KiB
C++
226 lines
8.2 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** This file is part of Qt Creator.
|
|
**
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3 as published by the Free Software
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
** included in the packaging of this file. Please review the following
|
|
** information to ensure the GNU General Public License requirements will
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#pragma once
|
|
|
|
#include "texteditor_global.h"
|
|
|
|
#include "textmark.h"
|
|
#include "textdocument.h"
|
|
|
|
#include <QTextBlockUserData>
|
|
#include <QPlainTextDocumentLayout>
|
|
|
|
namespace TextEditor {
|
|
struct Parenthesis;
|
|
typedef QVector<Parenthesis> Parentheses;
|
|
|
|
struct TEXTEDITOR_EXPORT Parenthesis
|
|
{
|
|
enum Type : char { Opened, Closed };
|
|
|
|
inline Parenthesis() : pos(-1), type(Opened) {}
|
|
inline Parenthesis(Type t, QChar c, int position)
|
|
: pos(position), chr(c), type(t) {}
|
|
int pos;
|
|
QChar chr;
|
|
Type type;
|
|
};
|
|
|
|
class TEXTEDITOR_EXPORT CodeFormatterData
|
|
{
|
|
public:
|
|
virtual ~CodeFormatterData();
|
|
};
|
|
|
|
class TEXTEDITOR_EXPORT TextBlockUserData : public QTextBlockUserData
|
|
{
|
|
public:
|
|
|
|
inline TextBlockUserData()
|
|
: m_foldingIndent(0)
|
|
, m_folded(false)
|
|
, m_ifdefedOut(false)
|
|
, m_lexerState(0)
|
|
, m_foldingStartIncluded(false)
|
|
, m_foldingEndIncluded(false)
|
|
, m_codeFormatterData(0)
|
|
{}
|
|
~TextBlockUserData();
|
|
|
|
inline TextMarks marks() const { return m_marks; }
|
|
void addMark(TextMark *mark);
|
|
inline bool removeMark(TextMark *mark) { return m_marks.removeAll(mark); }
|
|
|
|
inline TextMarks documentClosing() {
|
|
TextMarks marks = m_marks;
|
|
foreach (TextMark *mrk, m_marks)
|
|
mrk->setBaseTextDocument(0);
|
|
m_marks.clear();
|
|
return marks;
|
|
}
|
|
|
|
inline void setFolded(bool b) { m_folded = b; }
|
|
inline bool folded() const { return m_folded; }
|
|
|
|
inline void setParentheses(const Parentheses &parentheses) { m_parentheses = parentheses; }
|
|
inline void clearParentheses() { m_parentheses.clear(); }
|
|
inline const Parentheses &parentheses() const { return m_parentheses; }
|
|
inline bool hasParentheses() const { return !m_parentheses.isEmpty(); }
|
|
int braceDepthDelta() const;
|
|
|
|
inline bool setIfdefedOut() { bool result = m_ifdefedOut; m_ifdefedOut = true; return !result; }
|
|
inline bool clearIfdefedOut() { bool result = m_ifdefedOut; m_ifdefedOut = false; return result;}
|
|
inline bool ifdefedOut() const { return m_ifdefedOut; }
|
|
|
|
|
|
enum MatchType { NoMatch, Match, Mismatch };
|
|
static MatchType checkOpenParenthesis(QTextCursor *cursor, QChar c);
|
|
static MatchType checkClosedParenthesis(QTextCursor *cursor, QChar c);
|
|
static MatchType matchCursorBackward(QTextCursor *cursor);
|
|
static MatchType matchCursorForward(QTextCursor *cursor);
|
|
static bool findPreviousOpenParenthesis(QTextCursor *cursor, bool select = false,
|
|
bool onlyInCurrentBlock = false);
|
|
static bool findNextClosingParenthesis(QTextCursor *cursor, bool select = false);
|
|
|
|
static bool findPreviousBlockOpenParenthesis(QTextCursor *cursor, bool checkStartPosition = false);
|
|
static bool findNextBlockClosingParenthesis(QTextCursor *cursor);
|
|
|
|
// Get the code folding level
|
|
inline int foldingIndent() const { return m_foldingIndent; }
|
|
/* Set the code folding level.
|
|
*
|
|
* A code folding marker will appear the line *before* the one where the indention
|
|
* level increases. The code folding reagion will end in the last line that has the same
|
|
* indention level (or higher).
|
|
*/
|
|
inline void setFoldingIndent(int indent) { m_foldingIndent = indent; }
|
|
// Set whether the first character of the folded region will show when the code is folded.
|
|
inline void setFoldingStartIncluded(bool included) { m_foldingStartIncluded = included; }
|
|
inline bool foldingStartIncluded() const { return m_foldingStartIncluded; }
|
|
// Set whether the last character of the folded region will show when the code is folded.
|
|
inline void setFoldingEndIncluded(bool included) { m_foldingEndIncluded = included; }
|
|
inline bool foldingEndIncluded() const { return m_foldingEndIncluded; }
|
|
inline int lexerState() const { return m_lexerState; }
|
|
inline void setLexerState(int state) {m_lexerState = state; }
|
|
|
|
|
|
CodeFormatterData *codeFormatterData() const { return m_codeFormatterData; }
|
|
void setCodeFormatterData(CodeFormatterData *data);
|
|
|
|
private:
|
|
TextMarks m_marks;
|
|
int m_foldingIndent : 16;
|
|
uint m_folded : 1;
|
|
uint m_ifdefedOut : 1;
|
|
uint m_lexerState : 8;
|
|
uint m_foldingStartIncluded : 1;
|
|
uint m_foldingEndIncluded : 1;
|
|
Parentheses m_parentheses;
|
|
CodeFormatterData *m_codeFormatterData;
|
|
};
|
|
|
|
|
|
class TEXTEDITOR_EXPORT TextDocumentLayout : public QPlainTextDocumentLayout
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
TextDocumentLayout(QTextDocument *doc);
|
|
~TextDocumentLayout();
|
|
|
|
static void setParentheses(const QTextBlock &block, const Parentheses &parentheses);
|
|
static void clearParentheses(const QTextBlock &block) { setParentheses(block, Parentheses());}
|
|
static Parentheses parentheses(const QTextBlock &block);
|
|
static bool hasParentheses(const QTextBlock &block);
|
|
static bool setIfdefedOut(const QTextBlock &block);
|
|
static bool clearIfdefedOut(const QTextBlock &block);
|
|
static bool ifdefedOut(const QTextBlock &block);
|
|
static int braceDepthDelta(const QTextBlock &block);
|
|
static int braceDepth(const QTextBlock &block);
|
|
static void setBraceDepth(QTextBlock &block, int depth);
|
|
static void changeBraceDepth(QTextBlock &block, int delta);
|
|
static void setFoldingIndent(const QTextBlock &block, int indent);
|
|
static int foldingIndent(const QTextBlock &block);
|
|
static void setLexerState(const QTextBlock &block, int state);
|
|
static int lexerState(const QTextBlock &block);
|
|
static void changeFoldingIndent(QTextBlock &block, int delta);
|
|
static bool canFold(const QTextBlock &block);
|
|
static void doFoldOrUnfold(const QTextBlock& block, bool unfold);
|
|
static bool isFolded(const QTextBlock &block);
|
|
static void setFolded(const QTextBlock &block, bool folded);
|
|
|
|
class TEXTEDITOR_EXPORT FoldValidator
|
|
{
|
|
public:
|
|
FoldValidator();
|
|
|
|
void setup(TextDocumentLayout *layout);
|
|
void reset();
|
|
void process(QTextBlock block);
|
|
void finalize();
|
|
|
|
private:
|
|
TextDocumentLayout *m_layout;
|
|
bool m_requestDocUpdate;
|
|
int m_insideFold;
|
|
};
|
|
|
|
static TextBlockUserData *testUserData(const QTextBlock &block) {
|
|
return static_cast<TextBlockUserData*>(block.userData());
|
|
}
|
|
static TextBlockUserData *userData(const QTextBlock &block) {
|
|
TextBlockUserData *data = static_cast<TextBlockUserData*>(block.userData());
|
|
if (!data && block.isValid())
|
|
const_cast<QTextBlock &>(block).setUserData((data = new TextBlockUserData));
|
|
return data;
|
|
}
|
|
|
|
void requestExtraAreaUpdate();
|
|
|
|
void emitDocumentSizeChanged() { emit documentSizeChanged(documentSize()); }
|
|
|
|
int lastSaveRevision;
|
|
bool hasMarks;
|
|
double maxMarkWidthFactor;
|
|
|
|
int m_requiredWidth;
|
|
|
|
void setRequiredWidth(int width);
|
|
|
|
QSizeF documentSize() const;
|
|
|
|
TextMarks documentClosing();
|
|
void documentReloaded(TextMarks marks, TextDocument *baseextDocument);
|
|
void updateMarksLineNumber();
|
|
void updateMarksBlock(const QTextBlock &block);
|
|
|
|
signals:
|
|
void updateExtraArea();
|
|
void foldChanged(const int blockNumber, bool folded);
|
|
};
|
|
|
|
} // namespace TextEditor
|