views:

62

answers:

2

A bit of background. I would like to be able to process text for the line that the caret is on in a WPF RichTextBox. Please see my earlier question about the TextPointer class: http://stackoverflow.com/questions/3046162/how-to-keep-track-of-textpointer-in-wpf-richtextbox.

I know how to get the TextPointer at the start of the current line by using GetLineStartPosition with 0 as the argument but would now like to get the TextPointer at the end of the line. It was suggested in my previous question that this is possible using the GetLineStartPosition method.

I would appreciate it if someone can explain a bit about how the GetLineStartPosition works with regard to end of line pointers.

Thanks in advance for any help.

A: 

The end of the current line is the same as the start of the next line, so you can use GetLineStartPosition(1). Note that this will return null when you're on the last line, so you may want to use DocumentEnd in that case.

var currentLine = new TextRange(rtb.CaretPosition.GetLineStartPosition(0), rtb.CaretPosition.GetLineStartPosition(1) ?? rtb.CaretPosition.DocumentEnd).Text;
Quartermeister
Actually the end of the current line is not the same as the start of the next line. They may be in different paragraphs, table cells, etc. Even in a plain old text file this is true: There is a CRLF between the end of each line and the start of the next. This is even more true in a FlowDocument with so much complexity like tables, sections, spans, figures, floaters, InlineUIContainers, etc.
Ray Burns
+1  A: 

GetLineStartPosition is able to return you the start of a line but not the end of a line. For that you will have to combine it with GetInsertionPosition.

Here's how GetLineStartPosition works:

  • GetLineStartPosition(-1) gets the start of the previous line
  • GetLineStartPosition(0) gets the start of the current line
  • GetLineStartPosition(1) gets the start of the next line

You can also call it with larger integers to get lines further away.

To get the end of a line just get the start of the next line, then get the prior insertion position. Basically it is this:

pointer.GetLineStartPosition(1).GetInsertionPosition(LogicalDirection.Backward);

However this does not work when you are on the last line of a document: GetLineStartPosition returns null.

The easy way to fix it is to do this:

var nextStart = pointer.GetLineStartPosition(1)
var lineEnd = (nextStart !=null ? nextStart : pointer.DocumentEnd).GetInsertionPosition(LogicalDirection.Backward);

The reason GetInsertionPosition must be used rather than just moving one symbol over using GetNextContextPosition or GetPointerAtOffset is that every element in the FlowDocument element tree is a symbol. So for example if your current line is the last line in a table, GetLineStartPosition(1) will return a pointer inside the first Run in the first Paragraph following the table, whereas the end of the current line is a the end of the last Run in the last Paragraph in the last TableCell, ... you get the idea.

It is best to let WPF handle all the complexity of moving TextPointers around the FlowDocument, which means using GetInsertionPosition to find the end of the same line the original TextPointer points to.

Ray Burns
Thanks a lot Ray. Again, a really helpful response to my question.
Alan Spark