Module cCrystal.pkg

     1// Package Version 1.0
     2Use Windows.pkg
     3Use ptrmodes.pkg
     4Use LanguageText.Pkg
     5Use cCrystalcraxddrt.pkg
     6Use cCrystalCDO32.pkg
     7Use cCrystalPreview.pkg // activeX preview
     8Use seq_chnl.pkg
     9Use cApplication.pkg
    10
    11Struct tCRWCDO
    12    String sTable
    13    Handle hoCDO
    14End_Struct
    15
    16Define ttDATAFLEX for "crdb_p2bdfapi.dll" // .dat
    17Define ttCDO      for "crdb_fielddef.dll" // .ttx
    18
    19// This object is the server for all reports
    20Handle ghoCrystalReportServer
    21Move 0 to ghoCrystalReportServer
    22
    23
    24Class cCrystalReport is a cComAutomationObject
    25    Import_Class_Protocol cCrystalIReport
    26    Import_Class_Protocol cCrystalIReportEvent
    27
    28    Procedure Construct_Object
    29        Forward Send Construct_Object
    30        Set psProgID to "CrystalDesignRunTime.Report"
    31        Set psEventId to "{AF376802-6120-4E28-96DD-63FD2DC27B7A}"
    32        Set peAutoCreate to acNoAutoCreate
    33
    34        //Path to report file that was opened.
    35        Property String psReportLocation
    36
    37        //available objects within the report class
    38        
    39        Property Handle phoDatabase 0
    40
    41        
    42        Property Handle phoDatabaseTables 0
    43
    44        
    45        Property Handle phoFormulaFieldDefinitions 0
    46
    47        
    48        Property Handle phoParamFieldDefinitions 0
    49
    50        
    51        Property Handle phoSortFields 0
    52
    53        //holds handles of all paramterFieldDefinition objects
    54        
    55        Property Handle[] phoParams
    56
    57        //holds handles of all table objects
    58        
    59        Property Handle[] phoTables
    60
    61        //holds handles of all cdo objects
    62        
    63        Property tCRWCDO[] pCDOs
    64
    65        //## NOT FOR USE WITH MAINREPORT ##
    66        // name of subreport (if main report, it is empty)
    67        Property String  psSubReportName ""
    68
    69        //## NOT FOR USE WITH SUBREPORTS ##
    70
    71        
    72        Property Handle phoExportObject     0
    73
    74        
    75        Property Handle phoPreviewObject    0
    76
    77        //holds name and handle of all subreport objects
    78        
    79        Property Handle[] phoSubReports
    80
    81        
    82        Property Boolean pbSubReportsLoaded FALSE
    83
    84        //Printer specific properties
    85        Property Integer piPrinterCopies       1
    86        Property Integer piPrinterStartPage    1
    87        Property Integer piPrinterEndPage      0 // we treat zero as "to end of report"
    88        Property Boolean pbPrinterCollate      TRUE
    89        Property Boolean pbPrinterPrompt       False
    90
    91        Property Boolean pbExportPrompt        TRUE
    92
    93    End_Procedure // Construct_Object
    94
    95    //--------------------
    96    // Database Functions
    97
    98    //Returns handle to database object
    99    Function DatabaseObject Returns Handle
   100        Handle hoDatabase
   101        Variant vDatabase
   102        Boolean bAttached
   103
   104        Get phoDatabase to hoDatabase
   105        If (Not(hoDatabase)) Begin
   106            Get Create U_cCrystalDatabase to hoDatabase
   107            Get ComDatabase To vDatabase
   108            Set pvComObject Of hoDatabase To vDatabase
   109            Get IsComObjectCreated of hoDatabase to bAttached
   110            If (bAttached) Begin
   111                Set phoDatabase to hoDatabase
   112            End
   113            Else Begin
   114                Send Destroy of hoDatabase
   115                Move 0 to hoDatabase
   116                Error DFERR_CRYSTAL_REPORT C_$UnableToCreateDatabaseObject
   117            End
   118        End
   119        Function_Return hoDatabase
   120    End_Function // DatabaseObject
   121
   122    //Creates all table objects
   123    
   124    Procedure LoadDatabaseTables Handle hoDatabaseTables
   125        Variant vComDatabaseTable
   126        Handle hoDatabaseTable
   127        Integer iTableItem iTableCount
   128        Boolean bAttached
   129        Handle[] hoTables
   130        // create all tables
   131        Get ComCount of hoDatabaseTables to iTableCount
   132        For iTableItem From 1 To iTableCount
   133            Get ComItem of hoDatabaseTables Item iTableItem To vComDatabaseTable
   134            Get Create of hoDatabaseTables U_cCrystalDatabaseTable To hoDatabaseTable
   135            //Set Name of hoDatabaseTable to ("oTable_"-ComName(hoDatabaseTable)) // for debugging purposes only
   136            Set pvComObject Of hoDatabaseTable To vComDatabaseTable
   137            Get IsComObjectCreated Of hoDatabaseTable To bAttached
   138            If (bAttached) Begin
   139                Move hoDatabaseTable to hoTables[iTableItem-1]
   140            End
   141            Else Begin
   142                Send Destroy of hoDatabaseTable
   143                Error DFERR_CRYSTAL_REPORT C_$UnableToCreateTableObject
   144            End
   145        Loop
   146        Set phoTables to hoTables
   147    End_Procedure // LoadDatabaseTables
   148
   149    // Returns handle to database collection comautomation object. Creates connection if connection does not already exist.
   150    // This is used privately to fill the table array which is accessed publicly via get TableObjects.
   151    // We could destroy phoDatatables if we wanted but we are not currently doing so.
   152    
   153    Function DatabaseTablesObject Returns Handle
   154        Boolean bAttached
   155        Handle hoDatabaseTables hoDatabase
   156        Variant vComDatabaseTables
   157
   158        Get phoDatabaseTables To hoDatabaseTables
   159        If Not (hoDatabaseTables) Begin
   160            Get DatabaseObject to hoDatabase // this will return a valid object or generate an error
   161            If hoDatabase Begin
   162                Get Create Of hoDatabase U_cCrystalDatabaseTables To hoDatabaseTables
   163                Get ComTables Of hoDatabase To vComDatabaseTables
   164                Set pvComObject Of hoDatabaseTables To vComDatabaseTables
   165                //Set Name of hoDatabaseTables to "oComDatabaseTables" // for debugging purposes only
   166                Get IsComObjectCreated Of hoDatabaseTables To bAttached
   167                If (bAttached) Begin
   168                    Send LoadDatabaseTables hoDatabaseTables
   169                    Set phoDatabaseTables To hoDatabaseTables
   170                End
   171                Else Begin
   172                    Send Destroy of hoDatabaseTables // remove main object if failed.
   173                    Move 0 to hoDatabaseTables
   174                    Error DFERR_CRYSTAL_REPORT C_$UnableToCreateTableObjects
   175                End
   176            End
   177        End
   178        Function_Return hoDatabaseTables
   179    End_Function // DatabaseTablesObject
   180
   181    //Returns an array of tables in report
   182    Function TableObjects Returns Handle[]
   183        Handle hoDatabaseTables
   184        Handle[] hoTables
   185
   186        If Not (phoDatabaseTables(Self)) Begin
   187            // If tables haven't been loaded, load them now.
   188            Get DatabaseTablesObject to hoDatabaseTables
   189        End
   190        Get phoTables to hoTables
   191        Function_Return hoTables
   192    End_Function // TableObjects
   193
   194    //Returns handle of table from name. Note:ComName returns the object name. This is the
   195    //name given to a file (such as an alias name). ComLocation and ComConnectBufferString
   196    //return the file the connection is made to.
   197    Function GetTableObjectByName String sTable Returns Handle
   198        String sHoldName
   199        Integer iTableItem iTableCount
   200        Handle[] hoTables
   201
   202        Get TableObjects to hoTables
   203        Move (SizeOfArray(hoTables)) to iTableCount
   204        For iTableItem From 0 To (iTableCount-1)
   205            Get ComName Of hoTables[iTableItem] To sHoldName
   206            If (Lowercase(sHoldName) = Lowercase(sTable)) Begin
   207                Function_Return hoTables[iTableItem]
   208            End
   209        Loop
   210        Function_Return 0
   211    End_Function // GetTableObjectByName
   212
   213    // Set the location of a dataflex datafile (.dat or .int) to the data directory
   214    Function LocateDFFile Handle hoDatabaseTable Returns Boolean
   215        Integer iPath iNumPaths
   216        Boolean bExists
   217        String DirSep sTable sLocation
   218        //
   219        Move (Sysconf(SYSCONF_DIR_SEPARATOR)) To DirSep
   220        Get ComConnectBufferString of hoDatabaseTable to sTable
   221        While (Pos("=",sTable))
   222            Move (Replace(Left(sTable, Pos("=",sTable)), sTable, "")) to sTable
   223        Loop
   224        While (Pos(DirSep,sTable))
   225            // strip out path until left with filename + extension
   226            Move (Replace(Left(sTable, Pos(DirSep,sTable)), sTable, "")) to sTable
   227        Loop
   228        Get psDataPath Of (phoWorkspace(ghoApplication)) To sLocation
   229        If (Right(sLocation, 1) <> DirSep) Begin
   230            Append sLocation DirSep
   231        End
   232        Get CountOfPaths Of (phoWorkspace(ghoApplication)) sLocation To iNumPaths
   233        Move (FALSE) to bExists
   234        For iPath From 1 To iNumPaths
   235            Get psDataPath Of (phoWorkspace(ghoApplication)) To sLocation
   236            Get PathAtIndex Of (phoWorkspace(ghoApplication)) sLocation iPath To sLocation
   237            If (Right(sLocation, 1) <> DirSep) Begin
   238                Append sLocation DirSep
   239            End
   240            Append sLocation sTable
   241            File_Exist sLocation bExists
   242        Until ((bExists) Or (iPath=iNumPaths))
   243        If (bExists) Begin
   244            Set ComLocation Of hoDatabaseTable To sLocation
   245        End
   246        Else Error DFERR_CRYSTAL_REPORT (SFormat(C_$NoLocationSpecifiedForTable, sTable))
   247        Function_Return bExists
   248    End_Function // LocateDFFile
   249
   250    //## NOT FOR USE WITH SUBREPORTS ##
   251    // Locate all dataflex files for report and sub-report assign them to the data directory.
   252    // If table cannot be relocated, process stops and false is returned; else return true.
   253    Function LocateDFFiles Returns Boolean
   254        Handle[] SubReportArray
   255        Handle   hoSubReport
   256        Integer  iReportItem iReportCount
   257        Boolean  bOk
   258
   259        // locate for main report
   260        Get LocateDFFilesForReport to bOk
   261        If (Not(bOk)) Function_return False
   262
   263        // Set table locations for subreports
   264        Get SubReportObjects to SubReportArray
   265        Move (SizeOfArray(SubReportArray)) to iReportCount
   266        For iReportItem From 0 To (iReportCount-1)
   267            Move SubReportArray[iReportItem] to hoSubReport
   268            If (hoSubReport) Begin
   269                Get LocateDFFilesForReport of hoSubReport to bOk
   270                If (not(bOk)) Function_Return False
   271            End
   272        Loop
   273        Function_return True
   274    End_Function // LocateDFFiles
   275
   276
   277    // Locate all dataflex files for this one report assign them to the data directory.
   278    // If table cannot be relocated, process stops and false is returned; else return true
   279    
   280    Function LocateDFFilesForReport Returns Boolean
   281        String sTableType
   282        Integer iTableItem iTableCount
   283        Handle hoDatabaseTable
   284        Handle[] hoTables
   285        Boolean bOK
   286        //
   287        Get TableObjects to hoTables
   288        Move (SizeOfArray(hoTables)) to iTableCount
   289        // Loop through all tables
   290        For iTableItem From 0 To (iTableCount-1)
   291            Move hoTables[iTableItem] To hoDatabaseTable
   292            If (hoDatabaseTable) Begin
   293                // Check database type and set location if it matches.
   294                Get ComDllName of hoDatabaseTable to sTableType
   295                If (sTableType = ttDATAFLEX and ghoApplication) Begin
   296                    Get LocateDFFile hoDatabaseTable to bOK
   297                    If (Not(bOK)) Function_return False // Exit loop with error (so we don't get redundant errors)
   298                End
   299            End
   300        Loop
   301        Function_return True // it worked
   302    End_Function // LocateDFFilesForReport
   303
   304    //## NOT FOR USE WITH SUBREPORTS ##
   305    // Creates a CDO based on a TTX file.
   306    Function CreateCDO String sFileName Returns Handle
   307        Handle hoCDO
   308        tCRWCDO[] CRWCDOs
   309        Integer iCDOCount iFieldType iChannel
   310        Boolean bOK
   311        String sLocation sLine sField sFieldType DirSep
   312
   313        // Create CDO object
   314        Get Create U_cCrystalCrystalComObject to hoCDO
   315        // Set Name of hoCDO to ("oCDO_"-sFilename) // for debugging purposes only
   316        Send CreateComObject of hoCDO
   317        Get IsComObjectCreated of hoCDO To bOk
   318        If (bOk) Begin
   319            Get pCDOs to CRWCDOs
   320            Move (SizeOfArray(CRWCDOs)) to iCDOCount
   321            //
   322            Move (Sysconf(SYSCONF_DIR_SEPARATOR)) To DirSep
   323            Indicate Err FALSE
   324            Move (lowercase(sFileName)) to sFileName
   325            //
   326            Send ComReset of hoCDO
   327            If (ghoApplication) begin
   328                // Locate Filename in Datapath. This should be replaced with report directory.
   329                Get psDataPath Of (phoWorkspace(ghoApplication)) To sLocation
   330                If (Right(sLocation, 1) <> DirSep) Begin
   331                    Append sLocation DirSep
   332                End
   333                Get psReportLocation to sLocation
   334                If (Right(sLocation, 1) <> DirSep) Begin
   335                    Append sLocation DirSep
   336                End
   337            End
   338            File_Exist (sLocation-sFileName) bOk // True if file exists
   339            // If file exists...
   340            If (bOk) Begin
   341                Get Seq_New_Channel to iChannel
   342                If (iChannel = DF_SEQ_CHANNEL_NOT_AVAILABLE) Begin
   343                    Error DFERR_CRYSTAL_REPORT C_$NoIOChannelAvailableForCDO
   344                    Move False to bOk
   345                End
   346                Else Begin
   347                    Direct_Input channel iChannel (sLocation-sFileName)
   348                    Repeat
   349                        Readln sLine
   350                        If ( Not(SeqEof) and Trim(sLine<>"")) Begin
   351                            // Reset variables
   352                            Move "" to sField
   353                            Move "" to sFieldType
   354                            Move 0  to iFieldType
   355                            //
   356                            Move (Left(sLine, (Pos(" ", sLine)))) to sField
   357                            Move (Replace(sField, sLine, "")) to sLine
   358                            Move (Trim(sLine)) to sLine
   359                            If (Pos(" ", sLine)) Begin
   360                                Move (Left(sLine, (Pos(" ", sLine)))) to sFieldType
   361                            End
   362                            Else Move sLine to sFieldType
   363                            Move (Trim(lowercase(sFieldType))) to sFieldType
   364                            //
   365                            If (sFieldType = "blob")          Move OLE_VT_BSTR to iFieldType
   366                            Else If (sFieldType = "boolean")  Move OLE_VT_BOOL to iFieldType
   367                            Else If (sFieldType = "byte")     Move OLE_VT_UI1  to iFieldType
   368                            Else If (sFieldType = "currency") Move OLE_VT_CY to iFieldType
   369                            Else If (sFieldType = "date")     Move OLE_VT_DATE to iFieldType
   370                            Else If (sFieldType = "datetime") Move OLE_VT_DATE to iFieldType
   371                            Else If (sFieldType = "long")     Move OLE_VT_I4 to iFieldType
   372                            Else If (sFieldType = "memo")     Move OLE_VT_BSTR to iFieldType
   373                            Else If (sFieldType = "number")   Move OLE_VT_R8 to iFieldType
   374                            Else If (sFieldType = "short")    Move OLE_VT_I2 to iFieldType
   375                            Else If (sFieldType = "string")   Move OLE_VT_BSTR to iFieldType
   376                            Else                              Move False to bOK // we didn't find a legal datatype in the ttx file
   377
   378                            If (bOk) Begin
   379                                Get ComAddField of hoCDO (Trim(sField)) iFieldType to bOk
   380                                If (Not(bOk)) Begin
   381                                    Error DFERR_CRYSTAL_REPORT (SFormat(C_$ErrorCreatingCDOField, sField))
   382                                End
   383                            End
   384                            else Begin
   385                                Error DFERR_CRYSTAL_REPORT (SFormat(C_$UnknownFieldTypeForCDO, sFieldType, sField))
   386                            end
   387                        end
   388                    Until ((SeqEof) or not(bOk))
   389                    Close_Input channel iChannel
   390                    Send Seq_Release_Channel iChannel
   391                End
   392            End
   393            Else Begin
   394                Error DFERR_CRYSTAL_REPORT (SFormat(C_$UnableToLocateCDOFile, sFileName))
   395            End
   396        End
   397        Else Begin
   398            Error DFERR_CRYSTAL_REPORT (SFormat(C_$UnableToCreateCDOobject,sFileName))
   399        End
   400
   401        If bOk Begin
   402            Move (Left(sFileName, (Length(sFileName)-4))) to CRWCDOs[iCDOCount].sTable
   403            Move hoCDO to CRWCDOs[iCDOCount].hoCDO
   404            Set pCDOs to CRWCDOs
   405        End
   406        Else Begin
   407            Send Destroy of hoCDO
   408            Move 0 to hoCDO
   409        End
   410
   411        Function_Return hoCDO
   412    End_Function // CreateCDO
   413
   414    //## NOT FOR USE WITH SUBREPORTS ##
   415    // Converts a safe array to the correct datatypes used by the CDO and attaches data to CDO.
   416    // Example usage within OnInitializeReport: Send AppendCDOData of hoReport hoCDO vData
   417    Procedure AppendCDOData Handle hoCDO Variant[][] vCDOData
   418        Integer iCol iMaxCol iItem iCount iType
   419
   420        // Get number of columns
   421        Get ComGetColCount of hoCDO to iMaxCol
   422        // Get number of rows of data in array
   423        Move (SizeOfArray(vCDOData)) to iCount
   424        For iCol from 0 to (iMaxCol-1)
   425            Get ComGetFieldType of hoCDO iCol to iType
   426            // For each column Loop through each row and set the proper datatypes according to the CDO
   427            For iItem from 0 to (iCount-1)
   428                If (iType=OLE_VT_I4)            Move (Cast(vCDOData[iItem][iCol],Integer))   to vCDOData[iItem][iCol]
   429                Else If (iType = OLE_VT_BSTR)   Move (Cast(vCDOData[iItem][iCol],String))    to vCDOData[iItem][iCol]
   430                Else If (iType = OLE_VT_BOOL)   Move (Cast(vCDOData[iItem][iCol],Boolean))   to vCDOData[iItem][iCol]
   431                Else If (iType = OLE_VT_CY)     Move (Cast(vCDOData[iItem][iCol],Currency))  to vCDOData[iItem][iCol]
   432                Else If (iType = OLE_VT_DATE)   Move (Cast(vCDOData[iItem][iCol],Date))      to vCDOData[iItem][iCol]
   433                Else If (iType = OLE_VT_R8)     Move (Cast(vCDOData[iItem][iCol],Real))      to vCDOData[iItem][iCol]
   434                Else If (iType = OLE_VT_I2)     Move (Cast(vCDOData[iItem][iCol],Short))     to vCDOData[iItem][iCol]
   435                Else If (iType = OLE_VT_UI1)    Move (Cast(vCDOData[iItem][iCol],UChar))     to vCDOData[iItem][iCol]
   436            Loop
   437        Loop
   438        If (SizeOfArray(vCDOData)) Send ComAddRows of hoCDO vCDOData
   439    End_Procedure // AppendCDOData
   440
   441
   442    //## NOT FOR USE WITH SUBREPORTS ##
   443    // Attaches an existing CDO to a report. This is usefull when a cdo is shared by multiple reports.
   444    // Create the cdo once and attach it to each report.
   445    Procedure AttachCDO String sFileName Handle hoCDO
   446        Integer iCDOCount
   447        tCRWCDO[] CRWCDOs
   448        Get pCDOs to CRWCDOs
   449        Move (SizeOfArray(CRWCDOs)) to iCDOCount
   450        Move (Left(sFileName, (Length(sFileName)-4))) to CRWCDOs[iCDOCount].sTable
   451        Move hoCDO to CRWCDOs[iCDOCount].hoCDO
   452        Set pCDOs to CRWCDOs
   453    End_Procedure // AttachCDO
   454
   455    //## NOT FOR USE WITH SUBREPORTS ##
   456    // Attaches CDO data to report and subreport CDO objects
   457    
   458    Procedure AssignCDODataSources tCRWCDO[] CRWCDOs
   459        Handle hoCDO hoTable
   460        Integer iCDOItem iCDOCount
   461        String sTable DirSep sPath sType
   462        Move (Sysconf(SYSCONF_DIR_SEPARATOR)) To DirSep
   463        Move (SizeOfArray(CRWCDOs)) to iCDOCount
   464        For iCDOItem from 0 to (iCDOCount-1)
   465            Move CRWCDOs[iCDOItem].hoCDO to hoCDO
   466            If (hoCDO) Begin
   467                Get GetTableObjectByName CRWCDOs[iCDOItem].sTable to hoTable
   468                If (hoTable) Begin
   469                    Get ComDllName of hoTable to sType
   470                    If (sType=ttCDO) Begin
   471                        // Setting the datasource seems to be enough. Even though the TTX file is
   472                        // not in the same location on the deployment end as the development end,
   473                        // this does not appear to need to be set. I am leaving this code here in
   474                        // case someone has a problem with it and needs to use it.
   475                        //Get ComConnectBufferString of hoTable to sTable
   476                        //While (Pos("=",sTable))
   477                        //    Move (Replace(Left(sTable, Pos("=",sTable)), sTable, "")) to sTable
   478                        //Loop
   479                        //While (Pos(DirSep,sTable))
   480                        //    // strip out path until left with filename + extension
   481                        //    Move (Replace(Left(sTable, Pos(DirSep,sTable)), sTable, "")) to sTable
   482                        //Loop
   483                        // Set TTX location to same as report
   484                        //Get psReportLocation to sPath
   485                        //Set ComLocation of hoTable to (sPath-sTable)
   486                        Send ComSetDataSource of hoTable (pvComObject(hoCDO)) 3 1
   487                    End
   488                End
   489             End
   490        Loop
   491    End_Procedure // AssignCDODataSources
   492
   493
   494    // End Database functions
   495
   496
   497    //--------------------
   498    // Formula Functions
   499
   500    //Returns handle to report formula fields collection comautomation object.
   501    //Creates connection if connection does not already exist.
   502    // This is used privately by AssignFormula
   503    
   504    Function FormulaFieldDefinitionsObject Returns Handle
   505        Variant vFormulaFields
   506        Handle hoFormulaFieldDefinitions
   507        Boolean bAttached
   508
   509        Get phoFormulaFieldDefinitions To hoFormulaFieldDefinitions
   510        If Not (hoFormulaFieldDefinitions) Begin
   511            Get Create U_cCrystalFormulaFieldDefinitions To hoFormulaFieldDefinitions
   512            Set phoFormulaFieldDefinitions To hoFormulaFieldDefinitions
   513            Get ComFormulaFields To vFormulaFields
   514            Set pvComObject Of hoFormulaFieldDefinitions To vFormulaFields
   515            //Set Name of hoFormulaFieldDefinitions to "oComFormulaFieldDefinitions" // for debugging purposes only
   516        End
   517
   518        Get IsComObjectCreated Of hoFormulaFieldDefinitions To bAttached
   519        If (Not(bAttached)) Begin
   520            Send Destroy of hoFormulaFieldDefinitions
   521            Move 0 to hoFormulaFieldDefinitions
   522            Set phoFormulaFieldDefinitions To 0
   523            Error DFERR_CRYSTAL_REPORT C_$UnableToCreateFormulaFieldsObject
   524        End
   525        Function_Return hoFormulaFieldDefinitions
   526    End_Function // FormulaFieldDefinitionsObject
   527
   528    //Sets report formula field to sValue
   529    Procedure AssignFormula String sFormulaName String sValue
   530        Variant vFormulaField
   531        Handle hoFormulaFieldDefinitions hoFormulaFieldDefinition
   532        Boolean bAttached bDisplay
   533        // Find formula field and set it's value
   534        Get FormulaFieldDefinitionsObject To hoFormulaFieldDefinitions
   535        If (hoFormulaFieldDefinitions) Begin
   536            Get Create Of hoFormulaFieldDefinitions U_cCrystalFormulaFieldDefinition To hoFormulaFieldDefinition
   537            // if the name is not found, RDC generates an error which we want to suppress. We will show the error ourselves
   538            Get Display_Error_State to bDisplay
   539            Set Display_Error_State to false
   540            Get ComGetItemByName Of hoFormulaFieldDefinitions sFormulaName To vFormulaField
   541            Set Display_Error_State to bDisplay
   542            Set pvComObject Of hoFormulaFieldDefinition To vFormulaField
   543            Get IsComObjectCreated Of hoFormulaFieldDefinition To bAttached
   544            //Set Name of hoFormulaFieldDefinition to ("Formula_"-ComName(hoFormulaFieldDefinition)) // for debugging purposes only
   545            If (bAttached) Begin
   546                Set ComText Of hoFormulaFieldDefinition To sValue
   547            End
   548            Else Begin
   549                Error DFERR_CRYSTAL_REPORT (SFormat(C_$UnableToCreateFormulaFieldObject, sFormulaName))
   550            End
   551            Send Destroy of hoFormulaFieldDefinition
   552        End
   553    End_Procedure // AssignFormula
   554
   555    // End Formula Functions
   556
   557
   558    //--------------------
   559    // Parameter Functions
   560
   561    //Creates all Parameter objects
   562    
   563    Procedure LoadParams handle hoParamFieldDefinitions
   564        Variant vParameterFieldDefinition
   565        Handle hoParameterFieldDefinition
   566        Integer iParamItem iParamCount
   567        Handle[] hoParams
   568        Boolean bAttached
   569        // create all parameters
   570        Get ComCount of hoParamFieldDefinitions to iParamCount
   571        For iParamItem From 1 To iParamCount
   572            Get ComItem of hoParamFieldDefinitions iParamItem To vParameterFieldDefinition
   573            Get Create  of hoParamFieldDefinitions U_cCrystalParameterFieldDefinition To hoParameterFieldDefinition
   574            Set pvComObject Of hoParameterFieldDefinition To vParameterFieldDefinition
   575            //Set Name of hoParameterFieldDefinition to ("Parameter_"-ComName(hoParameterFieldDefinition)) // for debugging purposes only
   576            Get IsComObjectCreated Of hoParameterFieldDefinition To bAttached
   577            If (bAttached) Begin
   578                Move hoParameterFieldDefinition to hoParams[iParamItem-1]
   579            End
   580            Else Begin
   581                Send Destroy of hoParameterFieldDefinition
   582                Error DFERR_CRYSTAL_REPORT C_$ErrorLoadingParamFieldObjects
   583            End
   584        Loop
   585        Set phoParams to hoParams
   586    End_Procedure // LoadParams
   587
   588    //Returns handle to report parameter fields collection comautomation object.
   589    //Creates connection if connection does not already exist.
   590    //This is used privately to load the params object which is accessed publicly via Get ParamObjects.
   591    // We could destroy phoParamFieldDefinitions when this is done, but we are currently not doing so.
   592    
   593    Function ParameterFieldDefinitionsObject Returns Handle
   594        Variant vParamFields
   595        Handle hoParamFieldDefinitions
   596        Boolean bAttached
   597
   598        Get phoParamFieldDefinitions To hoParamFieldDefinitions
   599        If Not (hoParamFieldDefinitions) Begin
   600            Get Create U_cCrystalParameterFieldDefinitions To hoParamFieldDefinitions
   601            Set phoParamFieldDefinitions To hoParamFieldDefinitions
   602            Get ComParameterFields To vParamFields
   603            Set pvComObject Of hoParamFieldDefinitions To vParamFields
   604            Send LoadParams hoParamFieldDefinitions
   605            //Set Name of hoParamFieldDefinitions to "oComParamFieldDefinitions" // for debugging purposes only
   606        End
   607
   608        Get IsComObjectCreated Of hoParamFieldDefinitions To bAttached
   609        If not (bAttached) Begin
   610            Send Destroy of hoParamFieldDefinitions
   611            Move 0 to hoParamFieldDefinitions
   612            Set phoParamFieldDefinitions To 0
   613            Error DFERR_CRYSTAL_REPORT C_$UnableToCreateParamFieldsObject
   614        End
   615        Function_Return hoParamFieldDefinitions
   616    End_Function // ParameterFieldDefinitionsObject
   617
   618    //Returns an array of Parameters in report
   619    Function ParamObjects Returns Handle[]
   620        Handle hoParamFieldDefinitions
   621        Handle[] hoParams
   622
   623        If Not (phoParamFieldDefinitions(Self)) Begin
   624            // If Params haven't been loaded, load them now.
   625            Get ParameterFieldDefinitionsObject To hoParamFieldDefinitions
   626        End
   627        Get phoParams to hoParams
   628        Function_Return hoParams
   629    End_Function // ParamObjects
   630
   631    //Returns handle of Parameter object from name
   632    Function GetParamObjectByName String sParamName Returns Handle
   633        String sHoldName
   634        Integer iParamItem iParamCount
   635        Handle[] hoParams
   636
   637        Get ParamObjects to hoParams
   638        Move (SizeOfArray(hoParams)) to iParamCount
   639        For iParamItem From 0 To (iParamCount-1)
   640            Get ComParameterFieldName Of hoParams[iParamItem] To sHoldName
   641            If (Lowercase(sHoldName) = Lowercase(sParamName)) Begin
   642                Function_Return hoParams[iParamItem]
   643            End
   644        Loop
   645        Function_Return 0
   646    End_Function // GetParamObjectByName
   647
   648    // End Parameter Functions
   649
   650
   651    //--------------------
   652    // Sort Functions
   653
   654    //Returns handle to report record sort fields collection comautomation object.
   655    //Creates connection if connection does not already exist.
   656    // This is used privately by the various sort methods (DeleteSortOrder, DeleteNthSortField and AppendSortField)
   657    
   658    Function SortFieldsObject Returns Handle
   659        Variant vSortFields
   660        Handle hoSortFields
   661        Boolean bAttached
   662
   663        Get phoSortFields To hoSortFields
   664        If Not (hoSortFields) Begin
   665            Get Create U_cCrystalSortFields To hoSortFields
   666            Set phoSortFields To hoSortFields
   667            Get ComRecordSortFields To vSortFields
   668            Set pvComObject Of hoSortFields To vSortFields
   669            //Set Name of hoSortFields to "oSortFields" // for debugging purposes only
   670        End
   671
   672        Get IsComObjectCreated Of hoSortFields To bAttached
   673        If Not (bAttached) Begin
   674            Send Destroy of hoSortFields
   675            Move 0 to hoSortFields
   676            Set phoSortFields To 0
   677            Error DFERR_CRYSTAL_REPORT C_$UnableToCreateRecordSortObject
   678            Function_Return 0
   679        End
   680        Function_Return hoSortFields
   681    End_Function // SortFieldsObject
   682
   683    //Deletes current record sort order
   684    Procedure DeleteSortOrder
   685        Handle hoSortFields
   686        Integer iSortItem iSortCount
   687        Get SortFieldsObject To hoSortFields
   688        If (hoSortFields) Begin
   689            Get ComCount of hoSortFields to iSortCount
   690            For iSortItem From 1 To iSortCount
   691                Send ComDelete Of hoSortFields 1
   692            Loop
   693        End
   694    End_Procedure // DeleteSortOrder
   695
   696    //Deletes current record sort order at position iSortItem
   697    Procedure DeleteNthSortField Integer iSortItem
   698        Handle hoSortFields
   699        Get SortFieldsObject To hoSortFields
   700        If (hoSortFields) Begin
   701            Send ComDelete Of hoSortFields iSortItem
   702        End
   703    End_Procedure // DeleteNthSortField
   704
   705    //Adds table field to record sort order
   706    Procedure AppendSortField String sTable String sField crSortDirection SortOrder
   707        Variant vField vFields
   708        Handle  hoSortFields hoDatabaseTable hoTableFields hoTableField
   709        Boolean bOK bAttached
   710        string  sName
   711        integer iField iFieldCount
   712
   713        Get SortFieldsObject To hoSortFields
   714        If (hoSortFields) Begin
   715            Get GetTableObjectByName sTable To hoDatabaseTable
   716            Move (hoDatabaseTable) to bOk
   717            If bOk Begin
   718                // create temporary object for all table fields
   719                Get Create U_cCrystalDatabaseFieldDefinitions to hoTableFields
   720                Get Create U_cCrystalDatabaseFieldDefinition  to hoTableField
   721                Get ComFields of hoDatabaseTable To vFields
   722                Set pvComObject of hoTableFields To vFields
   723                Get IsComObjectCreated of hoTableFields to bAttached
   724                If (bAttached) Begin
   725                      Move (lowercase(sField)) to sField // we will do a lc comparison
   726                      Get ComCount of hoTableFields to iFieldCount
   727                      Move False to bOk
   728                      Move 1 to iField
   729                      While (not(bOk) and (iField<=iFieldCount))
   730                            Get ComItem     of hoTableFields iField to vField
   731                            Set pvComObject of hoTableField To vField
   732                            Get IsComObjectCreated of hoTableField to bAttached
   733                            If (bAttached) Begin
   734                                Get ComDatabaseFieldName of hoTableField to sName
   735                                If (lowercase(sName)=sField) Begin
   736                                    Send ComAdd Of hoSortFields vField SortOrder
   737                                    Move True to bOk
   738                                end
   739                            end
   740                            Increment iField
   741                      End
   742                End
   743                Send Destroy of hoTableField
   744                Send Destroy of hoTableFields
   745           End
   746            // if can't find table or field or whatever
   747            If Not bOk Begin
   748                Error DFERR_CRYSTAL_REPORT (SFormat(C_$CannotAddSortField, sTable, sField))
   749            end
   750        End
   751    End_Procedure // AppendSortField
   752
   753    //Adds formula field to record sort order
   754    Procedure AppendSortFormulaField String sFormulaName crSortDirection SortOrder
   755        Variant vField
   756        Handle  hoSortFields hoFormulaFieldDefinitions
   757        Boolean bOK bAttached bDisplay
   758
   759        Get SortFieldsObject To hoSortFields
   760        If (hoSortFields) Begin
   761            // Find formula field and set it's value
   762            Get FormulaFieldDefinitionsObject To hoFormulaFieldDefinitions
   763            If (hoFormulaFieldDefinitions) Begin
   764                // if the name is not found, RDC generates an error which we want to suppress. We will show the error ourselves
   765                Get Display_Error_State to bDisplay
   766                Set Display_Error_State to false
   767                Get ComGetItemByName Of hoFormulaFieldDefinitions sFormulaName To vField
   768                Set Display_Error_State to bDisplay
   769            End
   770            //
   771            Move (Not(IsNullComObject(vField))) To bOK
   772            If bOk Begin
   773                Send ComAdd Of hoSortFields vField SortOrder
   774            End
   775            // if can't find formula field
   776            If Not bOk Begin
   777                Error DFERR_CRYSTAL_REPORT (SFormat(C_$CannotAddSortFormulaField, sFormulaName))
   778            end
   779        End
   780    End_Procedure // AppendSortFormulaField
   781
   782
   783    // End Sort Functions
   784
   785
   786    //--------------------
   787    // Begin Subreport Functions
   788
   789    //## NOT FOR USE WITH SUBREPORTS ##
   790    //Loads all subreports into an array. Note: All objects are of a report are checked to check if it is
   791    //a subreport. If it is a subreport, it is then created at the report level. All test objects are
   792    //destroyed when this process is complete.
   793    
   794    Procedure LoadSubreports
   795        handle  hoAreas hoArea hoSections hoSection hoReportObjs hoSubReport hoTemp
   796        variant vAreas vArea vSections vSection vReports vSubReport vReport
   797        integer iArea iSection iSubReport iKind iCount
   798        Handle[] SubReportArray
   799        string  sName
   800        Boolean bAttached
   801
   802        Get phoSubReports to SubReportArray
   803        Get Create U_cCrystalAreas to hoAreas
   804        Get Create U_cCrystalArea to hoArea
   805        Get Create U_cCrystalSections to hoSections
   806        Get Create U_cCrystalSection to hoSection
   807        Get Create U_cCrystalReportObjects to hoReportObjs
   808        Get Create U_cCrystalSubreportObject to hoTemp // temp subreport object to test ComKind
   809
   810        Get ComAreas to vAreas
   811        Set pvComObject Of hoAreas To vAreas
   812
   813        For iArea from 1 to (ComCount(hoAreas))
   814            Get ComItem     of hoAreas iArea to vArea
   815            Set pvComObject Of hoArea To vArea
   816
   817            Get ComSections of hoArea to vSections
   818            Set pvComObject Of hoSections To vSections
   819
   820            For iSection from 1 to (ComCount(hoSections))
   821                Get ComItem     of hoSections iSection to vSection
   822                Set pvComObject Of hoSection To vSection
   823
   824                Get ComReportObjects of hoSection to vReports
   825                Set pvComObject Of hoReportObjs To vReports
   826
   827                For iSubReport from 1 to (ComCount(hoReportObjs))
   828                    Get ComItem     of hoReportObjs iSubReport to vSubReport
   829                    Set pvComObject Of hoTemp To vSubReport
   830
   831                    Get ComKind of hoTemp to iKind
   832                    If (iKind=crSubreportObject) begin
   833                        Get ComSubReportName of hoTemp to sName
   834                        // Open subreport
   835                        Get ComOpenSubreport of hoTemp To vReport
   836                        Get Create U_cCrystalReport To hoSubReport
   837                        //Set Name of hoSubReport to ("oSubReport_"-sName) // for debugging purposes only
   838                        Set pvComObject Of hoSubReport To vReport
   839                        Get IsComObjectCreated of hoSubReport to bAttached
   840                        If (bAttached) Begin
   841                            // Assign subreport the same location as main report
   842                            Set psReportLocation of hoSubReport to (psReportLocation(Self))
   843                            Move hoSubReport to SubReportArray[iCount]
   844                            Set psSubReportName of hoSubReport to (Trim(sName))
   845                            Increment iCount
   846                        End
   847                        Else Begin
   848                            Error DFERR_CRYSTAL_REPORT (SFormat(C_$SubreportCouldNotBeOpened, sName))
   849                            Send Destroy of hoSubReport
   850                        End
   851                    End
   852                Loop
   853            Loop
   854        Loop
   855        Send Destroy of hoAreas
   856        Set phoSubReports to SubReportArray
   857        Set pbSubReportsLoaded to TRUE
   858    End_Procedure // LoadSubreports
   859
   860    //## NOT FOR USE WITH SUBREPORTS ##
   861    //Returns an array of subreports in report
   862    Function SubReportObjects Returns Handle[]
   863        Handle[] SubReportArray
   864        If not (pbSubReportsLoaded(Self)) Begin
   865            // If subreports haven't been loaded, load them now.
   866            Send LoadSubreports
   867        End
   868        Get phoSubReports to SubReportArray
   869        Function_Return SubReportArray
   870    End_Function // SubReportObjects
   871
   872    //## NOT FOR USE WITH SUBREPORTS ##
   873    //Returns handle of subreport object comautomation object. Creates
   874    //connection if connection does not already exist
   875    Function GetSubReportObjectbyName String sReportName Returns Handle
   876        Integer iReportItem iReportCount
   877        String sHoldName
   878        Handle[] SubReportArray
   879
   880        // Check to see if subreport has already been accessed and return it's handle
   881        Get SubReportObjects to SubReportArray
   882        Move (SizeOfArray(SubReportArray)) to iReportCount
   883        For iReportItem From 0 To (iReportCount-1)
   884            If (SubReportArray[iReportItem]) Begin
   885                Move (psSubReportName(SubReportArray[iReportItem])) to sHoldName
   886                If (lowercase(sHoldName)=lowercase(sReportName)) Begin
   887                    Function_Return SubReportArray[iReportItem]
   888                End
   889            End
   890        Loop
   891    End_Function // GetSubReportObjectbyName
   892
   893    // End Subreport Functions
   894
   895
   896    //--------------------
   897    // Begin Output Funtions
   898
   899    //## NOT FOR USE WITH SUBREPORTS ##
   900    // Creates export object if it doesn't exist
   901    Function ExportObject Returns Handle
   902        Handle hoExport hcExportObject
   903        Variant vExport
   904        Boolean bAttached
   905
   906        Get phoExportObject to hoExport
   907        If (Not(hoExport)) Begin
   908            Delegate Get phcExportObject to hcExportObject
   909            Get Create hcExportObject to hoExport
   910            Get ComExportOptions To vExport
   911            Set pvComObject Of hoExport To vExport
   912            //Set Name of hoExport to "oExport" // for debugging purposes only
   913            Get IsComObjectCreated of hoExport to bAttached
   914            If (bAttached) Begin
   915                Set phoExportObject to hoExport
   916            End
   917            Else Begin
   918                Send Destroy of hoExport
   919                Move 0 to hoExport
   920                Set phoExportObject to 0
   921                Error DFERR_CRYSTAL_REPORT C_$ErrorCreatingExportObject
   922            End
   923        End
   924        Function_Return hoExport
   925    End_Function // ExportObject
   926    
   927     
   928    Function PreviewerName Returns String
   929        Handle hoPreviewerParent
   930        String sPreviewerName
   931
   932        Get Parent of Self to hoPreviewerParent
   933        If (hoPreviewerParent) Begin
   934            Get Object_Label of hoPreviewerParent to sPreviewerName
   935            Move (sPreviewerName - "_Previewer") to sPreviewerName
   936        End
   937        Else Begin
   938            Move "oPreviewer" to sPreviewerName
   939        End
   940        
   941        Function_Return sPreviewerName
   942    End_Function
   943
   944
   945    //## NOT FOR USE WITH SUBREPORTS ##
   946    // Returns handle of preview object
   947    Function PreviewObject Returns Handle
   948        Handle hoPreview hoMain hoClientArea hcPreviewObject
   949        String sTitle sPreviewerName
   950        Boolean bAttached
   951
   952        Get phoPreviewObject to hoPreview
   953        // This creates a MDI style viewer. The view that gets created (determined by phcPreviewObject)
   954        // will get placed inside of the client-area-object within the main-panel-id. If either of these
   955        // objects do not exist, this will not work (and you will get an error). In such a case, you'd
   956        // want to create a replacement PreviewObject method using this as a model where you'd place
   957        // the previewer wherever you want.
   958        If (Not(hoPreview)) Begin
   959            Delegate Get phcPreviewObject to hcPreviewObject
   960            Get Main_Panel_ID To hoMain
   961            If (not(hoMain)) Begin
   962                Error DFERR_CRYSTAL_REPORT C_$NoMainPanelObject
   963                Function_return 0
   964            end
   965            Get Client_Id of hoMain to hoClientArea
   966            If (not(hoClientArea)) Begin
   967                Error DFERR_CRYSTAL_REPORT C_$NoClientAreaObject
   968                Function_return 0
   969            End
   970            Get Create Of hoClientArea hcPreviewObject To hoPreview
   971            If (hoPreview) Begin
   972                Get PreviewerName to sPreviewerName
   973                Set Name of hoPreview to sPreviewerName
   974                Set phoPreviewObject to hoPreview
   975                Get ComReportTitle to sTitle
   976                If (sTitle = "") Get psReportName to sTitle
   977                // Note that this requires that the preview object understands the Label Message
   978                Set Label Of hoPreview To sTitle
   979            End
   980        End
   981        Function_Return hoPreview
   982    End_Function // PreviewObject
   983
   984    //## NOT FOR USE WITH SUBREPORTS ##
   985    // Override the default preview object
   986    Procedure AssignPreviewObject Handle hoPreview
   987        If (phoPreviewObject(Self)) Begin
   988            Error DFERR_CRYSTAL_REPORT C_$PreviewObjectAlreadyAssigned
   989        End
   990        Else Set phoPreviewObject to hoPreview
   991    End_Procedure // AssignPreviewObject
   992
   993    //## NOT FOR USE WITH SUBREPORTS ##
   994    // Return handle of activeX preview object.
   995    Function ActiveXReportViewerObject Returns Handle
   996        Handle hoPreview
   997        Get PreviewObject to hoPreview
   998        If (hoPreview) Function_Return (phoActiveXReportViewer(hoPreview))
   999        Function_Return 0
  1000    End_Function // ActiveXReportViewerObject
  1001
  1002    //## NOT FOR USE WITH SUBREPORTS ##
  1003    //Printer specific
  1004    
  1005    Procedure PrintReport
  1006        Integer iStartPage iEndPage iCopies
  1007        Boolean bCollate bPrompt
  1008        Get piPrinterStartPage to iStartPage
  1009        Get piPrinterEndPage   to iEndPage
  1010        Get pbPrinterCollate   to bCollate
  1011        Get piPrinterCopies    to iCopies
  1012        Get pbPrinterPrompt    to bPrompt
  1013        // if endpage is 0, we assume end of report. Startpage must always be specified (default=1).
  1014        If (iEndPage=0) begin
  1015            Send ComPrintout bPrompt iCopies bCollate iStartPage nothing
  1016        end
  1017        Else begin
  1018            Send ComPrintout bPrompt iCopies bCollate iStartPage iEndPage
  1019        end
  1020    End_Procedure //PrintReport
  1021
  1022    //Export specific
  1023    
  1024    Procedure ExportReport
  1025        Boolean bPrompt
  1026        Get pbExportPrompt to bPrompt
  1027        Send ComExport bPrompt
  1028    End_Procedure // ExportReport
  1029
  1030
  1031    // End Output Functions
  1032
  1033End_Class // cCrystalReport
  1034
  1035
  1036// Main crystal report class
  1037
  1038
  1039Class cCrystal is a cObject
  1040
  1041    Procedure Construct_Object
  1042        Forward Send Construct_Object
  1043
  1044        // destination of report
  1045        
  1046        
  1047        Property Integer peOutputDestination Print_To_Window
  1048
  1049        // name of report to be run
  1050        
  1051        Property String  psReportName ""
  1052
  1053        // controls if dat or int files are automatically located.
  1054        
  1055        Property Boolean pbAutoLocateDFFiles True
  1056
  1057        // Can be set true by the developer within the events to stop a report from running.
  1058        
  1059        Property Boolean pbCanceled False
  1060
  1061        // Determines if a report is canceled if an error occurs. This can changed as often as needed to handle any particular
  1062        // section of code. If you want to try to handle an error, set this false (and be careful).
  1063        
  1064        Property Boolean pbCancelIfError True
  1065
  1066        // Handle of classes used. Can be used to change the class of the composite child reportobject class
  1067        
  1068        Property Handle phcReportObject      U_cCrystalReport
  1069        
  1070        Property Handle phcPreviewObject     U_cCrystalPreview
  1071        
  1072        Property Handle phcExportObject      U_cCrystalExportOptions
  1073
  1074        // if set to false, Crystal errors will not be sent to the error object.
  1075        
  1076        
  1077        Property Integer Display_Error_State True
  1078        
  1079        Property Boolean pbHasErrors         False
  1080
  1081        
  1082        Property Handle Old_Error_Object_Id  0
  1083
  1084        
  1085        Property integer Error_Processing_State    False
  1086
  1087        // report com object.
  1088        
  1089        Property Handle  phoReportObject 0
  1090
  1091    End_Procedure  // Construct_Object
  1092
  1093    // returns handle of attached com applicaton object. This creates this as a global object
  1094    // which is assigned to ghoCrystalReportServer. It then does a com attach. If the attach fails,
  1095    // zero is returned indicating failure. This can be used to test if the crystal rdc com objects
  1096    // are installed.
  1097    Function ApplicationObject returns handle
  1098        Boolean bDisplay bAttached
  1099        Handle  hoError
  1100
  1101        If (Not(ghoCrystalReportServer)) Begin
  1102            // if no application object, create this at the destkop level.
  1103            Get Create of (If(ghoApplication, ghoApplication, desktop)) U_cCrystalApplication to ghoCrystalReportServer
  1104        End
  1105        Get IsComObjectCreated of ghoCrystalReportServer To bAttached
  1106
  1107        If not bAttached begin
  1108             Move Error_Object_Id to hoError
  1109             Get Display_Error_State to bDisplay
  1110             Move Self to Error_Object_Id
  1111             Set Display_Error_State to false
  1112             Send CreateComObject of ghoCrystalReportServer
  1113             Move hoError to Error_Object_Id
  1114             Set Display_Error_State to bDisplay
  1115
  1116             Get IsComObjectCreated of ghoCrystalReportServer To bAttached
  1117             If Not bAttached Begin
  1118                Function_Return 0
  1119             End
  1120        End
  1121        Function_return ghoCrystalReportServer
  1122    End_Function // ApplicationObject
  1123
  1124    // Return handle to existing report
  1125    Function ReportObject Returns Handle
  1126        Function_Return (phoReportObject(Self))
  1127    End_Function // ReportObject
  1128
  1129    // Intended to be used by developer
  1130    
  1131    Procedure OnInitializeReport Handle hoReport
  1132    End_Procedure // OnInitializeReport
  1133
  1134    // Intended to be used by developer
  1135    
  1136    Procedure OnPrintReport Handle hoReport
  1137    End_Procedure // OnPrintReport
  1138
  1139    // Intended to be used by developer
  1140    
  1141    Procedure OnExportReport Handle hoReport
  1142    End_Procedure // OnExportReport
  1143
  1144    // Intended to be used by developer
  1145    
  1146    Procedure OnDisplayReport Handle hoReport
  1147    End_Procedure // OnDisplayReport
  1148
  1149    // Can be sent by developer (and possible augmented) in more advanced situations
  1150    Procedure PrintReport Handle hoReport
  1151        boolean bCancel
  1152        Send OnPrintReport hoReport // developer event
  1153        Get pbCanceled to bCancel // developer might have canceled the report
  1154        If (not(bCancel)) Begin
  1155            Send PrintReport of hoReport
  1156        End
  1157    End_Procedure // PrintReport
  1158
  1159    // Can be sent by developer (and possible augmented) in more advanced situations
  1160    Procedure ExportReport Handle hoReport
  1161        boolean bCancel
  1162        Send OnExportReport hoReport // developer event
  1163        Get pbCanceled to bCancel // developer might have canceled the report
  1164        If (not(bCancel)) Begin
  1165            Send ExportReport of hoReport
  1166        End
  1167    End_Procedure // ExportReport
  1168
  1169    // Can be sent by developer (and possible augmented) in more advanced situations
  1170    Procedure DisplayReport Handle hoReport
  1171        boolean bCancel
  1172        Handle hoPreview
  1173        Send OnDisplayReport hoReport // developer event
  1174        Get pbCanceled to bCancel // developer might have canceled the report
  1175        If (not(bCancel)) Begin
  1176            Get PreviewObject of hoReport to hoPreview            
  1177            Send DisplayReport of hoPreview hoReport
  1178        End
  1179    End_Procedure // DisplayReport
  1180
  1181    // Creates report object if it doesn't exist. Opens psReportName.
  1182    Function OpenReport Returns Boolean
  1183        Handle hoReport hoWorkspace hoSubReport hcReportObject hoApplicationObject
  1184        Variant vReport
  1185        Boolean bAttached bExists bOk bCanceled
  1186        String sReport sReportPath DirSep
  1187        Integer iPath iNumPaths iReportItem iReportCount
  1188        tCRWCDO[] CRWCDOs
  1189        Handle[] SubReportArray
  1190
  1191
  1192        Set pbCanceled to False // this can be set true by developers within events to stop the report
  1193
  1194        // Check if Report application Server has already been created started. If not
  1195        // do so. If this returns 0, we failed and the report cannot be run.
  1196        Get ApplicationObject to hoApplicationObject
  1197        If not hoApplicationObject begin
  1198            Error DFERR_CRYSTAL_REPORT C_$FailedToConnectCrystalAutomation
  1199            Function_Return FALSE
  1200        End
  1201
  1202
  1203        // Create report Object if it doesn't exist
  1204        Get ReportObject to hoReport
  1205        If (hoReport) Begin
  1206            // Check to see if report object already has a report opened.
  1207            Get IsComObjectCreated Of hoReport To bAttached
  1208            If (bAttached) Begin
  1209                Error DFERR_CRYSTAL_REPORT C_$ReportIsAlreadyOpened
  1210                Function_Return FALSE
  1211            End
  1212        End
  1213
  1214        If (Not(hoReport)) Begin
  1215            Get phcReportObject to hcReportObject
  1216            Get Create hcReportObject to hoReport
  1217            //Set Name of hoReport to "oReport" // for debugging purposes only
  1218            Set phoReportObject to hoReport
  1219        End
  1220
  1221        If (not(hoReport)) Begin
  1222            Error DFERR_CRYSTAL_REPORT (SFormat(C_$ErrorOpeningReport, sReport))
  1223            Function_Return FALSE
  1224        End
  1225
  1226        // at this point the report is created (but not attached to a COM report).
  1227
  1228        Move True to bOk // keep track of errors during init process
  1229
  1230        Get psReportName to sReport
  1231        Move (Sysconf(SYSCONF_DIR_SEPARATOR)) To DirSep
  1232        // Did the user give us the path and file name?
  1233        If (Pos(DirSep, sReport)) Begin
  1234            File_Exist sReport bExists
  1235        End
  1236        Else Move (FALSE) to bExists
  1237
  1238        If bExists Begin
  1239            Move sReport To sReportPath
  1240        End
  1241        Else If (ghoApplication) Begin
  1242            // If no path was given, locate report in the workspace data directory(s). You can only
  1243            // do this if an application object exists.
  1244            Get phoWorkspace of ghoApplication to hoWorkspace
  1245            Get psDataPath of hoWorkspace To sReportPath
  1246            Get CountOfPaths Of hoWorkspace sReportPath To iNumPaths
  1247            // iterate through each datapath until report is located
  1248            For iPath From 1 To iNumPaths
  1249                Get psDataPath Of hoWorkspace To sReportPath
  1250                Get PathAtIndex Of hoWorkspace sReportPath iPath To sReportPath
  1251                // Make sure path ends with a DirSep before we append the report name.
  1252                If (Right(sReportPath,1)<>DirSep) Begin
  1253                    Append sReportPath DirSep sReport
  1254                End
  1255                File_Exist sReportPath bExists
  1256            Until ((bExists) Or (iPath=iNumPaths))
  1257        End
  1258
  1259        If (Not(bExists)) Begin
  1260            Move False to bOk
  1261            Error DFERR_CRYSTAL_REPORT (SFormat(C_$UnabletoLocateReport, sReport))
  1262        End
  1263
  1264        // If Report was found, open it.
  1265        If (bOk) Begin
  1266            //open report
  1267            Get ComOpenReport of hoApplicationObject sReportPath crOpenReportByDefault to vReport
  1268            Set pvComObject Of hoReport To vReport
  1269            Get IsComObjectCreated Of hoReport To bOk
  1270            If (Not(bOk)) Begin
  1271                Error DFERR_CRYSTAL_REPORT (SFormat(C_$ErrorOpeningReport, sReport))
  1272            End
  1273        End
  1274
  1275        If (bOk) Begin
  1276            //set report location
  1277            While (Right(sReportPath,1)<>DirSep)
  1278                Move (Left(sReportPath, (Length(sReportPath)-1))) to sReportPath
  1279            Loop
  1280            Set psReportLocation of hoReport to sReportPath
  1281            //auto locate dat and int files.
  1282            If (pbAutoLocateDFFiles(Self)) Begin
  1283                Get LocateDFFiles of hoReport to bOk
  1284            End
  1285        End
  1286
  1287        If (bOk) Begin
  1288            // Always default forumula syntax to crystal syntax
  1289            Set ComFormulaSyntax of hoReport to crCrystalSyntaxFormula
  1290
  1291            Send OnInitializeReport hoReport
  1292            Get pbCanceled to bCanceled // developer might have canceled the report
  1293            If bCanceled Move False to bOk
  1294        End
  1295
  1296        If (bOk) Begin
  1297            // Set CDO data source (if any)
  1298            Get pCDOs of hoReport to CRWCDOs
  1299            If (SizeOfArray(CRWCDOs)>0) Begin
  1300                Send AssignCDODataSources of hoReport CRWCDOs
  1301                // Set CDO data source (if any) for subreports
  1302                Get SubReportObjects of hoReport to SubReportArray
  1303                Move (SizeOfArray(SubReportArray)) to iReportCount
  1304                For iReportItem From 0 To (iReportCount-1)
  1305                    Move SubReportArray[iReportItem] to hoSubReport
  1306                    If (hoSubReport) Begin
  1307                        Send AssignCDODataSources of hoSubReport CRWCDOs
  1308                    End
  1309                Loop
  1310            End
  1311        End
  1312
  1313        // we check pbCancaled one last time at the at end of the process. It is possible that OnError is setting this
  1314        // to True and that could happen any time. So we check at the very last moment here.
  1315        If (bOk) Begin
  1316            Get pbCanceled to bCanceled
  1317            If bCanceled Move False to bOk
  1318        End
  1319
  1320        // if any error occured, destroy the report
  1321        If (not(bOk)) begin
  1322            Send DestroyReportObject
  1323        end
  1324        Function_Return bOk
  1325
  1326    End_Function // OpenReport
  1327
  1328    Procedure OutputReport
  1329        Handle hoReport
  1330        Integer iDestination
  1331        Get ReportObject to hoReport
  1332        // We would not have a report object if there was a
  1333        If (hoReport) Begin
  1334            Get peOutputDestination to iDestination
  1335            If (iDestination= PRINT_TO_WINDOW) Begin
  1336                Send DisplayReport hoReport
  1337            End
  1338            Else If (iDestination= PRINT_TO_FILE) Begin
  1339                Send ExportReport hoReport
  1340            End
  1341            Else If ((iDestination=PRINT_TO_PRINTER_NO_DIALOG) or (iDestination=PRINT_TO_PRINTER)) Begin
  1342                Send PrintReport hoReport
  1343            End
  1344            Else Error DFERR_CRYSTAL_REPORT C_$NoPrintDestinationDefined
  1345        End
  1346    End_Procedure // OuputReport
  1347
  1348    
  1349    Procedure DestroyReportObject
  1350        Boolean bCanClose
  1351
  1352        If (phoReportObject(Self)) Begin
  1353            Send Destroy of (phoReportObject(Self))
  1354            Set phoReportObject  to 0
  1355        End
  1356
  1357        // also close the application object. Closing a report and not closing the application object it
  1358        // uses seems to create a memory leak (which is a Crystal RDC thing that we cannot control). See BT 3901
  1359        If (ghoCrystalReportServer) Begin
  1360
  1361            // assertion: I don't think this can ever be a can't close. If so, this will report an
  1362            // assertion but it will keep doing whatever it does.
  1363            If (IsComObjectCreated(ghoCrystalReportServer)) Begin
  1364                Get ComCanClose of ghoCrystalReportServer to bCanClose
  1365                If not bCanClose Begin
  1366                    Error DFERR_CRYSTAL_REPORT "Assert: Cannot close crystal application object"
  1367                End
  1368            End
  1369
  1370            Send ReleaseComObject of ghoCrystalReportServer
  1371        End
  1372    End_Procedure // DestroyReportObject
  1373
  1374    Procedure CloseReport
  1375        Send DestroyReportObject
  1376    End_Procedure // CloseReport
  1377
  1378    Procedure RunReport
  1379        Boolean bOK
  1380        Set pbHasErrors to false
  1381        // direct to local error handler
  1382        Set Old_Error_Object_id to Error_Object_id
  1383        Move self to Error_Object_id
  1384
  1385        Get OpenReport to bOK
  1386        If (bOK) Begin
  1387            Send OutputReport
  1388            Send CloseReport
  1389        End
  1390
  1391        // restore error handler
  1392        Get Old_Error_Object_id to Error_Object_id
  1393    End_Procedure // RunReport
  1394
  1395    // when RunReport is called, errors are directed here.
  1396    
  1397    Procedure Error_Report integer ErrNum integer iErrLine string ErrMsg
  1398        Handle  hoError
  1399        Boolean bCancelOnError
  1400        If (error_processing_state(self)=False) Begin
  1401            Set Error_processing_State to True // prevents recursion
  1402            Set pbHasErrors to True  // an error has occurred in the report
  1403            If (Display_Error_State(self)) Begin // if we display, direct to old error object
  1404                Get Old_Error_Object_Id to hoError
  1405                Send Error_Report of (if(hoError, hoError, desktop)) ErrNum iErrLine ErrMsg
  1406            End
  1407            If (pbCancelIfError(self)) begin
  1408                Set pbCanceled to True
  1409            End
  1410            Send onError ErrNum iErrLine ErrMsg
  1411            Set Error_processing_State to False
  1412        End
  1413    End_procedure // Error_Report
  1414
  1415    // Event called by Error_Report. For augmentation.
  1416    
  1417    Procedure OnError integer ErrNum integer iErrLine string ErrMsg
  1418    End_procedure // OnError
  1419
  1420    Function CrystalString String sVal Returns String
  1421        String sNewVal sHold
  1422        If (Not(Pos('"', sVal))) Function_Return ('"'+sVal+'"')
  1423        If (Not(Pos("'", sVal))) Function_Return ("'"+sVal+"'")
  1424        Move '"' To sNewVal
  1425        While (Pos('"', sVal))
  1426            Move (Left(sVal, (Pos('"', sVal)-1))) To sHold
  1427            Move (Replace((sHold+'"'), sVal, "")) To sVal
  1428            If (sNewVal <> '"') Append sNewVal '+"'
  1429            Append sNewVal sHold '"' "+'" '"' "'"
  1430        Loop
  1431        If (Length(sVal)) Append sNewVal '+"' sVal '"'
  1432        Function_Return sNewVal
  1433    End_Function // CrystalString
  1434
  1435    Function CrystalInteger Integer iVal Returns String
  1436        Function_Return (String(iVal))
  1437    End_Function // CrystalInteger
  1438
  1439    Function CrystalNumber Number nVal Returns String
  1440        Integer iDecimalSeparator
  1441        String sFormattedNumber
  1442        // Crystal requires a string with a decimal point for numeric separator
  1443        Get_Attribute DF_DECIMAL_SEPARATOR To iDecimalSeparator
  1444        Move (string(nVal)) To sFormattedNumber
  1445        Move (Replace (Character (iDecimalSeparator), sFormattedNumber, ".")) To sFormattedNumber
  1446        Function_Return sFormattedNumber
  1447    End_Function // CrystalNumber
  1448
  1449    Function CrystalDate DateTime dtVal Returns String
  1450        Function_Return ("date(" + string(DateGetYear(dtVal)) + "," + string(DateGetMonth(dtVal)) + "," +string(DateGetDay(dtVal)) +")")
  1451    End_Function // CrystalDate
  1452
  1453    // compatibility methods
  1454
  1455    
  1456    Procedure Run_Report
  1457        Send RunReport
  1458    End_Procedure // Run_Report
  1459
  1460    // Get / sets report name
  1461    
  1462    Function Report_Name Returns String
  1463        Function_Return (psReportName(Self))
  1464    End_Function // Report_Name
  1465
  1466    
  1467    
  1468    Procedure Set Report_Name String sReportName
  1469        Set psReportName to sReportName
  1470    End_Procedure // Set Report_Name
  1471
  1472    // Gets / Sets report output destination. Will return output device mode of report view if none specified.
  1473    
  1474    Function Output_Destination Returns Integer
  1475        Function_Return (peOutputDestination(Self))
  1476    End_Function // Output_Destination
  1477
  1478    
  1479    
  1480    Procedure Set Output_Destination Integer DevMode
  1481        Set peOutputDestination To DevMode
  1482    End_Procedure // Set Output_Destination
  1483
  1484End_Class // cCrystal
  1485