FastReport集粹(一)

时间:2025-04-01 07:31:02
unit ReportClass;

interface

uses CommonForm, SysUtils, Classes, Forms, Dialogs, DB, Controls, Grids, ComCtrls,
     FR_DBSet, FR_Class, Printers, FR_Utils, FR_Pars, DBGrids, IniFiles, Variants,
     TypInfo;

type
  TReport = class
  private
    FDatasetArray: array of TDataset;
    FRDataset: array of TfrDBDataset;

    FrmCommon: TfrmCommon; //设置窗体
    FReport: TfrReport; //报表对象
    FPage: TfrPage; //页面对象
    FVariants: TfrVariables; //报表的数据字典变量
    FVariantList: TStrings; //变量列表

    FGridFields: TStrings; //表格的字段=标签
    FGridWidths: TStrings; //表格的字段=宽度
    FGridRowHeight: integer; //表格行高
    FGrid: TCustomGrid; //表格名称

    FTitleMemo: TfrMemoView; //标题头文本框
    FTitleDate: TfrmemoView; //标题日期文本框
    FPrintReportDate: Boolean; //是否打印页头日期
    FOrientation: Boolean; //报表打印方向
    FPrintPageNumber: Boolean; //是否打印页码

    FHeaderFields: TStrings; //页头字段列表
    FDetailFields: TStrings; //明细字段列表
    FFooterFields: Tstrings; //页脚字段列表

    INIFile: TINIFile;
    FFileName: string; //报表的文件名
    FDetailSourceType: string; //报表数据源类型('TDBGrid'或'TDataset')
    FReportName: string; //在INI文件中小节的报表的名称
    procedure ShowData;
    function GetVariant: TStrings;
    procedure SetVariants(const Value: TStrings);
  protected
    function GetReportTitle: string;
    procedure SetReportTitle(value: string);

    function GetOrientation: Boolean;
    procedure SetOrientation(value: Boolean);

    function GetPrintReportDate: Boolean;
    procedure SetPrintReportDate(value: Boolean);

    function GetPrintPageNumber: Boolean;
    procedure SetPrintPageNumber(value: Boolean);

    function GetHeaderFields: TStrings;
    procedure SetHeaderFields(const Value: TStrings);

    function GetFooterFields: TStrings;
    procedure SetFooterFields(const Value: TStrings);
    function GetDetailFields: TStrings;
    procedure SetDetailFields(const Value: TStrings);

    function GetFileName: string;
    procedure SetFileName(const Value: string);
   
    function GetGrid: TCustomGrid;
    procedure SetGrid(const Grid: TCustomGrid);
  public
    constructor Create;
    destructor Destroy; override;

    procedure Preview; overload;
    procedure Preview(FileName: string); overload;
    procedure Design; overload;
    procedure Design(FileName: string); overload;

    procedure SetMemoView(MemoView: TfrMemoView; //设置文本框属性
                          Alignment: integer;    //Alignment:文本对齐方式,frtaCenter表示居中对齐
                          FrameTyp: word);       //FrameTyp:边框样式,0:无边框;15:有边框
    function FindBandView(BandType: TfrBandType): TfrBandView; //根据BandView类型返回TfrBandview
    function FindMemoView(Text: string; var CompNo: integer): TfrMemoView; //根据MemoView的文本返回TfrMemoView和序号
    function ShowFieldListForm: Boolean;

    procedure SaveFieldList(ReportName: string); //存储字段列表到报表配置文件中
    procedure LoadFieldList(ReportName: string); //从配置文件中调用字段列表

    procedure SetDataset(Datasets: array of TDataset);
  published
    property Title: string read GetReportTitle write SetReportTitle; //报表标题
    property PrintPageNumber: Boolean read GetPrintPageNumber write SetPrintpageNumber default false; //是否打印报表页码
    property PrintReportDate: Boolean read GetPrintReportDate write SetPrintReportDate default true; //是否打印报表日期
    property Orientation: Boolean read GetOrientation write SetOrientation default true; //页面默认为纵向
    property HeaderFields: TStrings read GetHeaderFields write SetHeaderFields; //页头上要打印的字段列表
    property DetailFields: TStrings read GetDetailFields write SetDetailFields; //明细字体列表
    property FooterFields: TStrings read GetFooterFields write SetFooterFields; //页脚上要打印的字段列表
    property FileName: string read GetFileName write SetFileName;
    property DetailSourceType: string read FDetailSourceType write FDetailSourceType; //明细数据源是'TDBGrid'还是'TDataset'
    property Grid: TCustomGrid read GetGrid write SetGrid;
    property Variants: TStrings read GetVariant write SetVariants; //报表数据字典中的变量
  end;

implementation

{ TReport }

constructor ;
var
  band: TfrBandView;
  AppPath: string;
begin
  inherited;
  FHeaderFields := ; //标题字段列表
  FDetailFields := ; //明细字段列表
  FFooterFields := ; //页脚字段列表
  FGridFields := ; //表格字段列表
  FGridWidths := ; //表格字段宽度列表
  FPrintReportDate := true; //默认打印报表日期

  FOrientation := true;
  FReport := (nil); //建立报表
  //建立页面
  ;
  ;
  FPage := [0];
  FVariants := ;
  FVariantList := ;

  Band := ; //建立标题条
  (tbbLeft, tbbTop, tbbWidth, tbbHeight); //条位置
  := btReportTitle; //条类型
  (Band);

  FTitleMemo := ; //建立标题文本
  (tmbLeft, tmbTop, tmbWidth, tmbHeight);
  := baCenter; //文本框居中
  ['Alignment'] := frtaCenter or frtaMiddle; //文本的对齐方式(垂直居中)
  [''] := 2;
  [''] := mvFontName;
  [''] := 14;
  (FTitleMemo);

  FPrintReportDate := true;
  SetPrintReportDate(FPrintReportDate);

  frmCommon := (nil);
  AppPath := ExtractFilePath() + 'Reports/';
  if not DirectoryExists(AppPath) then CreateDir(AppPath);
  INIFile := (AppPath + '');
end;

destructor ;
begin
  FreeAndNil(INIFile);
  FreeAndNil(FHeaderFields);
  FreeAndNil(FFooterFields);
  FreeAndNil(FDetailFields);
  FreeAndNil(FGridWidths);
  FreeAndNil(FGridFields);
  FreeAndNil(frmCommon);
  FreeAndNil(FVariantList);
  FreeAndNil(FVariants);
  FreeAndNil(FReport);
  FDatasetArray := nil;
  inherited;
end;

function : string;
begin
  result :=
end;

procedure (value: string);
begin
  := value;
  := value;
end;

function : Boolean;
begin
  result := FOrientation;
end;

procedure (value: Boolean);
begin
  FOrientation := value;
  if FOrientation then
    (, , , , poPortrait) //页面横向
  else
    (, , , , poLandscape); //页面纵向
  (['Width'] - 70, tdbTop, tdbWidth, tdbHeight);
end;

procedure ;
begin
  ShowData;
  if then
    ;
end;

procedure (FileName: string);
var
  s: string;
begin
  s := FileName;
  if s = '' then s := FFileName;
  if s = '' then exit;
  (s);
  if then
    ;
end;

procedure (FileName: string);
var
  s: string;
begin
  s := FileName;
  if s = '' then s := FFileName;
  if s = '' then exit;
  (s);
  if then
    ;
end;

procedure ;
begin
  ShowData;
  if then
    ;
end;

function : Boolean;
begin
  result := FPrintReportDate;
end;

procedure (value: Boolean);
var
  Blank: integer;
begin
  FPrintReportDate := value;
  if FPrintReportDate then
  begin
    FTitleDate := ;
    if FOrientation then //如果是纵向打印
      Blank := 1480
    else
      Blank := 1200;
    ( - Blank, tdbTop, tdbWidth, tdbHeight);
    ('[Date]');
    (FTitleDate);
  end;
end;

function : Boolean;
begin
  result := FPrintPageNumber;
end;

procedure (value: Boolean);
const
  PageNumber = '[PAGE#]/[TOTALPAGES]';
var
  mv: TfrMemoView;
  Band: TfrBandView;
  CompNo: integer;
begin
  FPrintPageNumber := value;
  Band := FindBandView(btPageFooter);
  if band = nil then //如果没有FooterBand,那么新建一个
  begin
    Band := ;
    := btPageFooter;
    (fbbLeft, fbbTop, fbbWidth, fbbHeight);
    (Band);
  end;
  if FPrintPageNumber then //如果打印页码
  begin
    := true; //报表只有显示两次才能显示页码
   
    mv := ;
    := PageNumber;
    (0, fbbTop, 100, mvHeight);
    := baCenter;
    ['Alignment'] := frtaMiddle or frtaCenter;
    [''] := mvFontName;
    [''] := mvFontSize;
    (mv);
  end
  else //如果不打印页码,那么删除已存在的页码文本框
  begin
    mv := FindMemoView(PageNumber, CompNo);
    if mv <> nil then
    begin
      (CompNo);
      ;
    end;
  end;
end;

function (BandType: TfrBandType): TfrBandView;
var
  v: TfrBandView;
  i: integer;
begin
  result := nil;
  for i:=0 to -1 do //查找是否有FooterBand
  begin
    v := [i];
    if (('TfrBandView') and (TfrBandView(v).BandType = BandType)) then
    begin
      result := v;
      exit;
    end;
  end;
end;

function (Text: string; var CompNo: integer): TfrMemoView;
var
  i: integer;
  v: TfrMemoView;
begin
  result := nil;
  for i:=0 to -1 do
  begin
    v := [i];
    if (('TfrMemoView') and (TfrMemoView(v). = Text)) then
    begin
      CompNo := i;
      result := v;
      exit;
    end;
  end;
end;

procedure (MemoView: TfrMemoView; Alignment: integer; FrameTyp: word);
begin
  if MemoView = nil then exit;
  ['Alignment'] := frtaMiddle or Alignment;  //文本对齐方式(默认为垂直居中)
  [''] := mvFontName; //字体名称
  [''] := mvFontSize; //字体尺寸
  if then
    := FrameTyp; //边框
end;

给打印字段列表赋值/

function : TStrings;
begin
  result := FHeaderFields;
end;

procedure (const Value: TStrings);
begin
  (value);
end;

function : TStrings;
begin
  result := FFooterFields;
end;

procedure (const Value: TStrings);
begin
  (value);
end;

/

function : Boolean;
begin
  := HeaderFields;
  := DetailFields;
  := FooterFields;

  ; //显示报表设置窗体
  result := ;
  if not then exit; //如果窗体是Cancel方式关闭的

  := ; //返回报表标题

  //获得打印字段列表
  HeaderFields := ;
  DetailFields := ;
  FooterFields := ;

  if and (FReportName <> '') then //如果保存打印字段列表
    SaveFieldList(FReportName);
end;

procedure ;

  function AnalyseText(str: string): string;
  var
    n: integer;
  begin
    result := '';
    n := AnsiPos('.', str);
    if n = 0 then exit;
    insert('"', str, n+1);
    result := str + '"';
  end;

  function GetDisLabelWidth(FieldName: string): integer; //根据字段名得到字段标签的长度
  var
    n, i, j: integer;
    s: string;
  begin
    result := 0;
    s := Fieldname;
    n := AnsiPos('.', s);
    if n > 0 then //如果字段名带有表名前缀
    begin
      Delete(s, 1, n);
    end;
    if s = '' then exit;
    for i:=0 to High(FDatasetArray) do //这个函数没考虑到前缀对应的问题,以后有空改吧
    begin
      for j:=0 to FDatasetArray[i].FieldCount-1 do
      begin
        if FDatasetArray[i].Fields[j].FieldName = s then
        begin
          result := FDatasetArray[i].Fields[j].DisplayWidth * 5; //宽度增加5倍
          exit;
        end;
      end;
    end;
  end;

var
  iCol, iLeft, i, w, h: integer;
  mv: TfrMemoView;
  Band: TfrBandView;
begin
  if > 0 then
  begin
    iLeft := tmbLeft; //其它字段的位置在打印日期之下
    iCol := 1; //第一行
    for i:=0 to -1 do
    begin
      if (i mod 3 = 0) and (i > 0) then //一行只显示三列字段
      begin
        inc(iCol, 1);
        iLeft := tmbLeft;
      end;
      mv := ;
      SetMemoView(mv, frtaLeft, 0);
      (iLeft, tdbTop + (tdbHeight + 4) * iCol, 200, mvHeight); //4是上下Memo行之间的间隔
      := [i] + ':['
        + FDatasetArray[0]. + '.' + AnalyseText([i]) + ']';
      (mv);
      inc(iLeft, 200);
    end;
  end
  else //否则不打印页头上的字段列表
  begin
    band := FindBandView(btReportTitle);
    (tbbLeft, tbbTop, tbbWidth, 60); //缩小条高度
  end;

  / 明细 ///

  if then
    h :=
  else
    h := ; //得到文本框高度
  //页头
  Band := ;
  (0, PageBandTop, 0, h);
  := btPageHeader;
  ['OnFirstPage'] := true;
  (Band);

  Band := ; //主数据条
  (0, PageBandTop + DataBandHeight * 2, 0, h);
  := btMasterData;
  := 'frmCommon.' + ; //明细数据条需要一个数据源
  (Band);

  w := 0;
  for i:=0 to -1 do //求出明细文本框居中时的左起点
  begin
    if then //如果使用表格的字段列表的宽度
      w := w + StrToInt([i])
    else
      w := w + GetDisLabelWidth([i]); //得到数据字段的显示长度
  end;
  iLeft := (['Width'] - w) div 2;
  if iLeft < 32 then iLeft := 32;

  for i:=0 to -1 do
  begin
    mv := ; //标签
    SetMemoView(mv, frtaCenter, 15);
    if then //如果使用表格的字段列表的宽度
      w := StrToInt([i])
    else
      w := GetDisLabelWidth([i]); //得到数据字段的显示长度
    (iLeft, PageBandTop, w, h);
    ([i]);
    (mv);

    mv := ; //记录
    (iLeft, PageBandTop + DataBandHeight * 2, w, h);
    if then //文本框是否有边框
      SetMemoView(mv, frtaLeft, 15) //设置文本框属性
    else
      SetMemoView(mv, frtaLeft, 0);
    := '[' + FDatasetArray[0]. + '.'
      + AnalyseText([i]) + ']';
    (mv);
   
    inc(iLeft, w);
  end;

  if iLeft > ['width'] then //如果文本框超过了页面的长度,
  begin
    SetOrientation(false); //那么页面自动转为横向打印
    (['width'] - tdbWidth - 70, tdbTop, tdbWidth, tdbHeight);
  end;

  // 页脚

  if > 0 then
  begin
    Band := FindBandView(btReportSummary);
    if band = nil then //如果没有FooterBand,那么新建一个
    begin
      Band := ;
      := btReportSummary;
      (rfbLeft, rfbTop, rfbWidth, rfbHeight);
      (Band);
    end;

    iLeft := tmbLeft; //其它字段的位置在打印日期之下
    iCol := 0;
    for i:=0 to -1 do
    begin
      if (i mod 3 = 0) and (i > 0) then //一行只显示三列字段
      begin
        inc(iCol, 1);
        iLeft := tmbLeft;
      end;
      mv := ;
      SetMemoView(mv, frtaLeft, 0);
      (iLeft, rfbTop + (tdbHeight + 4) * iCol, 200, mvHeight);
      := [i] + ':['
        + FDatasetArray[0]. + '.' + AnalyseText([i]) + ']';
      (mv);
      inc(iLeft, 200);
    end;
  end
  else //否则不打印页脚上的字段列表
  begin

  end;
end;

function : string;
begin
  result := FFileName;
end;

procedure (const Value: string);
begin
  FFileName := value;
  (FFileName);
end;

procedure (ReportName: string);
var
  i: integer;
  s: string;
begin
  if ReportName = '' then exit;
  s := '';
  for i:=0 to -1 do
    s := s + [i] + ',';
  delete(s, length(s), 1); //删除最后一个','号
  (ReportName, 'Title', s);

  s := '';
  for i:=0 to -1 do
    s := s + [i] + ',' ;
  delete(s, length(s), 1);
  (ReportName, 'Detail', s);

  s := '';
  for i:=0 to -1 do
    s := s + [i] + ',' ;
  delete(s, length(s), 1);
  (ReportName, 'Footer', s);
end;

//从配置文件中得到要打印的字段,然后加入Listview中
procedure (ReportName: string);
var
  s: string;
begin
  FReportName := ReportName;
  := (ReportName, 'Title', '');
  //注意:当调用了LoadFieldList之后,再执行SetGrid,那么DetailFields的字段列表会被FGridFields的字段列表覆盖
  := (ReportName, 'Detail', '');
  := (ReportName, 'Footer', '');

  if > 0 then
  begin
    s := [-1]; ;//最后一个字段的前缀就是明细数据条的数据源
    := Copy(s, 1, AnsiPos('.', s)-1);
  end;
end;

function : TStrings;
begin
  result := FDetailFields;
end;

procedure (const Value: TStrings);
begin
  (value);
end;

procedure (Datasets: array of TDataset);
var
  i, j, iLen: integer;
  Dataset: TDataset;
  TvFields: TTreeView;
  MainNode: TTreeNode;
  b: Boolean;
begin
  for i:=0 to high(Datasets) do //把数据集逐一赋给FDatasetArray
  begin
    b := false;
    for j:=0 to high(FDatasetArray) do //先判断数组中是否已存在要加入的数据集
    begin
      b := false;
      if FDatasetArray[j].Name = Datasets[i].Name then
      begin
        b := true;
        break;
      end;
    end;

    if not b then
    begin
      iLen := high(FDatasetArray) + 2; //得到数组的上标(动态数组作为参数,其下标总是从0开始),长度加2才是数量
      SetLength(FDatasetArray, iLen);
      SetLength(FRDataset, iLen);

      dec(iLen);
      FDatasetArray[iLen] := Datasets[i]; //数量减去1才是维数
      FRDataset[iLen] := (frmCommon);
      FRDataset[iLen].DataSet := FDatasetArray[iLen];
      frDataset[iLen].Name := FDatasetArray[iLen].Name;
    end;

  end;

  tvFields := (nil); //因为打算用一个树形结构来表示所有字段,为便于赋值,干脆把整棵树赋值给frmCommon(有时间改为Stream比较好)
  try
    := frmCommon;
    for i:=0 to high(FDatasetArray) do
    begin
      Dataset := FDatasetArray[i];
      MainNode := (nil, );
      for j:=0 to -1 do
        (MainNode, [j].FieldName
          + '[' + [j].DisplayLabel + ']');
    end;

    (); //把树赋给frmCommon
  finally
    ;
  end;
end;

function : TCustomGrid;
begin
  result := FGrid;
end;

procedure (const Grid: TCustomGrid);
var
  PropInfo:PPropInfo;
  DBGrid: TDBGrid;
  i: integer;
  Dataset: TDataset;
begin
  FGrid := Grid;
  if not assigned(Grid) then exit;
  PropInfo := GetPropInfo(Grid,'DataSource'); //查找Grid是否有DataSource属性
  if PropInfo <> nil then //如果有
  begin
    DBGrid := (nil);
    try
      //得到表格数据源的数据集的名称
      //为了不引用也能得到,所以使用了RTTI
      //这样的好处不仅在TDBGridEh中体现,以后使用了别的DBGrid,只要它有DataSource,也能得到其数据集的名称
      SetObjectProp(DBGrid, 'DataSource', GetObjectProp(Grid, 'DataSource'));
      if not assigned() then exit;
      Dataset := ;
      if Dataset = nil then exit;
      SetDataset(DataSet);

      PropInfo := GetPropInfo(Grid, 'Columns'); //查找Grid是否有Columns属性
      if PropInfo <> nil then
      begin
        SetObjectProp(DBGrid, 'Columns', GetObjectProp(Grid, 'Columns'));
        if not assigned() then exit;
        for i:=0 to -1 do
        begin
          if [i].Visible then //只挑选可视的字段
          begin
            ( + '.' + [i].FieldName + '=' + [i].DisplayName);
            ( + '.' + [i].FieldName + '=' + IntToStr([i].Width));
          end;
        end;
      end;

      PropInfo := GetPropInfo(Grid, 'RowHeight'); //查找Grid是否有RowHeight属性
      if PropInfo <> nil then
      begin
        FGridRowHeight := GetPropValue(Grid, 'RowHeight', false);
        := FGridRowHeight;
      end;

      //把DetailFields的字段列表清空,并把Grid的字段加入DetailFields中
      //这样处理有个问题:当调用了LoadFieldList后,DetailFields从FGridFields中得到的字段列表会被配置文件中的字段列表覆盖
      ;
      for i:=0 to -1 do
        ([i]);

    finally
      FreeAndNil(DBGrid);
    end;
  end;

end;

function : TStrings;
begin
  result := FVariantList;
end;

procedure (const Value: TStrings);
var
  i: integer;
begin
  //注意:如果要传字符串变量,那么格式为'Name=''Value''';如果要传数值变量,那么格式为'Name=1'
  (value);
  [' VariantList'] := ''; //变量root
  for i:=0 to -1 do //把变量列表加入数据字典
    [[i]] := [i];
end;

end.