Click here to Skip to main content
Click here to Skip to main content

Line Numbering of RichTextBox in .NET 2.0

By , 26 Jan 2007
 

Sample Image - sample.jpg

Introduction

This article demonstrates how to add line numbers to a RichTextBox control. This is done by adding a PictureBox control on the left side of the rich text box control. When the appropriate events happen on the Form or on the RichTextBox control, we call a method that paints the line numbers on the PictureBox control. This approach is more accurate and efficient than other approaches I've seen where another textbox control is used for printing the line numbers. The PictureBox will only print the line numbers that are currently visible in the RichTextBox control. During the process of constructing this code sample, I initially based it on the code sample provided in the article: http://www.codeproject.com/cs/miscctrl/numberedtextbox.asp and enhanced it by doing the following:

  1. Replace the Label control for numbering with a PictureBox.
  2. I fixed the scrolling problem with that implementation due to the fact that the RichTextBox uses smooth scrolling (scrolling with the scrollbar scrolls the text in pixels, not in lines). As a result, sometimes the first line may be displayed in half, in which case, the line numbering should address it.

Here is the code for printing the line numbers. Note that we have a RichTextBox control called MyRichTextBox and a PictureBox control called MyPictureBox. The method gets an argument which is the Graphics of the PictureBox control. This is the method DrawRichTextBoxLineNumbers:

Private Sub DrawRichTextBoxLineNumbers(ByRef g As Graphics)
  'Calculate font heigth as the difference in Y coordinate 
  'between line 2 and line 1
  'Note that the RichTextBox text must have at least two lines. 
  '  So the initial Text property of the RichTextBox 
  '  should not be an empty string. It could be something 
  '  like vbcrlf & vbcrlf & vbcrlf 
  With MyRichTextBox
   Dim font_height As Single 
   font_height = .GetPositionFromCharIndex(.GetFirstCharIndexFromLine(2)).Y _
         - .GetPositionFromCharIndex(.GetFirstCharIndexFromLine(1)).Y
   If font_height = 0 Then Exit Sub

   'Get the first line index and location
   Dim first_index As Integer
   Dim first_line As Integer
   Dim first_line_y As Integer
   first_index = .GetCharIndexFromPosition(New _
         Point(0, g.VisibleClipBounds.Y + font_height / 3))
   first_line = .GetLineFromCharIndex(first_index)
   first_line_y = .GetPositionFromCharIndex(first_index).Y

   'Print on the PictureBox the visible line numbers of the RichTextBox
   g.Clear(Control.DefaultBackColor)
   Dim i As Integer = first_line
   Dim y As Single
   Do While y < g.VisibleClipBounds.Y + g.VisibleClipBounds.Height
    y = first_line_y + 2 + font_height * (i - first_line - 1)
    g.DrawString((i).ToString, .Font, Brushes.DarkBlue, MyPictureBox.Width _
          - g.MeasureString((i).ToString, .Font).Width, y)
    i += 1
   Loop
   'Debug.WriteLine("Finished: " & firstLine + 1 & " " & i - 1)
  End With
End Sub

This method should be called from the following locations:

  1. Upon RichTextBox's Resize event: DrawRichTextBoxLineNumbers(MyPictureBox.CreateGraphics).
  2. When the VScroll event happens on the RichTextBox: (MyRichTextBox.VScroll).DrawRichTextBoxLineNumbers(MyPictureBox.CreateGraphics).
  3. When the Paint event happens: (MyPictureBox.Paint).DrawRichTextBoxLineNumbers(e.Graphics). Note that it this method, we send e.Graphics and not MyPictureBox.CreateGraphics, as we would like to repaint only the required region within the PictureBox control.

These are the locations from which we need to invoke the line numbering method:

Private Sub r_Resize(ByVal sender As Object, ByVal e As System.EventArgs) _
     Handles MyRichTextBox.Resize
     MyPictureBox.Invalidate()
End Sub

Private Sub r_VScroll(ByVal sender As Object, ByVal e As System.EventArgs) _
        Handles MyRichTextBox.VScroll
  MyPictureBox.Invalidate()
End Sub

Private Sub p_Paint(ByVal sender As Object, _
        ByVal e As System.Windows.Forms.PaintEventArgs) _
        Handles MyPictureBox.Paint
  DrawRichTextBoxLineNumbers(e.Graphics)
End Sub

Private Sub Form1_Load(ByVal sender As System.Object, ByVal e As System.EventArgs) _
        Handles MyBase.Load
  MyRichTextBox.Text = vbCrLf & vbCrLf & vbCrLf
End Sub

Other general notes

  1. The code above assumes that the font name and the size of the RichTextBox control is the same for the entire text.
  2. The WordWrap property of the RichTextBox control should be set to False.
  3. I've placed this code in the Form_Load method to assure that the textbox has at least two lines always:
  4. MyRichTextBox.Text = vbCrLf & vbCrLf & vbCrLf
  5. I recommend docking the PictureBox to the left and then calculating the width of the PictureBox based on g.MeasureString("000", MyRichTextBox.Font).Width. The width of the PictureBox should be recalculated, and the PictureBox should be redrawn when the font of the RichTextBox changes.
  6. The PictureBox control may be replaced with a Panel or any other control that exposes the CreateGraphics() method.
  7. The control was designed for RichTextBox controls with WordWrap = False.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

About the Author

Michael Elly
United States United States
Member
No Biography provided

Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
Hint: For improved responsiveness ensure Javascript is enabled and choose 'Normal' from the Layout dropdown and hit 'Update'.
You must Sign In to use this message board.
Search this forum  
    Spacing  Noise  Layout  Per page   
BugYou saved me hours!memberJymmy09723 May '12 - 8:05 
QuestionHimemberBotsbushman11 Aug '08 - 13:09 
GeneralThanksmemberMember 44967306 May '08 - 8:51 
GeneralProblem with moving thumbmemberDipendu Paul5 Jul '07 - 1:18 
GeneralSimple code modificationsmemberJason Barrera22 Jun '07 - 11:55 
QuestionText Change in coordinate with font changes...memberbrottmayer22 Feb '07 - 3:10 
What I mean, is how do you compensate for the font size changing so that the number list matches that of the font size in the myrichtextbox? This is the only dilemma that i have, any help would be great. Thanks.
 
brottmayer
GeneralLine numbering in C# - complete [modified]memberbiopsy15 Feb '07 - 5:44 
GeneralRe: Line numbering in C# - complete [modified]memberPanki_17 Oct '11 - 22:32 
GeneralLooks greatmemberAlexandru Lungu30 Jan '07 - 10:08 
GeneralHimemberAhmed El-Badry25 Jan '07 - 1:47 
GeneralResource leak due to CreateGraphicsmembermav.northwind25 Dec '06 - 22:01 
GeneralRe: Resource leak due to CreateGraphicsmemberMichael Elly26 Jan '07 - 0:27 
QuestionAny way to make the code show line numbers only upto contained lines in the RTBox?memberkil0byte^bandit5 Dec '06 - 4:37 
AnswerRe: Any way to make the code show line numbers only upto contained lines in the RTBox?memberMichael Elly6 Dec '06 - 20:19 
GeneralRe: Any way to make the code show line numbers only upto contained lines in the RTBox?membergabegabe4 Jan '07 - 4:20 
GeneralRe: Any way to make the code show line numbers only upto contained lines in the RTBox?memberMichael Elly26 Jan '07 - 0:43 
Generalwordwrap linecount fixmembergabegabe17 Jan '07 - 21:29 
General[Message Deleted]memberArkady Lesniara26 Jun '06 - 4:24 
GeneralRe: Compile ProblemmemberGuardianStorm26 Jun '06 - 6:27 
GeneralRe: Compile Problemmembermicmelly27 Jun '06 - 6:43 
GeneralRe: Compile ProblemmemberArkady Lesniara27 Jun '06 - 6:54 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Permalink | Advertise | Privacy | Mobile
Web02 | 2.6.130516.1 | Last Updated 26 Jan 2007
Article Copyright 2006 by Michael Elly
Everything else Copyright © CodeProject, 1999-2013
Terms of Use
Layout: fixed | fluid