1
// Scintilla source code edit control
3
** Text document that handles notifications, DBCS, styling, words and end of line.
5
// Copyright 1998-2003 by Neil Hodgson <neilh@scintilla.org>
6
// The License.txt file describes the conditions under which this software may be distributed.
16
* A Position is a position within a document between two characters or at the beginning or end.
17
* Sometimes used as a character index where it identifies the character after the position.
20
const Position invalidPosition = -1;
23
* The range class represents a range of text in a document.
24
* The two values are not sorted as one end may be more significant than the other
25
* as is the case for the selection where the end position is the position of the caret.
26
* If either position is invalidPosition then the range is invalid and most operations will fail.
33
Range(Position pos=0) :
34
start(pos), end(pos) {
36
Range(Position start_, Position end_) :
37
start(start_), end(end_) {
41
return (start != invalidPosition) && (end != invalidPosition);
44
// Is the position within the range?
45
bool Contains(Position pos) const {
47
return (pos >= start && pos <= end);
49
return (pos <= start && pos >= end);
53
// Is the character after pos within the range?
54
bool ContainsCharacter(Position pos) const {
56
return (pos >= start && pos < end);
58
return (pos < start && pos >= end);
62
bool Contains(Range other) const {
63
return Contains(other.start) && Contains(other.end);
66
bool Overlaps(Range other) const {
68
Contains(other.start) ||
69
Contains(other.end) ||
70
other.Contains(start) ||
76
class DocModification;
80
* Interface class for regular expression searching
82
class RegexSearchBase {
84
virtual ~RegexSearchBase() {}
86
virtual long FindText(Document *doc, int minPos, int maxPos, const char *s,
87
bool caseSensitive, bool word, bool wordStart, int flags, int *length) = 0;
89
///@return String with the substitutions, must remain valid until the next call or destruction
90
virtual const char *SubstituteByPosition(Document *doc, const char *text, int *length) = 0;
93
/// Factory function for RegexSearchBase
94
extern RegexSearchBase *CreateRegexSearch(CharClassify *charClassTable);
101
const unsigned char *styles;
102
StyledText(size_t length_, const char *text_, bool multipleStyles_, int style_, const unsigned char *styles_) :
103
length(length_), text(text_), multipleStyles(multipleStyles_), style(style_), styles(styles_) {
105
// Return number of bytes from start to before '\n' or end of text.
106
// Return 1 when start is outside text
107
size_t LineLength(size_t start) const {
109
while ((cur < length) && (text[cur] != '\n'))
113
size_t StyleAt(size_t i) const {
114
return multipleStyles ? styles[i] : style;
120
virtual ~CaseFolder() {
122
virtual size_t Fold(char *folded, size_t sizeFolded, const char *mixed, size_t lenMixed) = 0;
125
class CaseFolderTable : public CaseFolder {
130
virtual ~CaseFolderTable();
131
virtual size_t Fold(char *folded, size_t sizeFolded, const char *mixed, size_t lenMixed);
132
void SetTranslation(char ch, char chTranslation);
133
void StandardASCII();
142
bool performingStyle; ///< Prevent reentrance
144
LexInterface(Document *pdoc_) : pdoc(pdoc_), instance(0), performingStyle(false) {
146
virtual ~LexInterface() {
148
void Colourise(int start, int end);
149
bool UseContainerLexing() const {
150
return instance == 0;
156
class Document : PerLine, public IDocument {
159
/** Used to pair watcher pointer with user data. */
160
class WatcherWithUserData {
164
WatcherWithUserData() {
170
enum charClassification { ccSpace, ccNewLine, ccWord, ccPunctuation };
174
CharClassify charClass;
178
int enteredModification;
180
int enteredReadOnlyCount;
182
WatcherWithUserData *watchers;
185
// ldSize is not real data - it is for dimensions and loops
186
enum lineData { ldMarkers, ldLevels, ldState, ldMargin, ldAnnotation, ldSize };
187
PerLine *perLineData[ldSize];
190
RegexSearchBase *regex;
200
/// Can also be SC_CP_UTF8 to enable UTF-8 mode
204
int actualIndentInChars;
207
bool backspaceUnindents;
209
DecorationList decorations;
218
virtual void InsertLine(int line);
219
virtual void RemoveLine(int line);
221
int SCI_METHOD Version() const {
225
void SCI_METHOD SetErrorStatus(int status);
227
int SCI_METHOD LineFromPosition(int pos) const;
228
int ClampPositionIntoDocument(int pos);
229
bool IsCrLf(int pos);
230
int LenChar(int pos);
231
bool InGoodUTF8(int pos, int &start, int &end) const;
232
int MovePositionOutsideChar(int pos, int moveDir, bool checkLineEnd=true);
233
int NextPosition(int pos, int moveDir) const;
234
bool NextCharacter(int &pos, int moveDir); // Returns true if pos changed
235
int SCI_METHOD CodePage() const;
236
bool SCI_METHOD IsDBCSLeadByte(char ch) const;
238
// Gateways to modifying document
239
void ModifiedAt(int pos);
240
void CheckReadOnly();
241
bool DeleteChars(int pos, int len);
242
bool InsertString(int position, const char *s, int insertLength);
245
bool CanUndo() { return cb.CanUndo(); }
246
bool CanRedo() { return cb.CanRedo(); }
247
void DeleteUndoHistory() { cb.DeleteUndoHistory(); }
248
bool SetUndoCollection(bool collectUndo) {
249
return cb.SetUndoCollection(collectUndo);
251
bool IsCollectingUndo() { return cb.IsCollectingUndo(); }
252
void BeginUndoAction() { cb.BeginUndoAction(); }
253
void EndUndoAction() { cb.EndUndoAction(); }
254
void AddUndoAction(int token, bool mayCoalesce) { cb.AddUndoAction(token, mayCoalesce); }
256
bool IsSavePoint() { return cb.IsSavePoint(); }
257
const char * SCI_METHOD BufferPointer() { return cb.BufferPointer(); }
259
int SCI_METHOD GetLineIndentation(int line);
260
void SetLineIndentation(int line, int indent);
261
int GetLineIndentPosition(int line) const;
262
int GetColumn(int position);
263
int FindColumn(int line, int column);
264
void Indent(bool forwards, int lineBottom, int lineTop);
265
static char *TransformLineEnds(int *pLenOut, const char *s, size_t len, int eolMode);
266
void ConvertLineEnds(int eolModeSet);
267
void SetReadOnly(bool set) { cb.SetReadOnly(set); }
268
bool IsReadOnly() { return cb.IsReadOnly(); }
270
bool InsertChar(int pos, char ch);
271
bool InsertCString(int position, const char *s);
272
void ChangeChar(int pos, char ch);
273
void DelChar(int pos);
274
void DelCharBack(int pos);
276
char CharAt(int position) { return cb.CharAt(position); }
277
void SCI_METHOD GetCharRange(char *buffer, int position, int lengthRetrieve) const {
278
cb.GetCharRange(buffer, position, lengthRetrieve);
280
char SCI_METHOD StyleAt(int position) const { return cb.StyleAt(position); }
281
void GetStyleRange(unsigned char *buffer, int position, int lengthRetrieve) const {
282
cb.GetStyleRange(buffer, position, lengthRetrieve);
284
int GetMark(int line);
285
int AddMark(int line, int markerNum);
286
void AddMarkSet(int line, int valueSet);
287
void DeleteMark(int line, int markerNum);
288
void DeleteMarkFromHandle(int markerHandle);
289
void DeleteAllMarks(int markerNum);
290
int LineFromHandle(int markerHandle);
291
int SCI_METHOD LineStart(int line) const;
292
int LineEnd(int line) const;
293
int LineEndPosition(int position) const;
294
bool IsLineEndPosition(int position) const;
295
int VCHomePosition(int position) const;
297
int SCI_METHOD SetLevel(int line, int level);
298
int SCI_METHOD GetLevel(int line) const;
300
int GetLastChild(int lineParent, int level=-1);
301
int GetFoldParent(int line);
303
void Indent(bool forwards);
304
int ExtendWordSelect(int pos, int delta, bool onlyWordCharacters=false);
305
int NextWordStart(int pos, int delta);
306
int NextWordEnd(int pos, int delta);
307
int SCI_METHOD Length() const { return cb.Length(); }
308
void Allocate(int newSize) { cb.Allocate(newSize); }
309
size_t ExtractChar(int pos, char *bytes);
310
bool MatchesWordOptions(bool word, bool wordStart, int pos, int length);
311
long FindText(int minPos, int maxPos, const char *search, bool caseSensitive, bool word,
312
bool wordStart, bool regExp, int flags, int *length, CaseFolder *pcf);
313
const char *SubstituteByPosition(const char *text, int *length);
314
int LinesTotal() const;
316
void ChangeCase(Range r, bool makeUpperCase);
318
void SetDefaultCharClasses(bool includeWordClass);
319
void SetCharClasses(const unsigned char *chars, CharClassify::cc newCharClass);
320
void SetStylingBits(int bits);
321
void SCI_METHOD StartStyling(int position, char mask);
322
bool SCI_METHOD SetStyleFor(int length, char style);
323
bool SCI_METHOD SetStyles(int length, const char *styles);
324
int GetEndStyled() { return endStyled; }
325
void EnsureStyledTo(int pos);
327
int GetStyleClock() { return styleClock; }
328
void IncrementStyleClock();
329
void SCI_METHOD DecorationSetCurrentIndicator(int indicator) {
330
decorations.SetCurrentIndicator(indicator);
332
void SCI_METHOD DecorationFillRange(int position, int value, int fillLength);
334
int SCI_METHOD SetLineState(int line, int state);
335
int SCI_METHOD GetLineState(int line) const;
336
int GetMaxLineState();
337
void SCI_METHOD ChangeLexerState(int start, int end);
339
StyledText MarginStyledText(int line);
340
void MarginSetStyle(int line, int style);
341
void MarginSetStyles(int line, const unsigned char *styles);
342
void MarginSetText(int line, const char *text);
343
int MarginLength(int line) const;
344
void MarginClearAll();
346
bool AnnotationAny() const;
347
StyledText AnnotationStyledText(int line);
348
void AnnotationSetText(int line, const char *text);
349
void AnnotationSetStyle(int line, int style);
350
void AnnotationSetStyles(int line, const unsigned char *styles);
351
int AnnotationLength(int line) const;
352
int AnnotationLines(int line) const;
353
void AnnotationClearAll();
355
bool AddWatcher(DocWatcher *watcher, void *userData);
356
bool RemoveWatcher(DocWatcher *watcher, void *userData);
357
const WatcherWithUserData *GetWatchers() const { return watchers; }
358
int GetLenWatchers() const { return lenWatchers; }
360
CharClassify::cc WordCharClass(unsigned char ch);
361
bool IsWordPartSeparator(char ch);
362
int WordPartLeft(int pos);
363
int WordPartRight(int pos);
364
int ExtendStyleRange(int pos, int delta, bool singleLine = false);
365
bool IsWhiteLine(int line) const;
367
int ParaDown(int pos);
368
int IndentSize() { return actualIndentInChars; }
369
int BraceMatch(int position, int maxReStyle);
372
bool IsWordStartAt(int pos);
373
bool IsWordEndAt(int pos);
374
bool IsWordAt(int start, int end);
376
void NotifyModifyAttempt();
377
void NotifySavePoint(bool atSavePoint);
378
void NotifyModified(DocModification mh);
385
UndoGroup(Document *pdoc_, bool groupNeeded_=true) :
386
pdoc(pdoc_), groupNeeded(groupNeeded_) {
388
pdoc->BeginUndoAction();
393
pdoc->EndUndoAction();
396
bool Needed() const {
403
* To optimise processing of document modifications by DocWatchers, a hint is passed indicating the
404
* scope of the change.
405
* If the DocWatcher is a document view then this can be used to optimise screen updating.
407
class DocModification {
409
int modificationType;
412
int linesAdded; /**< Negative if lines deleted. */
413
const char *text; /**< Only valid for changes to text, not for changes to style. */
417
int annotationLinesAdded;
420
DocModification(int modificationType_, int position_=0, int length_=0,
421
int linesAdded_=0, const char *text_=0, int line_=0) :
422
modificationType(modificationType_),
425
linesAdded(linesAdded_),
430
annotationLinesAdded(0),
433
DocModification(int modificationType_, const Action &act, int linesAdded_=0) :
434
modificationType(modificationType_),
435
position(act.position),
437
linesAdded(linesAdded_),
442
annotationLinesAdded(0),
447
* A class that wants to receive notifications from a Document must be derived from DocWatcher
448
* and implement the notification methods. It can then be added to the watcher list with AddWatcher.
452
virtual ~DocWatcher() {}
454
virtual void NotifyModifyAttempt(Document *doc, void *userData) = 0;
455
virtual void NotifySavePoint(Document *doc, void *userData, bool atSavePoint) = 0;
456
virtual void NotifyModified(Document *doc, DocModification mh, void *userData) = 0;
457
virtual void NotifyDeleted(Document *doc, void *userData) = 0;
458
virtual void NotifyStyleNeeded(Document *doc, void *userData, int endPos) = 0;
459
virtual void NotifyLexerChanged(Document *doc, void *userData) = 0;
460
virtual void NotifyErrorOccurred(Document *doc, void *userData, int status) = 0;