Offsets, Ranges, and Positions
There are several ways to reference the location of characters within a string or snapshot, each with their own benefits.
Offsets
Offsets are zero-based integer numbers indicating a character index from the beginning of a string or snapshot.
The first character in a document is at index 0
. The offset past the last character in a snapshot can be retrieved via the ITextSnapshot.Length property.
Text Ranges
It is often necessary to work with an ordered pair of offsets. For instance when getting a substring of text in a snapshot, you need to pass the offset range to obtain.
The TextRange structure represents an ordered pair of two offsets. It has many methods for comparing the range with other offsets and ranges.
To create a text range that spans 2
characters starting at offset 3
, you'd create a text range with start offset 3
and end offset 5
. Note that the end offset is the offset after the last character encapsulated by the range.
TextRange range = new TextRange(3, 5);
This code creates a TextRange with the same offset values but creates it using start offset and length combinations instead.
TextRange range = TextRange.FromSpan(3, 2);
Text Positions
TextPosition structures are similar to offsets in that they indicate a certain character within a string or snapshot. However they work in terms of a line index and a character index within that line.
The Line and Character values of text positions are zero-based so a text position representing line index 1
and character 7
references the eighth character on the second line in the text.
This code creates a TextPosition with line index 1
and character 7
:
TextPosition position = new TextPosition(1, 7);
Text Position Ranges
Similar to how TextRange provides a range of offsets, the TextPositionRange structure provides a range of TextPosition objects.
This code creates a TextPositionRange from position (1,7)
to (2,3)
:
TextPositionRange positionRange = new TextPositionRange(1, 7, 2, 3);
Converting Between Offsets and Text Positions
Sometimes it's useful to convert between offsets and text positions. Likewise, conversion between text ranges and text positions is possible.
These members in the ITextSnapshot inteface allow for conversion:
Member | Description |
---|---|
OffsetToPosition Method | Returns the TextPosition that represents the specified offset within the text lines. |
PositionRangeToTextRange Method | Returns the TextRange that represents the specified TextPositionRange within the snapshot. |
PositionToOffset Method | Returns the offset for the specified TextPosition within the text lines. |
TextRangeToPositionRange Method | Returns the TextPositionRange that represents the specified TextRange within the snapshot. |
Snapshot Offsets
The TextSnapshotOffset structure is a bit more advanced than a simple offset or TextPosition. It stores a reference to a particular location within an ITextSnapshot and provides access to the offset or TextPosition of that location.
The root ITextSnapshot and the ITextSnapshotLine that contains the snapshot offset are accessible from the snapshot offset.
This code creates a TextSnapshotOffset at offset 5
within the document's current ITextSnapshot:
TextSnapshotOffset snapshotOffset = new TextSnapshotOffset(document.CurrentSnapshot, 5);
Snapshot offsets can be translated to another snaphot. Details on this are available in the Snapshot Translation topic.
Snapshot Ranges
The TextSnapshotRange structure is similar to a TextSnapshotOffset in that it references a specific ITextSnapshot. The different is that it tracks two normalized locations within the snapshot.
Accordingly, it can do additional things like return the text contained within the range, give the offset/text positions of the start/end, etc.
To create a snapshot range that spans 2
characters starting at offset 3
, you'd create a snapshot range with start offset 3
and end offset 5
. Note that the end offset is the offset after the last character encapsulated by the range.
TextSnapshotRange snapshotRange = new TextSnapshotRange(document.CurrentSnapshot, 3, 5);
This code creates a TextSnapshotRange with the same offset values but creates it using start offset and length combinations instead.
TextSnapshotRange snapshotRange = TextSnapshotRange.FromSpan(document.CurrentSnapshot, 3, 2);
Snapshot ranges can be translated to another snaphot. Details on this are available in the Snapshot Translation topic.
Normalized Snapshot Range Collection
The NormalizedTextSnapshotRangeCollection class provides a collection implementation that normalizes all of its contained TextSnapshotRange objects.
This means that snapshot ranges in the collection are guaranteed to:
- All be based on the same ITextSnapshot.
- Be in sequential order by start offset.
- Not overlap (overlapping snapshot ranges are merged when the collection is built).
Version Ranges
As described in the Documents, Snapshots, and Versions topic, text versions are placeholders that provide the core functionality needed to link snapshots together and support features like Snapshot Translation.
The ITextVersionRange interface designates an object that can refer to a text range within an ITextVersion. By itself, the ITextVersionRange object is not very useful since it doesn't provide direct access to the text range data. It also holds no references to any text snapshots, which makes it very lightweight.
The nice thing about version ranges though is that they are a lightweight reference to a text range that can be easily translated to any snapshot range or other text range in a specific text version. This makes them ideal for use in scenarios where you wish to store a version-oriented range for a long period of time, without needing to retain the related snapshot in memory.
Version ranges can be created with the various ITextVersion.CreateRange methods.
ITextVersionRange versionRange = document.CurrentSnapshot.Version.CreateRange(1, 3);
Version ranges can be translated to any snapshot for the same document:
TextSnapshotRange snapshotRange = versionRange.Translate(document.CurrentSnapshot);
Version ranges also can be translated to a text range for particular lightweight ITextVersion that is also from the same document.
TextRange range = versionRange.Translate(document.CurrentSnapshot.Version);
Another way a version range can be created is from a TextSnapshotRange:
ITextVersionRange versionRange = snapshotRange.ToVersionRange(TextRangeTrackingModes.Default);