delphi - 修改系列点时是否可以仅重新绘制 TeeChart 的一个区域?

标签 delphi delphi-7 vcl teechart

我在刷新包含很多点的整个图表时遇到了性能问题,想询问是否有办法减少重新绘制图表所需的时间。

例如,我的图表包含一百个 TAreaSeries,每个值 1000 点。这些值被构建一次并且之后不会改变。但有一个简短的 TLineSeries 有 2 个点,可以在图形上“跳跃”,每 50 毫秒更改其坐标。尽管只修改了绘图的一小部分(TLineSeries 区域中的像素),但整个图形(具有数千个点)都被更新和重新绘制,这徒劳地花费了大量资源并导致巨大的延迟。在我的例子中,具有 2 个点的 TLineSeries 的下一个和上一个位置是已知的,所以我想知道是否有办法强制 TChart 仅重新绘制某些预定义的绘图区域,而不是重新绘制整个图形?

最佳答案

在实时应用程序中,我们通常建议遵循实时图表文章 here 中的说明。 .

就仅重新绘制图表的一个区域而言,您不能调用InvalidateRacte而只能重新创建矩形中的图表部分。 InvalidateRacte 只能用于重绘缓冲区位图的一部分(Chart1.Canvas.Bitmap,当使用 Chart1.BufferedDisplay 时),VCL框架不支持这一点,我们也不会从中获得任何好处。

Here您可以下载一个项目,该项目在一个图表中在另一个图表上绘制线系列,而无需重新创建它,这是最耗时的任务。有一个按钮可以计算缓冲区位图可以重新绘制的次数。我们每秒大约得到 5000 个。如果每次都需要重新创建图表(例如:绘制轴、系列、图例等),那么每秒最多 100 次。项目中显示的功能只能使用 new TeeChart Pro beta 来使用。 。项目代码是这样的:

type
  TSeriesAccess=class(TChartSeries);
  TChartAccess=class(TCustomTeePanel);

procedure TForm1.ScrollBar1Change(Sender: TObject);
begin
  // Example: Modify a Line value
  Series1.YValues[10]:=350+ScrollBar1.Position;

  // Paint Line to Chart2, just to check the change (this is optional)
  Series1.Repaint;

  // Redraw Chart1 fast, just the buffer bitmap, without repainting everything:
  TChartAccess(Chart1).Paint;

  // Draw the Line over Chart1, on top of it, instead of "inside" it
  TSeriesAccess(Series1).FParent:=Chart1;
  TSeriesAccess(Series1).DrawAllValues;
  TSeriesAccess(Series1).FParent:=Chart2;
end;

// Count the number of times a Chart buffer can be redisplayed
// (without redrawing everything)
procedure TForm1.Button1Click(Sender: TObject);
var t1 : Cardinal;
    tmpCount : Integer;
begin
  tmpCount:=0;

  t1:=GetTickCount;

  // Loop for one second
  while GetTickCount-t1 < 1000 do
  begin
    TChartAccess(Chart1).Paint;
    Inc(tmpCount);
  end;

  Caption:='Redraws per second: '+IntToStr(tmpCount);
end;

另一个选项是this ,使用与我们的项目大致相同的 ColorLine,使用 Pen.Mode = pmXor 进行绘画。

关于您的项目,您在刷新系列之前错过了将AutoRepaint设置为true。此代码有效:

procedure TForm1.Timer1Timer(Sender: TObject);
begin
    // The 'jumping' TLineSeries will move to the right 1 point every 1 second.
    Form1.Chart1.Series[1].XValue[0]:= Form1.Chart1.Series[1].XValue[0] + 1;
    Form1.Chart1.Series[1].XValue[1]:= Form1.Chart1.Series[1].XValue[1] + 1;

    Form1.Chart1.AutoRepaint:=True;
    Form1.Chart1.Series[1].Repaint;
    Form1.Chart1.AutoRepaint:=False;
end;

此外,您还可以使用TColorLineTool来代替垂直线系列。您应该删除第二行系列,在设计时添加 ColorLine 工具并实现如下代码:

unit UnitTeeChartRefresh;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, ExtCtrls, TeeProcs, TeEngine, Chart, Series, StdCtrls,
  TeeGDIPlus, TeeTools;

type
  TForm1 = class(TForm)
    Chart1: TChart;
    Series1: TAreaSeries;
    Button1: TButton;
    Timer1: TTimer;
    Label1: TLabel;
    ChartTool1: TColorLineTool;
    procedure FormCreate(Sender: TObject);
    procedure Button1Click(Sender: TObject);
    procedure Timer1Timer(Sender: TObject);
    procedure Chart1AfterDraw(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }
  end;

var
  Form1: TForm1;
  times_redrawn: cardinal = 0;   // A counter indicating how many times our chart was redrawn.

implementation

{$R *.dfm}

procedure TForm1.FormCreate(Sender: TObject);
var
    i: integer;
begin
  //Let's add one constant TAreaSeries and one 'jumping' across the graph TLineSeries
  Randomize;
  Form1.Chart1.Axes.Left.SetMinMax(0, 10);
  Form1.Chart1.Axes.Bottom.SetMinMax(0, 20);

  for i:= 1 to 20 do
  begin
    Form1.Chart1.Series[0].AddXY(i, Random(10))
  end;

  ChartTool1.Axis:=Chart1.Axes.Bottom;
  ChartTool1.Value:=0;

  // If I enable AutoRepaint I will be able to see all changes on the whole plot in real-time
  // and have no problem.
  // But I want to manually control the refreshing process and even refresh some regions of the
  // graph separately.
  Form1.Chart1.AutoRepaint:=False;
end;


procedure TForm1.Button1Click(Sender: TObject);
begin
  // This button launches the animation of the 'jumping' TLineSeries.
  Form1.Timer1.Enabled:= not Form1.Timer1.Enabled;
end;

procedure TForm1.Timer1Timer(Sender: TObject);
begin
  // The 'jumping' TLineSeries will move to the right 1 point every 1 second.
  Form1.Chart1.AutoRepaint:=True;
  ChartTool1.Value:=ChartTool1.Value + 1;
  Form1.Chart1.AutoRepaint:=False;
end;

procedure TForm1.Chart1AfterDraw(Sender: TObject);
begin
  times_redrawn:= times_redrawn + 1;
  Form1.Label1.Caption:= intToStr(times_redrawn);
end;

end.

关于delphi - 修改系列点时是否可以仅重新绘制 TeeChart 的一个区域?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17924435/

相关文章:

delphi - 在 Delphi 中存储一组值

c++ - 在 vector 数组中查找元素

delphi - 如何重新设计 Delphi 6 应用程序的外观?

delphi - Firemonkey 中的 ColorToRGB 函数等效

delphi - 将 AlphaBlend 属性添加到面板

function - 如何将另一个函数返回的函数分配给函数变量?结果而不是生成函数本身

delphi - 无法使用 SendInput(Edit1.Text) 复制俄语(西里尔字母或 Unicode)符号

delphi - 如何将滚动条添加到TComponent?

delphi - 具有 TabOrder 属性的 TSpeedButton

delphi - 我需要知道一种方法来确定文件是以独占模式还是以其他方式(读写等)打开?