Delphi: Center Specific Line in TRichEdit by Scrolling

匿名 (未验证) 提交于 2019-12-03 10:24:21

问题:

I have a Delphi 2007 TRichEdit with several lines in it. I want to scroll the richedit vertically such that a specific line number if approximately centered in the visible/display area of the richedit. For example, I want to write the code for CenterLineInRichEdit in this example:

procedure CenterLineInRichEdit(Edit: TRichEdit; LineNum: Integer); begin   ...   Edit.ScrollTo(...); end;  procedure TForm1.FormCreate(Sender: TObject); var   REdit: TRichEdit;   i: Integer; begin   REdit := TRichEdit.Create(Self);   REdit.Parent := Self;   Redit.ScrollBars := ssVertical;   REdit.SetBounds(10, 10, 200, 150);   for i := 1 to 25 do     REdit.Lines.Add('This is line number ' + IntToStr(i));   CenterLineInRichEdit(REdit, 13); end; 

I looked into using the WM_VSCROLL message, and it allows scrolling up/down one line, etc. but not scrolling to center a specific line.

回答1:

Give this a try;

procedure VertCenterLine(RichEdit: TRichEdit; LineNum: Integer); // I don't know the reason but the RichEdit 2 control in VCL does not // respond to the EM_SCROLLCARET in Richedit.h but it does so to the // constant in WinUser.h const   EM_SCROLLCARET  = $00B7; var   TextPos: lResult;   Pos: TSmallPoint; begin   TextPos := SendMessage(RichEdit.Handle, EM_LINEINDEX, LineNum, 0);    if TextPos <> -1 then begin     // Go to top     SendMessage(RichEdit.Handle, EM_SETSEL, 0, 0);     SendMessage(RichEdit.Handle, EM_SCROLLCARET, 0, 0);      // Get the coordinates for the beginning of the line     Longint(Pos) := SendMessage(RichEdit.Handle, EM_POSFROMCHAR, TextPos, 0);      // Scroll from the top     SendMessage(RichEdit.Handle, WM_VSCROLL,         MakeWParam(SB_THUMBPOSITION, Pos.y - RichEdit.ClientHeight div 2), 0);      // Optionally set the caret to the beginning of the line     SendMessage(RichEdit.Handle, EM_SETSEL, TextPos, TextPos);   end; end; 

The below is an alternative in that it centers the first occurance of a string instead of a line number;

procedure VertCenterText(RichEdit: TRichEdit; Text: string); const   EM_SCROLLCARET  = $00B7; var   FindText: TFindText;   TextPos: lResult;   Pos: TSmallPoint; begin   FindText.chrg.cpMin := 0;   FindText.chrg.cpMax := -1;   FindText.lpstrText := PChar(Text);   TextPos := SendMessage(RichEdit.Handle, EM_FINDTEXT,       FR_DOWN or FR_WHOLEWORD, Longint(@FindText));    if TextPos <> -1 then begin     SendMessage(RichEdit.Handle, EM_SETSEL, 0, 0);     SendMessage(RichEdit.Handle, EM_SCROLLCARET, 0, 0);      Longint(Pos) := SendMessage(RichEdit.Handle, EM_POSFROMCHAR, TextPos, 0);     SendMessage(RichEdit.Handle, WM_VSCROLL,         MakeWParam(SB_THUMBPOSITION, Pos.y - RichEdit.ClientHeight div 2), 0);      SendMessage(RichEdit.Handle, EM_SETSEL, TextPos, TextPos);   end; end; 


回答2:

Based on the ideas here, I came up with one solution. It assumes that all the lines in the richedit are the same height and that the richedit's default font correctly reports its height, but it might be useful to some people:

type   TCustomEditHack = class(TCustomEdit);  procedure CenterLineInEdit(Edit: TCustomEdit; LineNum: Integer); var   VisibleLines: Integer;   TopLine: Integer;   FirstLine: Integer; begin   FirstLine := Edit.Perform(EM_GETFIRSTVISIBLELINE, 0, 0);   VisibleLines := Round(Edit.ClientHeight / Abs(TCustomEditHack(Edit).Font.Height));    if VisibleLines <= 1 then     TopLine := LineNum   else     TopLine := Max(LineNum - Round((VisibleLines/2)) + 1, 0);    if FirstLine <> TopLine then     Edit.Perform(EM_LINESCROLL, 0, TopLine - FirstLine); end; 

I tested this with TRichEdit, but it might work for TMemo as well.



回答3:

Send an EM_LINESCROLL message to the RichEdit:

SendMessage(REdit.Handle, EM_LINESCROLL, 0, NumberOfVerticalLinesToScroll); 

See the EM_LINESCROLL MSDN topic.



回答4:

You will need to use a couple of Windows messages to manipulate this aspect of your control in a generic fashion:

You will need to calculate how many lines to scroll up/down from the current top-line to bring a desired absolute line number into view, but you will have to calculate the number of lines visible in the control yourself (using font metrics and control height).

Note that with a RichEdit control the height of each line may vary according to fonts applied to the text in the control so any approach based on line numbers alone is likely to be only approximately accurate. Also I'm not sure that it's possible to determine the current visible range of the control (i.e. the number of lines currently visible) directly, so calculating it yourself is necessary.

From memory, the SynEdit control offers some additional control over such things, providing both a read/write TopLine property as well as a LinesInWindow property. However, I think SynEdit is not rich text capable, but if this is not actually a concern in your application (i.e. you can use a consistent font for all lines in the content) then it may be an attractive or suitable alternative.



易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!