ILE RPG Goes Free and Mobile!

RPG
Typography
  • Smaller Small Medium Big Bigger
  • Default Helvetica Segoe Georgia Times

A tip of the programmer's beanie to IBM for providing a powerful, expressive RPG syntax.

 

Back in my early days of the IBM midrange platforms, RPG was known as primarily being a Report Program Generator (although the "Generator" part was a little over-optimistic!). This was back when writing interactive programs with RPG was a black art of shoehorning the RPG cycle into something into which it clearly did not fit.

 

But with enough effort, some slamming your head into the wall, and a little time with a Shelly Cashman textbook or two, it was possible back in the day to write serviceable interactive RPG programs. That code was never going to win a Most Beautiful Code Contest, but it did the job. After spreading my wings and learning other languages, I realized the value of indented, expressive, and obvious codecode written as much for eyeballs as for a compiler. Alas, even with the advent of the mid-90s era of ILE RPG, RPG persisted as a compile-only language very unfriendly to both eyeballs and quick comprehension. I'll go out on a huge limb here and flatly state that our old RPGI don't care how good a programmer you werewas simply impervious to being expressive and obvious.

 

I am late to the party, but the other day I started noodling around with the 7.1 Technical Refresh of ILE RPG, the one that provides full "free-form" RPG capabilities. In very short order, I was outrageously impressed. Nits remain to be picked with several syntactical aspects of the language, but for the first time ever, I am writing RPG that is expressive and comprehensible. If you write RPG for anything serious, you owe it to yourself to dig into the 7.1 Technical Refresh of RPG. It will forever change how you write RPG. While the refresh doesn't add any great new functional changes, RPG programming power previously straight-jacketed by column reliance quickly comes shining through.

 

This article features an example RPG program I wrote with the TR 7.1 RPG free-format syntax. I'm a rusty RPG coder to be sure, so if you see anything glaringly stupid (for which I have a knack!), please let me know. I cut a few corners to keep the code short, and at 191 lines I'm probably pressing my luck. But it's 2015, and articles like this are no longer constrained by the printed page, so bear with me. I won't cover every line of the RPG in detail in this article, but all of the project's source, including the RPG, is available here for easy download or further inspection.

 

Making It Mobile

I thought ILE RPG's spanking new syntax was worthy of a spanking new user interface, so the program I wrote is an ASNA Mobile RPG (MR) app. MR apps are HTML5 apps primarily intended for smartphones and tablets, but they also run on desktop browsers. They feature performant and secure IBM i database connectivity. The app presented here is a simple little customer CRUD app with, for a little more sizzle, a map of the customer's address.

 

The intent of this article is twofold:

  1. To show TR 7.1 ILE RPG in action. A full RPG program shows more context and capabilities than just a few snippets of RPG here and there.
  2. To show an RPG model that provides a (use your imagination here) work-with panel-like UI for a mobile app. Mobile idioms are vastly different than what we used to use in the green-screen, but as you'll see, MR abstracts away mobile uniqueness and empowers RPG coders to create IBM i mobile apps with nothing but RPG.

 

The ASNA Mobile RPG Mobile Display File

Before digging into the ILE RPG for this mobile app, let's take a quick tour of MR's mobile display file. This isn't really an article about ASNA Mobile RPG (MR) as much as it is about ILE RPG (I wrote an article awhile back that goes into more Mobile RPG detail that can be read here). While there are a few things to understand about Mobile RPG, thanks to IBM's Open Access API, there's less than you think. Mobile RPG provides a Windows-based mobile UI designer. It lets you create mobile display files with record formats, just like old-school display files. MR enforces the traditional indicator-driven "contract" between an RPG program and its display file.

 

The Three Steps to Creating and Running an ASNA Mobile RPG App

Figure 1 below summarizes the three steps for creating and running an MR mobile app.

 

060315ASNAFig1

Figure 1: Creating and running an MR mobile app requires three steps.

 

A little detail for each step follows:

 

Step 1: Create the mobile UI. MR provides a Windows-based designer for creating the mobile UI. It includes all of the user interface elements you'd expect for a mobile user interface (including text boxes, buttons, navigational bars, map, data charts, data list, signature capture, images, and many others). Once you've created the mobile display file, you export it through MR as a traditional display file object on the IBM i. This display file will never be seen by eyeballs; rather, it exists purely to compile an associated RPG program. During this export step, you can optionally save the exported display file source, which is mostly useful for learning purposes to see how all of MR's user interface elements map to RPG idioms.

 

Step 2: Write an RPG program. This program provides the logic and file IO for your mobile app. It is compiled against the display file object created by the export process in Step 1.

 

Step 3: Run the mobile app. The MR mobile app is an HTML5 browser-based mobile app that runs in mobile (and desktop) browsers. At runtime, IBM's Open Access API intercepts this RPG program's display file data and routes it to and from the MR mobile user interface. That the traditional display file is swapped out for the mobile display file at runtime is completely transparent to the RPG program.

 

Let's take a closer look at the example app's three display panels (which are actually surfaced as display file record formats).

 

The Mobile List Panel

The initial screen for this example app is the list panel shown in Figure 2A.

 

060315ASNAFig2A

Figure 2A: The mobile list display looks like this.

 

The initial screen for the list panel's format name is CUSTLIST. It is scrollable, and a user can select a row one of two ways: the user can either tap on the customer's name or tap on the chevron on the right end of the row. This list effectively presents the work-with panel model for mobile devices.

 

Because this is an example of an IBM i app, it assumes it might be using an input file with a great deal of records, thus the "Next" button to get the next page of rows. For this example, six rows are shown and the display isn't scrollable. However, the number of rows displayed is controlled by a constant in the underlying RPG program. Unlike a tethered green-screen, apps like this need to be written with performance in mind. The user doesn't want to wait for hundreds of rows to load (but it might also be reasonable to load more than I am loading here).

 

The Mobile RPG designer is used to map function keys to button (and other UI element) taps. In this panel, the F3 key is mapped to the "End" button and the F5 key is mapped to the "Next" button. The record format name for this list panel is CUSTINFO.

 

This customer list is populated in the ILE RPG as a very simple subfile. This subfile has the following configuration assignments:

 

Description                          Value

Subfile name                        CSTSBF

Subfile controller name        CSTCTRL

Clear subfile indicator          99

Main text field name             CSTXT

Main text field length            70

Secondary text field name   CSTDTL

Secondary text field length  50

Selection field name            CSTSEL

Value field                           CSTVAL

Value field length                30

 

The CSTSEL field is a single-character field that is populated implicitly by MR with a '1' when a user taps the row. RPG's READC operation is then used to identify the row selected. The CSTVAL field is a "hidden" field used to stash hidden data for a row. In this case, the customer number is stored in this field (so that when a row is selected, via READC, the customer number is available).

 

The Update Panel

The mobile update panel is shown in Figure 2B.

 

060315ASNAFig2B

Figure 2B: The mobile update panel shows a customer's information.

 

The update panel, which is format CUSTINFO, is shown by tapping on a customer name from the list panel. The user can change any of the fields presented and tap OK or tap Cancel. Tapping either button returns the user to the list panel.

 

In this panel, the F2 key is mapped to the "Back" button and to the "Cancel" button, and the F8 key is mapped to the "OK" button.

 

The Map Panel

The map panel, shown below, provides geo-location functionality.

 

060315ASNAFig2C

Figure 2C: The map panel shows a customer's location.

 

The map panel, which is format CUSTMAP, is displayed for a customer when the chevron is tapped on the list panel. With an ILE RPG program providing this mobile app's logic, Mobile RPG needed an idiomatic RPG way of providing the addresses for a map. In this case, a single address is mapped, but the Mobile RPG map control can map many addresses.

 

The map control is fed addresses through a simple RPG subfile. This subfile has the following configuration assignments:

 

Description                          Value

Subfile name                       MAPSBF

Subfile controller name        MAPCTRL

Clear subfile indicator         99

Address field name             Location

Main text field length           60

 

Clicking back returns the user to the list.

 

After exporting the display file these three record formats comprise, the residual DDS display file source member is shown (Figure 3 below). Take a look at it and notice that its fields and subfiles resolve to those explained above.

 

0001 A                                    DSPSIZ(27 132 *DS4)

0002 A                                     INDARA

0003   *--------------------------------------------------------

0004 A         R CUSTLIST

0005 A                                     OVERLAY

0006 A            MOREROWS     10A O 1 2

0007   *--------------------------------------------------------

0008 A         R CUSTINFO

0009 A                                     OVERLAY

0010 A           CUSTKEY       50A O 1 2

0011 A           CMNAME        40A B 2 1

0012 A           CMADDR1       35A B 2 43

0013 A           CMCITY       30A B 3 1

0014 A           CMSTATE       2A B 3 33

0015 A           CMPOSTCODE   10A B 3 37

0016   *--------------------------------------------------------

0017 A         R CUSTMAP

0018 A                                     OVERLAY

0019 A           CMNAME       40A O 1 2

0020 A           CSZ           60A O 2 1

0021   *--------------------------------------------------------

0022 A         R CSTSBF                   SFL

0023 A           CSTSEL         1A H

0024 A           CSTTXT       70A O 1 5

0025 A           CSTVAL       30A H

0026 A           CSTDTL       50A O 3 1

0027   *--------------------------------------------------------

0028 A         R CSTCTRL                   SFLCTL(CSTSBF)

0029 A                                     SFLSIZ(2)

0030 A                                    SFLPAG(1)

0031 A N99                                 SFLDSP

0032 A N99                                 SFLDSPCTL

0033 A 99                                 SFLCLR

0034 A                                     OVERLAY

0035   *--------------------------------------------------------

0036 A         R MAPSBF                   SFL

0037 A           LOCATION     60A O 1 2

0038   *--------------------------------------------------------

0039 A         R MAPCTRL                   SFLCTL(MAPSBF)

0040 A                                     SFLSIZ(2)

0041 A                                     SFLPAG(1)

0042 A N99                                 SFLDSP

0043 A N99                                 SFLDSPCTL

0044 A 99                                 SFLCLR

0045 A                                     OVERLAY

 

Figure 3: This DDS display file source is generated when MR mobile UI is exported as a display file object.

 

Think of MR's exported display file object as a proxy for the mobile UI. At compile time, the RPG program will reference this proxy display file, but at runtime, through Open Access, the MR mobile display file is used. Once again, the proxy IBM i display file is never seen by human eyeballs.

 

With the MR mobile display file and its IBM i proxy display file created, let's turn our attention to some of the interesting parts of the ILE RPG source code. Because MR so well abstracts away the notion that we're really creating a mobile app, the RPG is written (and can be examined) without regard for it really being a mobile app. Just think of it as powering three simple, traditional display file formats (which, by the way, is exactly what it's doing).

 

The ILE RPG Source

The full RPG source is shown below in Figure 4A, and the single /COPY member it includes is shown in Figure 4B. A brief code narrative follows these two listings.

 

0001 Ctl-Opt Option(*srcstmt) Dftactgrp(*No) ActGrp('rpmobile');

0002

0003 Dcl-F smplst WORKSTN Infds(infds)

0004                       Handler('MOBILERPG')

0005                       SFile(CSTSBF:CSTRRN)

0006                       SFile(MAPSBF:CSTRRN);

0007

0008 Dcl-F CustomerL2 Disk(*ext) Usage(*Input) Keyed

0009                   Rename(RCMMASTER:RCUSTL2);

0010

0011 Dcl-F CustomerL1 Disk(*ext) Usage(*Update) Keyed

0012                   Rename(RCMMASTER:RCUSTL1);

0013

0014 Dcl-DS CustL2Key LikeRec(RCUSTL2:*Key);

0015 Dcl-DS TopRow   LikeRec(RCUSTL2:*Key);

0016

0017 Dcl-S CstRRN   Zoned(4:0);

0018 Dcl-C MAXROWS Const(6);

0019

0020 /copy RPMRDATA/QRPGLESRC,KEYMAPF

0021

0022 Dcl-DS Action Qualified;

0023     Exit           Char(1) Inz(F03);

0024     Cancel         Char(1) Inz(F02);

0025     Back           Char(1) Inz(F02);

0026     ItemTapped     Char(1) Inz(F01);

0027     ChevronTapped Char(1) Inz(F04);

0028     More           Char(1) Inz(F05);

0029     Backward       Char(1) Inz(F06);

0030     Find           Char(1) Inz(F07);

0031     OK             Char(1) Inz(F08);

0032     Display       Char(1) Inz(F09);

0033     Remove         Char(1) Inz(F10);

0034 End-DS;

0035

0036 SetLL *LoVal CustomerL2;

0037 LoadCstLst();

0038 ExFmt CUSTLIST;

0039

0040 Dow ActionRequest <> Action.Exit;

0041     Select;

0042         When CurrentFormat = 'CUSTLIST';

0043               Select;

0044                   When ActionRequest = Action.ItemTapped;

0045                       ReadC CSTSBF;

0046                       If ReadCustById(%INT(CSTVAL));

0047                         // Customer not found.

0048                       EndIf;

0049                       ExFmt CUSTINFO;

0050

0051                   When ActionRequest = Action.ChevronTapped;

0052                       ReadC CSTSBF;

0053                       If ReadCustById(%INT(CSTVAL));

0054                         // Customer not found.

0055                       EndIf;

0056                       CSZ = %TRIM(CMADDR1) + ', ' +

0057                             %TRIM(CMCITY) + ', ' +

0058                             CMState;

0059                       ShowMap(CSZ);

0060                       ExFmt CUSTMAP;

0061

0062                   When ActionRequest = Action.More;

0063                       LoadCstLst();

0064                       ExFmt CUSTLIST;

0065

0066                  Other;

0067                       ExFmt CUSTLIST;

0068               EndSl;

0069

0070         When CurrentFormat = 'CUSTINFO';

0071             Select;

0072                 When ActionRequest = Action.OK;

0073                     UpdateCust();

0074                     RefreshCustList(CMName:

0075                                     CMCustNo);

0076                     ExFmt CUSTLIST;

0077

0078                 When ActionRequest = Action.Cancel;

0079                     RefreshCustList(TopRow.CMName:

0080                                     TopRow.CMCustNo);

0081                     ExFmt CUSTLIST;

0082

0083                 When ActionRequest = Action.Back;

0084                     SetLL *LoVal CustomerL2;

0085                     LoadCstLst();

0086                     ExFmt CUSTLIST;

0087

0088                 Other;

0089                     SetLL *LoVal CustomerL2;

0090                     LoadCstLst();

0091                     ExFmt CUSTLIST;

0092             EndSl;

0093

0094         When CurrentFormat = 'CUSTMAP';

0095             Select;

0096                 When ActionRequest = Action.OK OR

0097                       ActionRequest = Action.Back;

0098                     RefreshCustList(CMName:

0099                                     CMCustNo);

0100                     ExFmt CUSTLIST;

0101             EndSl;

0102

0103     EndSl;

0104 EndDo;

0105

0106 *InLR = *On;

0107 Return;

0108

0109 Dcl-Proc UpdateCust;

0110     Update RCustL1;

0111 End-Proc;

0112

0113 Dcl-Proc LoadCstLst;

0114     Dcl-S RowCount Packed(12:0);

0115

0116     *IN99 = *On;

0117     Write CSTCTRL;

0118     CstRRN = 0;

0119     RowCount = 0;

0120

0121     DoW (RowCount < MAXROWS);

0122         Read CustomerL2;

0123         If NOT %EOF;

0124             If RowCount = 0;

0125                 TopRow.CMName = CMName;

0126                 TopRow.CMCustNo = CMCustNo;

0127             EndIf;

0128             RowCount = RowCount + 1;

0129             CstRRN = CstRRN + 1;

0130             CSTSEL = '0';

0131             CSTTXT = CMName;

0132             CSTDTL = %Trim(CMCITY) + ', ' + CMSTATE;

0133             CSTVAL = %CHAR(CMCustNO);

0134             Write CSTSBF;

0135         Else;

0136            Leave;

0137         EndIf;

0138     EndDo;

0139

0140     *In99 = *Off;

0141     Write CSTCTRL;

0142     PeekForEOF();

0143 End-Proc;

0144

0145 Dcl-Proc PeekForEOF;

0146     Read CustomerL2;

0147     If NOT %EOF;

0148         Eval MOREROWS = 'More...';

0149         ReadP CustomerL2;

0150     Else;

0151         Eval MOREROWS = 'No More';

0152         SETLL *LOVAL CustomerL2;

0153     EndIf;

0154 End-Proc;

0155

0156 Dcl-Proc ReadCustById;

0157     Dcl-Pi *N Ind;

0158         CustNo Packed(9:0) CONST;

0159     End-Pi;

0160

0161     Chain CustNo CustomerL1;

0162     Return %EOF();

0163 End-Proc;

0164

0165 Dcl-Proc RefreshCustList;

0166     Dcl-Pi *N;

0167         Name Like(CMName);

0168         CustNo Like(CMCustNo);

0169    End-Pi;

0170

0171     CustL2Key.CMName = Name;

0172     CustL2Key.CMCustNo = CustNo;

0173     SetLL %KDS(CustL2Key) CustomerL2;

0174     LoadCstLst();

0175 End-Proc;

0176

0177 Dcl-Proc ShowMap;

0178     Dcl-Pi *N;

0179         Address Char(60);

0180     End-Pi;

0181

0182     *IN99 = *On;

0183     Write MAPCTRL;

0184

0185     CstRRN = 1;

0186     Location = Address;

0187     Write MAPSBF;

0188

0189     *In99 = *Off;

0190     Write MAPCTRL;

0191 End-Proc;

 

Figure 4A: Here's the full ILE RPG source listing.

 

0001 Dcl-Ds Infds;

0002     ActionRequest Char(1) Pos(369);

0003     CurrentFormat Char(10) Pos(261);

0004 End-Ds;

0005

0006 Dcl-C F01 const(x'31');

0007 Dcl-C F02 const(x'32');

0008 Dcl-C F03 const(x'33');

...

0028 Dcl-C F23 const(x'bb');

0029 Dcl-C F24 const(x'bc');

0030

0031 // Other attention keys

0032 Dcl-C Clear_Key       const(x'bd');

0033 Dcl-C Enter           const(x'f1');

0034 Dcl-C Help           const(x'f3');

0035 Dcl-C PageUp         const(x'f4');

0036 Dcl-C PageDown       const(x'f5');

0037 Dcl-C Print_Key       const(x'f6');

0038 Dcl-C Auto_Enter     const(x'50');

 

Figure 4B: This is the external source member to define the INFDS data structure and its constants.

 

Narrative for several chunks of the RPG follows:

 

Line                        Description

0001                       Define compile time options for the program.

 

0002 – 0012            Declare the workstation file and two disk files (CustomerL2 is keyed by customer name and number, and CustomerL1 is keyed by customer number only). The workstation file continuation on line 4 registers this RPG program with the Open Access API. This is the RPG program's only nod to it not using a traditional display file.

 

0014 – 0015            A very cool RPG feature, these lines provide fully qualified key definition data structures for the two disk files. These one-line shortcuts serve as KLIST alternatives (albeit vastly more programmer-friendly).

 

0017 – 0018            These are two global variables for the program (using local variables in procedures substantially reduces global variable dependence).

 

0020                       Include the INFDS copy member (from Figure 4B) and its constants. This enables the very old-school trick of interrogating the hex value of INFDS's position 369 to determine what function (or special) key was pressed. Positions 260-270 also define and interrogate the last-written display file format as "CurrentFormat". This field is instrumental in guiding this program's logic. (As an aside, I was amazed to realize that, with Mobile RPG and IBM's Open Access API, this INFDS data structure feature works just fine! The INFDS data flows naturally out of, and back into, the RPG program through Open Access.)

 

0022 – 0034            Declare a fully qualified data structure named "Action" that maps the function key constants (declared in Figure 4B's copy member) to semantic actions. This lets the programmer think not about function key presses, but about user actions selected (such as a list's chevron having been tapped). My longer-term plan for this data structure is to map reusable and generic "actions" to function keys and then move that definition to the Figure 4B copy member. Note that a couple of these actions are defined multiple times. This is for those times when a "Back" action is different from a "Cancel" action (for example) semantically but both are implemented with an F2 key press. I was surprised to find what appeared to be keyword-type conflicts with some field names in RPG data structures. For example, the ILE RPG does not at all tolerate an action being named "Next," to my great disappointment. This is why there is a "More" action; I'd sure rather have a "Next" action.

 

0036 – 0038            Here's the mainline code to bootstrap the program. These three lines populate the initial customer list. There aren't any subroutines in this program; those have been dispatched and replaced with procedures. Thanks to TR 7.1's free-format procedure declaration, you no longer need to bleed to death to remember how to declare a procedure and its interface.

 

0040 -0104             This is the main loop of this program. This loop is way too long to be considered good code in anyone's book. I sacrificed code quality here for slightly shorter source listing and less eyeball hopping required to comprehend the code.

 

0042 – 0049            This is the point in the narrative, were I explaining this to my father, where he'd throw up his hands and say, "Enough about the labor pains! Show me the baby!" This is where the fully qualified data structures, mapping function keys to semantic actions, and RPG automatically tracking the most recent record format really pay off. While it's really a stretch, this RPG code almost presents itself as three controllers for three routes (the three record formats), each having at least one action. You can easily see that, for the CUSTLIST format, the program allows three actions: ItemTapped, ChevronTapped, and More. These three actions (and the Exit action) are the actions available on the CUSTLIST panel.

 

0046 – 0046            You didn't get to write RPG like line 46 when Saturday Night Fever was number 1 at the box office! This line calls a procedure that accepts a single integer argument and returns a Boolean value that indicates if a record was read. I've left the error-handling as an exercise to the reader (That's usually what writers say when they don't know how to do something! In this case, I promise, it's to minimize the code sample).

 

0074 – 0075            When the CUSTINFO panel is displayed (Figure 4B) and the OK button is tapped, the program needs to return to the customer list (Figure 4A) but reposition it to the customer just updated. RefreshCustList is a procedure that does just that. It is passed the customer name and number for which to position the list.

 

0113 – 0143            The LoadCstLst procedure loads up to the constant MAXROWS value of rows into the subfile displayed in Figure 4A. Note the variable RowCount is declared locally (so therefore it isn't available to the rest of the program), and don't forget that disk files can also be declared locally to procedures. Otherwise, there isn't much RPG here of any major interest, but it is interesting to note that this journeyman RPG is actually populating a list to be displayed on a smartphone.

 

0156 – 0163            The ReadCustById reads a customer record from the CustomerL1 file by customer number. Line 157 declares the procedure's interface (where *N is a much-needed shorthand for the procedure name). If an RPG data type is declared after the *N, that is the type that the procedure returns. There must be a Return opcode with a variable of that type in the procedure to return the value to the caller. Any variable declarations between the Dcl-Pi and the End-Pi define required arguments that must be passed to the procedure. For simple procedures (those that effectively take the place of subroutines), the Dcl-Pi/End-Pi declarations can be omitted.

 

There's more to explore in the RPG presented than what's covered in the narrative, but it all follows the patterns and explanations provided in the code narrative.

 

This article intended to show TR 7.1 free-format ILE RPG in action and also that, with ASNA Mobile RPG, you can put that ILE RPG to work building a vastly nontraditional but very powerful new application type for your IBM i.

 

Reading code is always much harder than writing code, but IBM's latest RPG refresh substantially raises the bar for being able to write readable RPG. IBM deserves a big tip of the hat for finally coming 'round and providing us with this powerful, expressive RPG syntax. The IBM i is much better for it, and you'll be a much better RPG programmer using it. Please, please, investigate this new syntax. It is very much worth breaking out of your old school RPG comfort zone!

 

Resources for Learning TR 7.1 ILE RPG

Four Reasons RPG Geezers Should Care About The New Free-Form RPG, Jon Paris in IT Jungle (or anything else that Jon Paris or Susan Ganter have written about RPG—get your Google on!)

 

Simon Hutchinson's RPGPGM.COM blog

BLOG COMMENTS POWERED BY DISQUS

LATEST COMMENTS

Support MC Press Online

$0.00 Raised:
$