zoukankan      html  css  js  c++  java
  • 在Report Builder中增加自定义函数

    前几日,网友Miracle提到此问题,由于自己也被此问题困扰过,因此决心实现一下;此外,对于这个问题,我想在应用中如果用到了RB并使用了她的最终用户方案即RAP技术,那么这个问题也肯定会遇到。具体来说也就是这样一种情况:假若在一个应用中可能有几个全局的变量,我们想在RB的报表中体现出来,作为报表的内容,怎么实现?

      没想到,原来RB对此在其RAP技术中已经支持的非常完善,在参考了其demo及help文档,很容易就实现了,因此,相信你也能:),当然,在你继续看下去之前,应该知道什么是Report Builder以及RAP,并且在你的应用中已经用到,要不然,你可能困难了。

    实现方法

    先确定一下我们要实现什么?
    在RAP中,我们增加一个这样的函数:
    Function GetCorpName:String//取得软件用户的公司名称。
    注意,我们是要在RAP中实现。如果对于Delphi的代码,这很易实现,比如这是在Delphi实现代码:
    Function GetCorpName:String
    Var
    LsIni:TiniFile;
    Begin
    With lsIni.Create(‘test.ini’) do
    Try
    Result:=ReadString(‘main’,’CorpName’,’’);
    Finally
    Free;
    End;
    End;
      现在已经知道了要干什么了。那么如何利用RB的方案来实现呢?具体来说,要经过如下几步来实现:
    1.定义TraSystemFunction的子类;
    2.重载TraSystemFunction几个方法;
    3.利用raRegisterFunction注册实现的子类;

    第一步,如何定义 TraSystemFunction?
      在RAP中,已经实现了TraSystemFunction类的5个子类,因此我们要实现的类可以直接从现有的子类中继承。看一下这几个子类:
    TraSystemFunction
    |
    -- TraStringFunction
    |
    -- TraConversionFunction
    |
    -- TraFormatFunction
    |
    -- TraDateTimeFunction
    |
    TraMathFunction
      由于要定义的函数GetCorpName返回的是String,所以从TraStringFunction继承就是了。具体定义如下:
    TmyGetCorpNameFunction=class (TmyStringFunction)
    Public
    Procedure ExecuteFunction(aParams:TraParamList);override;
    Class Function GetSignature:String;override;
    Class Function HasParams:Boolean;override; < End;
    这里,TraSystemFunction及raRegisterFunction在raFunc单元中定义,因此你一定要在uses子句中引用raFunc。
    第二步,重载TraSystemFunction的方法
      通过第一步的定义,已经看到在TmyGetCorpNameFunction类中定义了三个方法,那你一定知道要重载的具体内容了。

      1.Procedure ExecuteFunction(aParams:TraParamList);override;
      从其名称不难看出,这里的代码就是在RAP中具体调用的Delphi原生代码!参数aParams的作用是:引入自定义函数参数列表及返回函数的结果。比如:自定义的函数的格式为:
    Function MaxNum(I,I :integer):Boolean;
      那么,当在RAP中以这样的格式调用MaxNum(12,20),那么12,20会通过aParams传进去,其Boolean型结果也会通过aParams返回。为了取得传进的参数,我们会用到
    GetParamValue(ParamIndex,LocalVar);
    接上例MaxNum,为了得到12及20,这样写代码:
    GetParamValue(0,lsInteger1);//返回12
    GetParamValue(1,lsInteger2);//返回20
      这里,ParamIndex对应传进参数的序号;LocalVar是接受参数值的临时变量,也就是需要在ExecuteFunction中声明的,当然是有几个参数需要定义几个这样的变量。
    如何返回函数的结果呢?这要用到
    SetParamValue(MaxParamIndex,lsResult);
    接上例,则这样调用:SetParamValue(2,lsResult);
    为什么MaxParamIndex传进2作为其值呢?通过上面的GetParamValue就可以看出来,不是吗?

    2. Class Function GetSignature:String;override;
      这个函数返回RAP中自定义函数的声明,注意,不是Delphi版本中声明。
    例如,对于上文中提到GetCorpName函数,应这样实现:
    class Function TmyGetCorpName.GetSignature:String;
    begin
    result:=’function GetCorpName:String’;
    end;
    实际上,此函数结果在注册后会出现在报表设计器CALC页中的Tool Language中,以供用户使用。
    3.Class Function HasParams:Boolean;override;
      此函数从其名称可以猜到是判断自定义函数是否有参数。实际就是如此。如果返回True,则自定义函数有参数,否则没有参数。对于GetCorpName自定义函数,就要这么写代码:
    class function HasParams:Boolean;override;
    begin
    result:=false;//此定义函数无参数
    end;
      实际上,HasParams就是针对无参数的自定义函数而作,如果要实现的自定义函数有参数,那就不需要重载此函数。
      关于此要重载的函数,除了上述说明外,还有一个isFunction,如果你定义的函数是一个过程,则需要重载,以返回False.实现过程同HasParams差不多。
    第三步,注册为定义函数而定制的类
      这里要用到raRegisterFunction(const aFunctionName: String; aClass: TraSystemFunctionClass);
    一个定制的类,只有通过raRegisterfunction的注册,才能使其实现在函数出现在 RAP的Code ToolBox中。此外,注册函数应写在单元的initialization段中。
    如:
    initialization
    raRegisterFunction(‘GetCorpName’,TmyGetCorpFunction);
    写到这里,一个无参数的自定义函数GetCorpName已经定义完成,最后让我们看看其完整的代码:
    unit myRAPFUC;

    interface
    uses
    SysUtils, Windows, Messages, Classes, Graphics, Controls,
    Forms, Dialogs, raFunc, ppRTTI, DataMdl;

    type

    TMyGetCorpNameFunction = class(TraStringFunction)
    public
    procedure ExecuteFunction(aParams: TraParamList); override;
    class function GetSignature: string; override;
    class function HasParams: boolean; override;
    end;

    implementation

    procedure TmyGetCorpNameFunction.ExecuteFunction(aParams: TraParamList);
    var
    lsResult: string;
    begin
    lsResult := dm.czyMC;
    SetParamValue(0, lsResult);
    end;

    class Function tmyGetCorpNameFunction.GetSignature: string;
    begin
    result := ’Function GetCorpName:String;’;
    end;

    class function TmyGetCorpNameFunction.HasParams: Boolean;
    begin
    result := false;
    end;

    initialization
    raRegisterFunction(’GetCorpName’, TmyGetCorpNameFunction);

    end.
      所附单元在delphi4 rb5enterprise通过,我作了最大的简化以减少入门的难度,当你掌握此处的实现方法,就可以进一步学习定义带参数的自定义函数等等,实际上,你可以在ExecuteFunction中尽情的发挥你的想象力,好让你的用户在自定义报表过程中不受你Delphi代码的束缚!
      尽情享受Report builder带给我们的自由空间吧。

  • 相关阅读:
    Spring IoC 容器和 AOP
    MySQL 锁与事务控制
    MySQL 存储引擎的选择
    如何理解MySQL 索引最左前缀原则
    MySQL 索引
    Java 线程池
    Java多线程 ReentrantLock、Condition 实现生产者、消费者协作模式
    Java多线程并发中 CAS 的使用与理解
    Java多线程中协作机制
    Mysql-SQL生命周期-show profile
  • 原文地址:https://www.cnblogs.com/leonkin/p/2363666.html
Copyright © 2011-2022 走看看