File indexing completed on 2024-09-08 03:40:38

0001 /*
0002     SPDX-FileCopyrightText: 2013 Christoph Cullmann <cullmann@kde.org>
0003 
0004     SPDX-License-Identifier: LGPL-2.0-or-later
0005 */
0006 
0007 #ifndef KATE_TEXTFOLDING_H
0008 #define KATE_TEXTFOLDING_H
0009 
0010 #include <ktexteditor_export.h>
0011 
0012 #include "ktexteditor/range.h"
0013 
0014 #include <QFlags>
0015 #include <QJsonArray>
0016 #include <QJsonDocument>
0017 #include <QObject>
0018 
0019 #include <functional>
0020 
0021 namespace Kate
0022 {
0023 class TextBuffer;
0024 class TextCursor;
0025 
0026 /**
0027  * Class representing the folding information for a TextBuffer.
0028  * The interface allows to arbitrary fold given regions of a buffer as long
0029  * as they are well nested.
0030  * Multiple instances of this class can exist for the same buffer.
0031  */
0032 class KTEXTEDITOR_EXPORT TextFolding : public QObject
0033 {
0034     Q_OBJECT
0035 
0036 public:
0037     /**
0038      * Create folding object for given buffer.
0039      * @param buffer text buffer we want to provide folding info for
0040      */
0041     explicit TextFolding(TextBuffer &buffer);
0042 
0043     /**
0044      * Cleanup
0045      */
0046     ~TextFolding() override;
0047 
0048     /**
0049      * Folding state of a range
0050      */
0051     enum FoldingRangeFlag {
0052         /**
0053          * Range is persistent, e.g. it should not auto-delete after unfolding!
0054          */
0055         Persistent = 0x1,
0056 
0057         /**
0058          * Range is folded away
0059          */
0060         Folded = 0x2
0061     };
0062     Q_DECLARE_FLAGS(FoldingRangeFlags, FoldingRangeFlag)
0063 
0064     /**
0065      * Create a new folding range.
0066      * @param range folding range
0067      * @param flags initial flags for the new folding range
0068      * @return on success, id of new range >= 0, else -1, we return no pointer as folding ranges might be auto-deleted internally!
0069      *         the ids are stable for one Kate::TextFolding, e.g. you can rely in unit tests that you get 0,1,.... for successfully created ranges!
0070      */
0071     qint64 newFoldingRange(KTextEditor::Range range, FoldingRangeFlags flags = FoldingRangeFlags());
0072 
0073     /**
0074      * Returns the folding range associated with @p id.
0075      * If @p id is not a valid id, the returned range matches KTextEditor::Range::invalid().
0076      * @note This works for either persistent ranges or folded ranges.
0077      *       Note, that the highlighting does not add folds unless text is folded.
0078      *
0079      * @return the folding range for @p id
0080      */
0081     KTextEditor::Range foldingRange(qint64 id) const;
0082 
0083     /**
0084      * Fold the given range.
0085      * @param id id of the range to fold
0086      * @return success
0087      */
0088     bool foldRange(qint64 id);
0089 
0090     /**
0091      * Unfold the given range.
0092      * In addition it can be forced to remove the region, even if it is persistent.
0093      * @param id id of the range to unfold
0094      * @param remove should the range be removed from the folding after unfolding? ranges that are not persistent auto-remove themself on unfolding
0095      * @return success
0096      */
0097     bool unfoldRange(qint64 id, bool remove = false);
0098 
0099     /**
0100      * Query if a given line is visible.
0101      * Very fast, if nothing is folded, else does binary search
0102      * log(n) for n == number of folded ranges
0103      * @param line real line to query
0104      * @param foldedRangeId if the line is not visible and that pointer is not 0, will be filled with id of range hiding the line or -1
0105      * @return is that line visible?
0106      */
0107     bool isLineVisible(int line, qint64 *foldedRangeId = nullptr) const;
0108 
0109     /**
0110      * Ensure that a given line will be visible.
0111      * Potentially unfold recursively all folds hiding this line, else just returns.
0112      * @param line line to make visible
0113      */
0114     void ensureLineIsVisible(int line);
0115 
0116     /**
0117      * Query number of visible lines.
0118      * Very fast, if nothing is folded, else walks over all folded regions
0119      * O(n) for n == number of folded ranges
0120      */
0121     int visibleLines() const;
0122 
0123     /**
0124      * Convert a text buffer line to a visible line number.
0125      * Very fast, if nothing is folded, else walks over all folded regions
0126      * O(n) for n == number of folded ranges
0127      * @param line line index in the text buffer
0128      * @return index in visible lines
0129      */
0130     int lineToVisibleLine(int line) const;
0131 
0132     /**
0133      * Convert a visible line number to a line number in the text buffer.
0134      * Very fast, if nothing is folded, else walks over all folded regions
0135      * O(n) for n == number of folded ranges
0136      * @param visibleLine visible line index
0137      * @return index in text buffer lines
0138      */
0139     int visibleLineToLine(int visibleLine) const;
0140 
0141     /**
0142      * Queries which folding ranges start at the given line and returns the id + flags for all
0143      * of them. Very fast if nothing is folded, else binary search.
0144      * @param line line to query starting folding ranges
0145      * @return vector of id's + flags
0146      */
0147     QList<QPair<qint64, FoldingRangeFlags>> foldingRangesStartingOnLine(int line) const;
0148 
0149     /**
0150      * Query child folding ranges for given range id. To query the toplevel
0151      * ranges pass id -1
0152      * @param parentRangeId id of parent range, pass -1 to query top level ranges
0153      * @return vector of id's + flags for child ranges
0154      */
0155     QList<QPair<qint64, FoldingRangeFlags>> foldingRangesForParentRange(qint64 parentRangeId = -1) const;
0156 
0157     /**
0158      * Return the current known folding ranges a QJsonDocument to store in configs.
0159      * @return current folds as variant list
0160      */
0161     QJsonDocument exportFoldingRanges() const;
0162 
0163     /**
0164      * Import the folding ranges given as a QJsonDocument like read from configs.
0165      * @param folds list of folds to import
0166      */
0167     void importFoldingRanges(const QJsonDocument &folds);
0168 
0169     /**
0170      * Dump folding state as string, for unit testing and debugging
0171      * @return current state as text
0172      */
0173     QString debugDump() const;
0174 
0175     /**
0176      * Print state to stdout for testing
0177      */
0178     void debugPrint(const QString &title) const;
0179 
0180     void editEnd(int startLine, int endLine, std::function<bool(int)> isLineFoldingStart);
0181 
0182 public Q_SLOTS:
0183     /**
0184      * Clear the complete folding.
0185      * This is automatically triggered if the buffer is cleared.
0186      */
0187     void clear();
0188 
0189 Q_SIGNALS:
0190     /**
0191      * If the folding state of existing ranges changes or
0192      * ranges are added/removed, this signal is emitted.
0193      */
0194     void foldingRangesChanged();
0195 
0196 private:
0197     /**
0198      * Data holder for text folding range and its nested children
0199      */
0200     class KTEXTEDITOR_NO_EXPORT FoldingRange
0201     {
0202     public:
0203         /**
0204          * Construct new one
0205          * @param buffer text buffer to use
0206          * @param range folding range
0207          * @param flags flags for the new folding range
0208          */
0209         FoldingRange(TextBuffer &buffer, KTextEditor::Range range, FoldingRangeFlags flags);
0210 
0211         /**
0212          * Cleanup
0213          */
0214         ~FoldingRange();
0215 
0216         FoldingRange(const FoldingRange &) = delete;
0217         FoldingRange &operator=(const FoldingRange &) = delete;
0218 
0219         /**
0220          * Vector of range pointers
0221          */
0222         typedef QList<FoldingRange *> Vector;
0223 
0224         /**
0225          * start moving cursor
0226          * NO range to be more efficient
0227          */
0228         Kate::TextCursor *start;
0229 
0230         /**
0231          * end moving cursor
0232          * NO range to be more efficient
0233          */
0234         Kate::TextCursor *end;
0235 
0236         /**
0237          * parent range, if any
0238          */
0239         FoldingRange *parent;
0240 
0241         /**
0242          * nested ranges, if any
0243          * this will always be sorted and non-overlapping
0244          * nested ranges are inside these ranges
0245          */
0246         FoldingRange::Vector nestedRanges;
0247 
0248         /**
0249          * Folding range flags
0250          */
0251         FoldingRangeFlags flags;
0252 
0253         /**
0254          * id of this range
0255          */
0256         qint64 id;
0257     };
0258 
0259     /**
0260      * Clear all folding range collections but leave global id counter intact.
0261      */
0262     KTEXTEDITOR_NO_EXPORT
0263     void clearFoldingRanges();
0264 
0265     /**
0266      * Fill known folding ranges in a QVariantList to store in configs.
0267      * @param ranges ranges vector to dump
0268      * @param folds current folds as variant list, will be filled
0269      */
0270     KTEXTEDITOR_NO_EXPORT
0271     static void exportFoldingRanges(const TextFolding::FoldingRange::Vector &ranges, QJsonArray &folds);
0272 
0273     /**
0274      * Dump folding state of given vector as string, for unit testing and debugging.
0275      * Will recurse if wanted.
0276      * @param ranges ranges vector to dump
0277      * @param recurse recurse to nestedRanges?
0278      * @return current state as text
0279      */
0280     KTEXTEDITOR_NO_EXPORT
0281     static QString debugDump(const TextFolding::FoldingRange::Vector &ranges, bool recurse);
0282 
0283     /**
0284      * Helper to insert folding range into existing ones.
0285      * Might fail, if not correctly nested.
0286      * Then the outside must take care of the passed pointer, e.g. delete it.
0287      * Will sanitize the ranges vectors, purge invalid/empty ranges.
0288      * @param parent parent folding range if any
0289      * @param existingRanges ranges into which we want to insert the new one
0290      * @param newRange new folding range
0291      * @return success, if false, newRange should be deleted afterwards, else it is registered internally
0292      */
0293     KTEXTEDITOR_NO_EXPORT
0294     bool insertNewFoldingRange(FoldingRange *parent, TextFolding::FoldingRange::Vector &existingRanges, TextFolding::FoldingRange *newRange);
0295 
0296     /**
0297      * Helper to update the folded ranges if we insert a new range into the tree.
0298      * @param newRange new folding range that was inserted, will already contain its new nested ranges, if any!
0299      * @return any updated done? if yes, the foldingRangesChanged() signal got emitted!
0300      */
0301     KTEXTEDITOR_NO_EXPORT
0302     bool updateFoldedRangesForNewRange(TextFolding::FoldingRange *newRange);
0303 
0304     /**
0305      * Helper to update the folded ranges if we remove a new range from the tree.
0306      * @param oldRange new folding range that is removed, will still contain its new nested ranges, if any!
0307      * @return any updated done? if yes, the foldingRangesChanged() signal got emitted!
0308      */
0309     KTEXTEDITOR_NO_EXPORT
0310     bool updateFoldedRangesForRemovedRange(TextFolding::FoldingRange *oldRange);
0311 
0312     /**
0313      * Helper to append recursively topmost folded ranges from input to output vector.
0314      * @param newFoldedFoldingRanges output vector for folded ranges
0315      * @param ranges input vector to search recursively folded ranges inside
0316      */
0317     KTEXTEDITOR_NO_EXPORT
0318     void appendFoldedRanges(TextFolding::FoldingRange::Vector &newFoldedFoldingRanges, const TextFolding::FoldingRange::Vector &ranges) const;
0319 
0320     /**
0321      * Compare two ranges by their start cursor.
0322      * @param a first range
0323      * @param b second range
0324      */
0325     KTEXTEDITOR_NO_EXPORT
0326     static bool compareRangeByStart(FoldingRange *a, FoldingRange *b);
0327 
0328     /**
0329      * Compare two ranges by their end cursor.
0330      * @param a first range
0331      * @param b second range
0332      */
0333     KTEXTEDITOR_NO_EXPORT
0334     static bool compareRangeByEnd(FoldingRange *a, FoldingRange *b);
0335 
0336     /**
0337      * Compare range start with line
0338      * @param line line
0339      * @param range range
0340      */
0341     KTEXTEDITOR_NO_EXPORT
0342     static bool compareRangeByStartWithLine(int line, FoldingRange *range);
0343 
0344     /**
0345      * Compare range start with line
0346      * @param range range
0347      * @param line line
0348      */
0349     KTEXTEDITOR_NO_EXPORT
0350     static bool compareRangeByLineWithStart(FoldingRange *range, int line);
0351 
0352     /**
0353      * Internal helper that queries which folding ranges start at the given line and returns the id + flags for all
0354      * of them. Will recursively dive down starting with given vector
0355      * @param results vector that is filled with id's + flags
0356      * @param ranges ranges vector to search in
0357      * @param line line to query starting folding ranges
0358      */
0359     KTEXTEDITOR_NO_EXPORT
0360     void foldingRangesStartingOnLine(QList<QPair<qint64, FoldingRangeFlags>> &results, const TextFolding::FoldingRange::Vector &ranges, int line) const;
0361 
0362 private:
0363     /**
0364      * parent text buffer
0365      * is a reference, and no pointer, as this must always exist and can't change
0366      * can't be const, as we create text cursors!
0367      */
0368     TextBuffer &m_buffer;
0369 
0370     /**
0371      * toplevel folding ranges
0372      * this will always be sorted and non-overlapping
0373      * nested ranges are inside these ranges
0374      */
0375     FoldingRange::Vector m_foldingRanges;
0376 
0377     /**
0378      * folded folding ranges
0379      * this is a sorted vector of ranges
0380      * all non-overlapping
0381      */
0382     FoldingRange::Vector m_foldedFoldingRanges;
0383 
0384     /**
0385      * global id counter for the created ranges
0386      */
0387     qint64 m_idCounter;
0388 
0389     /**
0390      * mapping: id => range
0391      */
0392     QHash<qint64, FoldingRange *> m_idToFoldingRange;
0393 };
0394 
0395 Q_DECLARE_OPERATORS_FOR_FLAGS(TextFolding::FoldingRangeFlags)
0396 
0397 }
0398 
0399 #endif