2007-06-18 13:02:34 +00:00
|
|
|
// Scintilla source code edit control
|
|
|
|
/** @file PositionCache.h
|
|
|
|
** Classes for caching layout information.
|
|
|
|
**/
|
|
|
|
// Copyright 1998-2007 by Neil Hodgson <neilh@scintilla.org>
|
|
|
|
// The License.txt file describes the conditions under which this software may be distributed.
|
|
|
|
|
|
|
|
#ifndef POSITIONCACHE_H
|
|
|
|
#define POSITIONCACHE_H
|
|
|
|
|
|
|
|
#ifdef SCI_NAMESPACE
|
|
|
|
namespace Scintilla {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static inline bool IsEOLChar(char ch) {
|
|
|
|
return (ch == '\r') || (ch == '\n');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*/
|
|
|
|
class LineLayout {
|
|
|
|
private:
|
|
|
|
friend class LineLayoutCache;
|
|
|
|
int *lineStarts;
|
|
|
|
int lenLineStarts;
|
|
|
|
/// Drawing is only performed for @a maxLineLength characters on each line.
|
|
|
|
int lineNumber;
|
|
|
|
bool inCache;
|
|
|
|
public:
|
|
|
|
enum { wrapWidthInfinite = 0x7ffffff };
|
|
|
|
int maxLineLength;
|
|
|
|
int numCharsInLine;
|
|
|
|
enum validLevel { llInvalid, llCheckTextAndStyle, llPositions, llLines } validity;
|
|
|
|
int xHighlightGuide;
|
|
|
|
bool highlightColumn;
|
|
|
|
int selStart;
|
|
|
|
int selEnd;
|
|
|
|
bool containsCaret;
|
|
|
|
int edgeColumn;
|
|
|
|
char *chars;
|
|
|
|
unsigned char *styles;
|
|
|
|
int styleBitsSet;
|
|
|
|
char *indicators;
|
|
|
|
int *positions;
|
|
|
|
char bracePreviousStyles[2];
|
|
|
|
|
|
|
|
// Hotspot support
|
|
|
|
int hsStart;
|
|
|
|
int hsEnd;
|
|
|
|
|
|
|
|
// Wrapped line support
|
|
|
|
int widthLine;
|
|
|
|
int lines;
|
2009-07-04 09:19:36 +00:00
|
|
|
int wrapIndent; // In pixels
|
2007-06-18 13:02:34 +00:00
|
|
|
|
|
|
|
LineLayout(int maxLineLength_);
|
|
|
|
virtual ~LineLayout();
|
|
|
|
void Resize(int maxLineLength_);
|
|
|
|
void Free();
|
|
|
|
void Invalidate(validLevel validity_);
|
|
|
|
int LineStart(int line) const;
|
|
|
|
int LineLastVisible(int line) const;
|
2007-12-02 11:58:17 +00:00
|
|
|
bool InLine(int offset, int line) const;
|
2007-06-18 13:02:34 +00:00
|
|
|
void SetLineStart(int line, int start);
|
|
|
|
void SetBracesHighlight(Range rangeLine, Position braces[],
|
|
|
|
char bracesMatchStyle, int xHighlight);
|
|
|
|
void RestoreBracesHighlight(Range rangeLine, Position braces[]);
|
|
|
|
int FindBefore(int x, int lower, int upper) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*/
|
|
|
|
class LineLayoutCache {
|
|
|
|
int level;
|
|
|
|
int length;
|
|
|
|
int size;
|
|
|
|
LineLayout **cache;
|
|
|
|
bool allInvalidated;
|
|
|
|
int styleClock;
|
|
|
|
int useCount;
|
|
|
|
void Allocate(int length_);
|
|
|
|
void AllocateForLevel(int linesOnScreen, int linesInDoc);
|
|
|
|
public:
|
|
|
|
LineLayoutCache();
|
|
|
|
virtual ~LineLayoutCache();
|
|
|
|
void Deallocate();
|
|
|
|
enum {
|
|
|
|
llcNone=SC_CACHE_NONE,
|
|
|
|
llcCaret=SC_CACHE_CARET,
|
|
|
|
llcPage=SC_CACHE_PAGE,
|
|
|
|
llcDocument=SC_CACHE_DOCUMENT
|
|
|
|
};
|
|
|
|
void Invalidate(LineLayout::validLevel validity_);
|
|
|
|
void SetLevel(int level_);
|
|
|
|
int GetLevel() { return level; }
|
|
|
|
LineLayout *Retrieve(int lineNumber, int lineCaret, int maxChars, int styleClock_,
|
|
|
|
int linesOnScreen, int linesInDoc);
|
|
|
|
void Dispose(LineLayout *ll);
|
|
|
|
};
|
|
|
|
|
|
|
|
class PositionCacheEntry {
|
|
|
|
unsigned int styleNumber:8;
|
|
|
|
unsigned int len:8;
|
|
|
|
unsigned int clock:16;
|
|
|
|
short *positions;
|
|
|
|
public:
|
|
|
|
PositionCacheEntry();
|
|
|
|
~PositionCacheEntry();
|
|
|
|
void Set(unsigned int styleNumber_, const char *s_, unsigned int len_, int *positions_, unsigned int clock);
|
|
|
|
void Clear();
|
|
|
|
bool Retrieve(unsigned int styleNumber_, const char *s_, unsigned int len_, int *positions_) const;
|
|
|
|
static int Hash(unsigned int styleNumber, const char *s, unsigned int len);
|
|
|
|
bool NewerThan(const PositionCacheEntry &other);
|
|
|
|
void ResetClock();
|
|
|
|
};
|
|
|
|
|
|
|
|
// Class to break a line of text into shorter runs at sensible places.
|
|
|
|
class BreakFinder {
|
|
|
|
// If a whole run is longer than lengthStartSubdivision then subdivide
|
|
|
|
// into smaller runs at spaces or punctuation.
|
|
|
|
enum { lengthStartSubdivision = 300 };
|
|
|
|
// Try to make each subdivided run lengthEachSubdivision or shorter.
|
|
|
|
enum { lengthEachSubdivision = 100 };
|
|
|
|
LineLayout *ll;
|
|
|
|
int lineStart;
|
|
|
|
int lineEnd;
|
|
|
|
int posLineStart;
|
2007-12-02 11:58:17 +00:00
|
|
|
bool utf8;
|
2007-06-18 13:02:34 +00:00
|
|
|
int nextBreak;
|
2007-12-02 11:58:17 +00:00
|
|
|
int *selAndEdge;
|
|
|
|
unsigned int saeSize;
|
2007-06-18 13:02:34 +00:00
|
|
|
unsigned int saeLen;
|
|
|
|
unsigned int saeCurrentPos;
|
|
|
|
int saeNext;
|
|
|
|
int subBreak;
|
|
|
|
void Insert(int val);
|
|
|
|
public:
|
2007-12-02 11:58:17 +00:00
|
|
|
BreakFinder(LineLayout *ll_, int lineStart_, int lineEnd_, int posLineStart_, bool utf8_, int xStart);
|
|
|
|
~BreakFinder();
|
2007-06-18 13:02:34 +00:00
|
|
|
int First();
|
|
|
|
int Next();
|
|
|
|
};
|
|
|
|
|
|
|
|
class PositionCache {
|
|
|
|
PositionCacheEntry *pces;
|
|
|
|
size_t size;
|
|
|
|
unsigned int clock;
|
|
|
|
bool allClear;
|
|
|
|
public:
|
|
|
|
PositionCache();
|
|
|
|
~PositionCache();
|
|
|
|
void Clear();
|
|
|
|
void SetSize(size_t size_);
|
|
|
|
int GetSize() { return size; }
|
|
|
|
void MeasureWidths(Surface *surface, ViewStyle &vstyle, unsigned int styleNumber,
|
|
|
|
const char *s, unsigned int len, int *positions);
|
|
|
|
};
|
|
|
|
|
|
|
|
inline bool IsSpaceOrTab(int ch) {
|
|
|
|
return ch == ' ' || ch == '\t';
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SCI_NAMESPACE
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|