报表开发人员手册:自定义报表组件编写

FastReport vcl是用于 Delphi、C++ Builder、RAD Studio 和 Lazarus 的报告和文档创建 VCL 库。它提供了可视化模板设计器,可以访问 30 多种格式,并可以部署到云、网站、电子邮件和打印中。

FastReport 有大量的组件,可以放在报表设计页面上。它们是:文本、图片、线条、几何图形、OLE、Rich、条码、图表等。您也可以编写自己的自定义组件,然后将其附加到FastReport。

在 FastReport 中定义了几个类,从中继承了组件。有关更多详细信息,请参阅“ FastReport 类层次结构”一章。这个TfrxView类是我们最感兴趣的,因为大多数报表组件都是从它继承而来的。

至少应该有基本类中Draw定义的方法TfrxReportComponent。

procedure Draw(Canvas: TCanvas; ScaleX, ScaleY, OffsetX, OffsetY: Extended); virtual;

在设计器中、预览窗口中和输出打印期间绘制组件时调用此方法。TfrxView覆盖此方法以绘制对象框架和背景。此方法应该在“Canvas”绘图表面上绘制组件内容。对象坐标和大小都存储在AbsLeft,AbsTop,Width和Height相应属性。

ScaleX 和 ScaleY 参数分别定义 X 轴和 Y 轴上的对象缩放。这些参数在 100% 缩放时等于 1,如果用户在设计器或预览窗口中修改缩放,则这些参数可能会有所不同。OffsetX 和 OffsetY 参数通过 X 轴和 Y 轴进行点位移。因此,当考虑这些参数时,左上角坐标如下:

X := Round(AbsLeft * ScaleX + OffsetX);

为了简化坐标操作,在类中定义了BeginDraw方法(其参数类似于Draw方法)TfrxView

procedure BeginDraw(Canvas: TCanvas; ScaleX, ScaleY, OffsetX, OffsetY: Extended); virtual;

它应该在Draw方法的第一行中调用。此方法将坐标转换为FX, , FY, FX1, FY1, FDX, FDY,FFrameWidth整数值,以后可以在 TCanvas 方法中使用。此方法还将 Canvas、ScaleX 和 ScaleY 值复制到FCanvas、FScaleX、FScaleY变量中,可以从任何类方法中引用这些变量。

还有两种为TfrxView类中的对象绘制背景和框架的方法。

    procedure DrawBackground;
    procedure DrawFrame;

BeginDraw 方法应该在调用这些方法之前被调用。

让我们检查创建一个将显示箭头的组件。

type
  TfrxArrowView = class(TfrxView)
  public
    { we should override only two methods }
    procedure Draw(Canvas: TCanvas; ScaleX, ScaleY, OffsetX, OffsetY: Extended); override;
    class function GetDescription: String; override;
  published
    { Place required properties into the published section }
    property BrushStyle;
    property Color;
    property Frame;
  end;

class function TfrxArrowView.GetDescription: String;
begin
  { component description will be displayed next to its icon in toolbar }
  Result := 'Arrow object';
end;

procedure TfrxArrowView.Draw(Canvas: TCanvas; ScaleX, ScaleY, OffsetX, OffsetY: Extended);
begin
  { call this method to perform coordinates transformation }
  BeginDraw(Canvas, ScaleX, ScaleY, OffsetX, OffsetY);
  with Canvas do
  begin
    { set colors }
    Brush.Color := Color;
    Brush.Style := BrushStyle;
    Pen.Width := FFrameWidth;
    Pen.Color := Frame.Color;
    { draw arrow }
    Polygon(
      [Point(FX, FY + FDY div 4),
      Point(FX + FDX * 38 div 60, FY + FDY div 4),
      Point(FX + FDX * 38 div 60, FY),
      Point(FX1, FY + FDY div 2),
      Point(FX + FDX * 38 div 60, FY1),
      Point(FX + FDX * 38 div 60, FY + FDY * 3 div 4),
      Point(FX, FY + FDY * 3 div 4)]);
  end;
end;

{ registration }
var
  Bmp: TBitmap;
initialization
  Bmp := TBitmap.Create;
  Bmp.LoadFromResourceName(hInstance, 'frxArrowView');
  frxObjects.RegisterObject(TfrxArrowView, Bmp);

finalization
  { delete component from list of available ones }
  frxObjects.Unregister(TfrxArrowView);
  Bmp.Free;

end.

创建其显示从一个DB之一的任何数据传输应的组件DataSet,DataField属性分为“发布”部分,然后重写GetData方法。让我们以TfrxCheckBoxView标准组件为例对其进行检查。

该组件可以通过基本类中声明的DataSet和DataField属性连接到 DB 字段TfrxView。此外,该组件具有Expression可放置表达式的属性。一旦计算出来,结果将被放入Checked属性中。如果Checked属性等于“True” ,则此组件显示一个叉号。您可以在下方看到组件的初始声明文本(最重要的部分)。

  TfrxCheckBoxView = class(TfrxView)
  private
    FChecked: Boolean;
    FExpression: String;
    procedure DrawCheck(ARect: TRect);
  public
    procedure Draw(Canvas: TCanvas; ScaleX, ScaleY, OffsetX, OffsetY: Extended); override;
    procedure GetData; override;
  published
    property Checked: Boolean read FChecked write FChecked default True;
    property DataField;
    property DataSet;
    property Expression: String read FExpression write FExpression;
  end;

procedure TfrxCheckBoxView.Draw(Canvas: TCanvas; ScaleX, ScaleY, OffsetX, OffsetY: Extended);
begin
  BeginDraw(Canvas, ScaleX, ScaleY, OffsetX, OffsetY);
  DrawBackground;
  DrawCheck(Rect(FX, FY, FX1, FY1));
  DrawFrame;
end;

procedure TfrxCheckBoxView.GetData;
begin
  inherited;
  if IsDataField then
    FChecked := DataSet.Value[DataField]
  else if FExpression <> '' then
    FChecked := Report.Calc(FExpression);
end;

如果您对 FastReport 感兴趣,欢迎加入 FastReport QQ 交流群:702295239

posted @ 2021-07-01 13:54  roffey  阅读(164)  评论(0编辑  收藏  举报