Translate

Tuesday, May 4, 2021

Oracle Forms: How to select multiple records in a block using a Record Group

 Oracle Forms: How to select multiple records in a block using a Record Group

I decided to create this demo because the question of how to select multiple records in a multi-record data block is a fairly common topic in the Oracle Technology Network (OTN) Forms discussion forum.  There are other ways of accomplishing this task, but I like to use Oracle Forms Record Groups because I ran into a situation where I needed to use a Record Group (RG) to perform a task in a form so I created a wrapper package to simplify the process of using a RG.  Over the years, the package has grown and I’ve come across situations that were easily solved by using the RG package I had created.

 

Disclaimer:  This demo is built using Oracle Forms 10g R2 (10.1.2.0.2) but it should work with older (as well as newer) versions of Oracle Forms.  This demo also uses the new sample HR schema.  If you don’t have access to this schema, you will have to modify the code to suit your needs.

 

There are two ways you can use this demo form.  First, you can just dig into the sample form provided and figure out what I did or second, you can follow the steps listed to recreate the sample form.

 

What you’ll need to build this demo:

  1. Oracle Forms 10g (Oracle Developer Suite 10g R2)
  2. A valid Oracle database account
    1. The demo was built using the HR schema
  3. The Craig.pll library.  This library contains the Record Group package needed by this demo form.

 

Note:  as I indicated above, you can use Oracle Forms 6i or any version of Oracle Forms newer than Forms 6i.  I have not tested the Record Group package with any version of Oracle Forms older than Forms 6i so you are on your own if you try to make this demo work with a version older than Forms 6i.

 

Let’s get started…

 

  1. Open Oracle Forms Builder and create a new Forms module.
  2. Connect to your database with a user that has SELECT privilege to the HR schema tables.
  3. Create a new data block using the Data Block Wizard and select the HR.EMPLOYEES table and all columns.
    1. Let the Data Block Wizard call the Layout Wizard to display all columns in a Tabular Layout.
  4. Add a new “Non-database” item as the last navigable item in the EMPLOYEES block and set the following properties of the item:
    1. NAME: SELECTED
    2. Item Type:  Check Box
    3. Value when Checked:  1
    4. Value when Unchecked:  0
    5. Check Box Mapping of Other Values:  Not Allowed
    6. Initial Value:  0
    7. Database Item:  No
    8. Width:  14
    9. Prompt:  Selected
    10. Prompt Attachment Edge:  Top
    11. Prompt Alignment:  Center
  5. Open the Property Pallet for the following item: JOB_ID, MANAGER_ID and DEPARTMENT_ID and scroll to the Physical set of properties and set the following properties:
    1. Visible:  No
    2. Canvas:  <NULL>
  6. Select the EMPLOYEES block and set the following properties:
    1. Show Scroll Bar:  Yes
    2. Scroll Bar Canvas:  CANVAS4 (use the name of the canvas created by the Layout Wizard).
  7. Now open the Layout Editor for you primary canvas and arrange the items so they appear similar to the following:

 

 

Note:  Because the Layout Wizard added most everything to your canvas, the FRAME will have the Update Layout property set to Automatically.  The quickest way to rearrange your layout is to size the enclosing Frame to a smaller size and let Forms automatically resize the frame to fit your displayed items and the scrollbar.  At this point, I typically change the Update Layout property to Manually because I don’t always like the way Forms adjusts the layout.  This is a preference issue and not something you need to change.

 

  1. Open the property pallet for the canvas created by the Layout Wizard and set the following properties:
    1. Name: EMPLOYEES
    2. Window: WINDOW1 (should already be set)
    3. Width:  717
    4. Height:  260
  2. Now it’s time to attach the CRAIG.PLL Forms library.  Go to the Object Navigator (ON) and click on the Attached Libraries node then click on the add button () in the ON  toolbar. This will open the Attach Library dialog.
    1. Click the Browse… button and navigate to the directory when you stored the CRAIG.pll and select this library file.
    2. Now click the Attach button.  Forms Builder will display an alert informing you that the library contains a non-portable directory specification and will ask if you want to remove the path.  Select “Yes” to this Alert.
  3. Now we need to create a new window object in the Windows node of the Object Navigator.  Open the property pallet for the new window and set the following properties:
    1. Name:  SELECTED

b.      Title:  Selected Rows

c.       Width:  313

d.      Height:  187

  1. Next, we need to create the Canvas that will be used to display the selected records output.  Click on the Canvases node of the Object Navigator and click the add button.
  2. Open the property pallet for the new canvas and set the following properties:
    1. Name:  SELECTED
    2. Window: SELECTED
    3. Width:  313
    4. Height:  187
  3. Now, let’s create a CONTROL block to store the Selected? button and the SELECTED_ITEMS Text-Item (which will be used to display which items were selected).
    1. Select the Data Blocks node in the Object Navigator (ON) and then click the add button in the ON toolbar.
    2. The New Data Block alert will display; select the Build a new data block manually option and click the OK button.
    3. Make sure the new block is placed after the EMPLOYEES block in the Data Blocks node.
  4. Set the following properties of the control block:
    1. Name:  CONTROL
    2. Database Data Block:  No
  5. Add a new item (Selected Items) to the Control block and set the following properties:
    1. Name:  SELECTED_ITEMS
    2. Item Type:  Text Item
    3. Multi-line: Yes
    4. Maximum Length:  2000
    5. Database Item:  No
    6. Insert Allowed:  No
    7. Update Allowed:  No
    8. Canvas:  SELECTED
    9. X Position:  12
    10. Y Position:  15
    11. Width:  288
    12. Bevel:  Plain
    13. Prompt:  Selected Rows
    14. Prompt Attachment Edge:  Top
  6. Add another new item (Selected Button) to the Control block and set the following properties:
    1. Name:  BTN_SELECTED
    2. Item Type:  Push Button
    3. Label:  Selected?
    4. Number of Items Displayed:  0
    5. Canvas:  EMPLOYEES
    6. X Position:  257
    7. Y Position:  232
    8. Width:  82
    9. Height:  16

 

At this point, the SELECTED canvas should look similar to the following:

 

 

Now it is time to start writing the code to make the form work.  Let’s start at the top and work our way down. 

 

  1. Select the Triggers node in Object Navigator.  This node is at the top of the object navigator and is commonly referred to as Form or Module level triggers.
  2. Click the Add button and add a WHEN-NEW-FORM-INSTANCE trigger with the following code.

 

DECLARE

            bIgnore                        BOOLEAN;

BEGIN

            -- RG - Name the Record Groups (RG) used to track the

            -- rows in the Employees table as well as which records

            -- are selected.

            Form_Vars.RG1_name := 'EMPLOYEES';

            Form_Vars.RG2_name := 'MULTISELECT';

           

            --RG - Now Initialize the RGs

            bIgnore := Rec_Group.Initialize(Form_Vars.RG1_name);

            bIgnore := Rec_Group.Initialize(Form_Vars.RG2_name);

           

            Go_Block('EMPLOYEES');

            Execute_Query;

END;

 

  1. Add a WHEN-WINDOW-CLOSED trigger with the following code:

 

IF ( Get_Window_Property('SELECTED', VISIBLE) = 'TRUE' ) THEN

            Hide_Window('SELECTED');

            Go_Block('EMPLOYEES');

ELSE

            Exit_Form;

END IF;

 

  1. Now we need to create three Alert objects.  These are used by the MSG package in the CRAIG.pll.  Click on the Alerts node and add the following alerts and set their properties accordingly.
    1. Stop

                                                              i.      Name:  STOP

                                                            ii.      Title:  Error

                                                          iii.      Alert Style:  Stop

                                                          iv.      Button 1 Label:  OK

                                                            v.      Button 2 Label:  null

    1. Caution

                                                              i.      Name:  CAUTION

                                                            ii.      Title:  Warning

                                                          iii.      Alert Style:  Caution

                                                          iv.      Button 1 Label:  OK

                                                            v.      Button 2 Label:  null

    1. Note

                                                              i.      Name:  Note

                                                            ii.      Title:  Information

                                                          iii.      Alert Style:  Note

                                                          iv.      Button 1 Label:  OK

                                                            v.      Button 2 Label:  null

  1. Now it is time to add the remaining code needed to make the process work.  We’ll start with the Program Units. 
    1. Open the Program Units node and add a Package Specification named: FORM_VARS with the following code:

PACKAGE Form_Vars IS

    RG1_name  VARCHAR2(30);

    RG2_name  VARCHAR2(30);

END;

Note:  The purpose of the FORM_VARS package specification is simply to create the equivalent of GLOBAL variables.  The benefit of using package specification variables instead of GLOBAL variables is that package specification variables use fewer resources and have greater flexibility in the DATA Types available.  I could have used Items in the CONTROL block as well, but I prefer to use a Package Specification because it is cleaner and less confusing than using a CONTROL block.

 

    1. Add a second Program Unit Procedure named SET_SELETED with the following code:

PROCEDURE set_selected (b_selected BOOLEAN DEFAULT TRUE) IS

   vName          VARCHAR2(9) := 'EMPLOYEES';

   vFirstItem    VARCHAR2(65) := vName||'.'||Get_Block_Property(vName,FIRST_ITEM);

   vCurrItem    VARCHAR2(65);

   vNextNavItem         VARCHAR2(65);

   vLastItem     VARCHAR2(65) := vName||'.'||Get_Block_Property(vName,LAST_ITEM);

   vNavigable   VARCHAR2(5) := 'TRUE';

   vDummy      VARCHAR2(65) := 'TRUE';

   vMsgLvl       VARCHAR2(3) := :SYSTEM.Message_Level;

BEGIN

            Go_Block(vName)

            :SYSTEM.Message_Level := 25;

            vNavigable := Get_Item_Property(vFirstItem,NAVIGABLE);

            :SYSTEM.Message_Level := vMsgLvl;

           

            IF ( vNavigable = 'TRUE' ) THEN 

                        Go_Item(vFirstItem);

            ELSE

                        vDummy := NULL||vName||'.'||Get_Item_Property(vFirstItem,NEXTITEM);

                       

                        IF ( Get_Item_Property(vDummy,NAVIGABLE) = 'TRUE' ) THEN

                                    Go_Item(vDummy);

                        END IF;

            END IF;

           

            vCurrItem := :SYSTEM.CURSOR_ITEM;

           

            <<BLOCK_LIST>>

            WHILE ( vCurrItem != vLastItem ) LOOP

                        vNextNavItem := vName||'.'|| Get_Item_Property(vCurrItem,NEXT_NAVIGATION_ITEM);

                       

                        vDummy := NULL||Get_Item_Property(vCurrItem,VISIBLE);

                       

                        IF ( Get_Item_Property(vCurrItem,VISIBLE) = 'TRUE' ) THEN

                                    IF ( b_selected ) THEN

                                                Set_Item_Instance_Property(vCurrItem,CURRENT_RECORD, VISUAL_ATTRIBUTE, 'SELECTED');

                                    ELSE

                                                Set_Item_Instance_Property(vCurrItem,CURRENT_RECORD, VISUAL_ATTRIBUTE, 'DEFAULT');

                                    END IF;

                        END IF;

                        vCurrItem := vNextNavItem;

                        Go_Item(vCurrItem);

            END LOOP BLOCK_LIST;

END;

 

 

 

  1. Now, we need to add the Item trigger code.  Open the EMPLOYEES data block and add the following Block level triggers and code:
    1. Trigger:  POST-QUERY

 

DECLARE

            bIgnore                        BOOLEAN := FALSE;

BEGIN

            --Add a record to the Record Group.

            bIgnore := rec_group.add_value(p_name=>Form_Vars.RG1_name,

                                                             p_value=>:Employees.Employee_id,

                                                             p_record=>:system.trigger_record);

END;

Note: The Post-Query trigger ensures a record is added to the EMPLOYEES record group; which contains an entry for every record displayed in the multi-record block.

 

    1. Trigger:  WHEN-REMOVE-RECORD

 

DECLARE

            ignore              NUMBER;     

BEGIN

            -- Remove the Current Record from the Record Group (No DML occurs).

            ignore := rec_group.delete_value(Form_Vars.RG1_name,:Employees.Employee_id);

END;

Note:  The When-Remove-Record trigger ensures when a row is cleared or deleted from the multi-record block the corresponding record is removed from the EMPLOYEES record group.

 

  1. Now, we need to add the Item level triggers for the EMPLOYEES data block.  Open the EMPLOYEES block and add the following Item Triggers:
    1. ROW_NUM – WHEN-NEW-ITEM-INSTANCE

/* This instruction is needed just to prevent the user from navigating to the ROW_NUM column.

    I could have set the NAVIGATION property of the item, but this presented problems in the

    SET_SELECTED program unit and as this is just a demo form, I didn’t want to spend too

    much time on the development. :) */

 

Next_Item;

 

    1. SELECTED – WHEN-CHECKBOX-CHANGED

 

DECLARE

            bIgnore                        BOOLEAN;

            nIgnore                        NUMBER;

            nRec                            NUMBER;

BEGIN

            IF ( CHECKBOX_CHECKED('EMPLOYEES.SELECTED') ) THEN

                        --Checked.

                        bIgnore := Rec_Group.Add_Value(p_name => Form_Vars.RG2_name

                                                                        ,p_value => :EMPLOYEES.EMPLOYEE_ID

                                                                        ,p_inline => FALSE

                                                                        ,p_record => :system.trigger_record);

                        Set_Selected(TRUE);

            ELSE

                        --Unchecked.

                        --Check if the value exists first.

                        nRec := Rec_Group.Delete_Value(p_name => Form_Vars.RG2_name

                                                                          ,p_value => :EMPLOYEES.EMPLOYEE_ID);

                        Set_Selected(FALSE);

            END IF;

END;

Note:  This trigger takes care of adding or removing the selected/unselected records from the MULTISELECT record group.

 

  1. The final Item level trigger is the CONTROL block, BTN_SELECTED When-Button-Pressed trigger.  Open the CONTROL block and add the WHEN-BUTTON-PRESSED trigger and following code to the BTN_SELECTED item.

 

DECLARE

            nCount1          NUMBER := 0;

            nCount2          NUMBER := 0;

            nRec                NUMBER := 0;

            vDummy         VARCHAR2(10);

           

BEGIN

            --Get total number of selected records.

            nCount1 := Rec_Group.Get_Count(Form_Vars.RG2_name);

            nCount2 := Rec_Group.Get_Count(Form_Vars.RG1_name);

           

            IF ( nCount1 > 0 ) THEN

                        Go_Item('CONTROL.SELECTED_ITEMS');

                        Show_Window('SELECTED',136,45);

                        IF ( :CONTROL.SELECTED_ITEMS IS NOT NULL ) THEN

                                    :CONTROL.SELECTED_ITEMS := NULL;

                        END IF;

 

                        --Get the Rec Num of each selected record. 

                        <<SELECTED>>      

                        FOR i IN 1..nCount1 LOOP

                                    vDummy := Rec_Group.Get_Value(Form_Vars.RG2_name,i);

                                    nRec := Rec_Group.Get_Number(Form_Vars.RG1_name, Rec_Group.Get_Value(Form_Vars.RG2_name,i) );

 

                                    --Display value of each selected record.

                                    --Msg.Show('N','Rec#: '||nRec||' and Emp ID: '||vDummy||' selected.');                                

                                    IF ( :CONTROL.SELECTED_ITEMS IS NULL ) THEN

                                                :CONTROL.SELECTED_ITEMS := 'You selected the following records:'||CHR(10)||'Rec #: '||nRec||' and Emp ID: '||vDummy;

                                    ELSE

                                                :CONTROL.Selected_Items := :CONTROL.Selected_Items||CHR(10)||'Rec #: '||nRec||' and Emp ID: '||vDummy;

                                    END IF;

                                   

                        END LOOP SELECTED;

            ELSE

                        GO_Block('EMPLOYEES');             

                        Msg.Show('S','You haven''t selected any records!');

                        RAISE Form_Trigger_Failure;                      

            END IF;

END;

Note:  This is where the main work is performed and we loop through the selected record and match them with the record in the multi-record block.

 

  1. Lastly, we need to create the Visual Attributes that are used by the SET_SELECTED program unit.  Open the Visual Attributes node and add the following attribute with the listed properties.
    1. DEFAULT

                                                              i.      All properties are left at their default values

                                                            ii.      In other words, we don’t set any of the values as we want the default values.

    1. SELECTED

                                                              i.      Background Color:  r75g100b75

                                                            ii.      Font Weight:  Bold

 

That’s it!  Just compile and run the form.  The run product should look similar to the following screen shots.

 

 


 

 


Wednesday, April 28, 2021

ORA-06503: PL/SQL Function returned without value in Package INV_Validate_Trolin Procedure Attributes.

ORA-06503: PL/SQL Function returned without value in Package

INV_Validate_Trolin Procedure Attributes

When trying to create a new batch, following error is occurring:

ORA-06503: PL/SQL Function returned without value in Package INV_Validate_Trolin Procedure Attributes 


Cause:

The issue is occurring when, an item is used in the formula with an UOM that is neither primary or secondary item UOM and there is no

conversion defined between the formula UOM and primary or secondary UOM.

Solution:

Check whether wrong UOM is entered mistakenly in Formula. If UOM’s were entered as required then define correct UOM conversions for

items used in formula.


Thursday, February 11, 2021

HZ Tables in Oracle Apps r12

HZ_PARTIES:
The HZ_PARTIES table stores basic information about parties that can be shared with any relationship that the party might establish with another party. The primary key for this table is PARTY_ID.
Few Important Columns are

  • PARTY_ID: Party identifier
  • PARTY_NUMBER: Unique identification number for this party
  • PARTY_NAME: Name of the party
  • PARTY_TYPE: The party type can only be Person, Organization, Group or Relationship.

HZ_PARTY_SITES:
The HZ_PARTY_SITES table links a party (HZ_PARTIES) and a location (HZ_LOCATIONS) and stores location-specific party information. One party can optionally have one or more party sites. One location can optionally be used by one or more parties. The primary key for this table is PARTY_SITE_ID.
Few Important Columns are

  • PARTY_SITE_ID: Party site identifier.
  • PARTY_ID: Identifier for the party. Foreign key to the HZ_PARTIES table.
  • LOCATION_ID: Identifier for the party site. Foreign key to the HZ_LOCATIONS table.
  • PARTY_SITE_NUMBER: Party site number.
  • PARTY_SITE_NAME: User-defined name for the site.
  • ADDRESSEE: Addressee information.

HZ_LOCATIONS:
The HZ_LOCATIONS table stores information about a delivery or postal address such as building number, street address, postal code, and directions to a location. This table provides physical location information about parties (organizations and people) and customer accounts. The primary key for this table is LOCATION_ID.
Few Important Columns are

  • LOCATION_ID: Unique identifier for this location
  • COUNTRY: Country code from the TERRITORY_CODE column in the FND_TERRITORY table
  • ADDRESS1: First line for address
  • ADDRESS2: Second line for address
  • ADDRESS3: Third line for address
  • ADDRESS4: Fourth line for address
  • CITY: City
  • POSTAL_CODE: Postal Code
  • STATE: State
  • ADDRESS_KEY: Derived key that facilitates fuzzy searches

HZ_CUST_ACCOUNTS:
The HZ_CUST_ACCOUNTS table stores information about customer accounts , or business relationships that the deploying company establishes with a party of type Organization or Person. This table focuses on business relationships and how transactions are conducted in the relationship. Since a party can have multiple customer accounts, this table might contain several records for a single party. For example, an individual person can establish a personal account, family account, and a professional account for a consulting practice. The primary key for this table is CUST_ACCOUNT_ID.
Few Important Columns are

  • CUST_ACCOUNT_ID: Customer account identifier
  • PARTY_ID: A foreign key to the HZ_PARTY table.
  • ACCOUNT_NUMBER: Account Number
  • CUSTOMER_TYPE: Receivables lookup code for the CUSTOMER_TYPE attribute. I for internal customers, R for revenue generating external customers.
  • CUSTOMER_CLASS_CODE: Customer class identifier

HZ_CUST_ACCT_SITES_ALL:
The HZ_CUST_ACCT_SITES_ALL table stores all customer account sites across all operating units. Customer account sites are addresses, for customer accounts, where the deploying company does business with its customers. One customer account can have multiple customer account sites, and customer account sites for one customer account can belong to multiple operating units. The primary key for this table is CUST_ACCT_SITE_ID.
Few Important Columns are

  • CUST_ACCT_SITE_ID: Customer site identifier
  • CUST_ACCOUNT_ID: Identifier for a customer account. Foreign key to the HZ_CUST_ACCOUNTS table
  • PARTY_SITE_ID: Identifier for a party site. Foreign key to the HZ_PARTY_SITES table
  • BILL_TO_FLAG: Indicates if this is a Bill-To site.
  • SHIP_TO_FLAG: Indicates if this is a Ship-To site.
  • MARKET_FLAG: Indicates if this is a Marketing site.

HZ_CUST_SITE_USES_ALL:
The HZ_CUST_SITE_USES_ALL table stores business purposes assigned to customer account sites, for example Bill-To, Ship-To, and Statements. Each customer account site can have one or more purposes. This table is a child of the HZ_CUST_ACCT_SITES_ALL table, with the foreign
key CUST_ACCT_SITE_ID. The HZ_CUST_SITE_USES_ALL table also stores operating unit identifier, though the HZ_CUST_ACCT_SITES_ALL table itself stores the operating unit for customer account sites. The primary key for this table is SITE_USE_ID.
Few Important Columns are

  • SITE_USE_ID: Site use identifier
  • CUST_ACCT_SITE_ID: Identifier for the customer account site. Foreign key to the HZ_CUST_ACCT_SITES_ALL table
  • SITE_USE_CODE: Business purpose assigned to customer site account, such as Bill-To, Market, and Statements.
  • PRIMARY_FLAG: Indicates if this site is the primary site for this customer account. Y for the primary customer account site. N for other customer account sites.

HZ_CUSTOMER_PROFILES:
The HZ_CUSTOMER_PROFILES table stores information about the credit characteristics of a single customer account or a customer account site or a party. A profile class defined in the
HZ_CUSTOMER_PROFILE_CLASSES table can be used to provide default values for the attributes in this table. The primary key for this table is CUST_ACCOUNT_PROFILE_ID.
Few Important Columns are

  • CUST_ACCOUNT_PROFILE_ID: Unique identifier of this customer profile
  • CUST_ACCOUNT_ID: Identifier for the Customer Account. Foreign key to the HZ_CUST_ACCOUNTS table.
  • STATUS: Indicates whether the profile is active or inactive

HZ_CUST_PROFILE_CLASSES:
The HZ_CUST_PROFILE_CLASSES table stores information about the credit characteristics that are common across a group of customer accounts. The characteristics specified in this table can be used as default characteristics for similar customer accounts. The primary key for this table is PROFILE_CLASS_ID.
HZ_PARTY_RELATIONSHIPS:
The HZ_PARTY_RELATIONSHIPS table stores information about relationships between parties.

Relationship between the tables

Accounts Receivable-AR setups in Oracle EBS Functional Financials - Apps R12

   1. Oracle Receivables -> Setup -> System ->System Options     a. Receivable GL related setup    b. Transaction and Customer rela...