Previous: 13 Large fonts and dual monitors
Next: 15 Plotting graphs
In this section:
The whole business of creating and editing tables of data reminds me that this is one of my least favourite areas of Windows programming and of using ClearWin+. Traditionally, the input to many Fortran programs (including mine) was basically a set of tables. Some programmers think that while using Windows, the user should basically be creating inputs using dialogs in which this they can create those tables. In my view, that is to miss the whole point of Windows and a graphical user interface in the first place.
Near where I live there are many superstores, builders’ merchants and factories that operate out of single-storey sheds with pitched roofs made up of steel frames with some sort of cladding. Let’s take for example a typical frame with three bays just as I have sketched in figure 14.1.
Just thinking of the steel framing, the structure might be thought of as having 9 joints, and 9 structural members. If we gave every joint a number and every member a number, then the definition of the shape of that frame is a function of two tables: the first of these tables being a list of the joint coordinates, and the second a table of the joint numbers at each end of a structural member. I will forget for the moment the need to define the properties of each structural member and also where the loads are applied and what magnitude they have. Instead, I will just consider the need for the two tables.
A graphical interface will simply never present the user with the need to enter data into those two tables. The two tables will exist, however. They will exist in the internal database of the program. They will exist when the data set is saved, and if the program loads the data set, then it will be in the form of those tables. However, when creating that dataset with a graphical user interface there may well be some interactive input starting from the user clicking perhaps on the toolbar button that generates a typical frame where the user then goes on to define the span and height by clicking on the image and responding to the appropriate dialogs. The user might have a toolbar setting which adds another bay. It might be possible to click on a structural member to set its properties via a dialog box, or to inspect them where the choice might be another dialog box or perhaps some display on the status line.
The only problem with a graphical interface lies in the resolution of the monitor which does not permit a very accurate positioning of any particular joint, and so at some stage its coordinates will be required, although only one joint at a time.
Imagining the user interface is part of developing the Windows design concept. Unfortunately, there are simply occasions when a table type of interface is absolutely necessary.
In the time that I’ve been using personal computers I have had the opportunity to use a variety of different spreadsheet applications whose names in some cases have vanished into the annals of computing history: Visicalc, Supercalc, Lotus 1-2-3, Quattro Pro. Microsoft Excel seems to have swept them all into oblivion. I did for a short time (when first exposed to a spreadsheet) think that it might displace programming with a high-level language for many routine tasks of tabulating and listing, and indeed it has. However, there are computing tasks where it simply cannot compete because it is so much slower.
I think that it is worth taking time to look deeply at how the tabular input in Excel is actually managed. I’m not sure that duplicating Excel is a useful way of spending one’s time, but looking at it will not only reveal how clever and sophisticated it is but also to identify some of the things that make it more intuitive to use particularly when simply entering a table of data.
When Excel loads, it presents a grid, and along the top of the grid taking up the first row are a set of letters and down the left-hand side taking up the first column there are a set of numbers. These enable you to select an entire column or an entire row as well as referencing every single cell by means of the column letter and the row number. You can click on any cell and enter numbers or text or indeed formulae. What you see above the top row of the grid are some boxes. In one of them you see the address is: letter and row number of the cell you have highlighted. The next control along is basically a toolbar to accept or reject anything you edit. Then you have a long data entry box where you can put in your numbers, text or formulae. Icons in the toolbox allow you to accept or reject what you have done.
Early spreadsheets only permitted this out-of-grid editing mode, and it stays there even in Excel partly because users are used to it and also because the length of the data input box allows formulae much longer than the size of the cell, which only really needs to contain the result.
If you click on a column letter you can insert a column at that position or delete it. Similarly, you can do those operations on rows. The grid in Excel can be scrolled and the mechanism for scrolling is via scroll bars. It’s pretty clever, and many tables simply don’t need that degree of flexibility – in Excel, the tables are flexible because by and large, it is about tables, whereas tables in many other applications don’t need to be so flexible because the tables are about something other than simply being a table!
If you use Excel on a regular basis then the editing actions are not only straightforward but become second nature. Users of your program, including yourself, will probably expect to edit any table using the same sort of interface as one finds in Excel. However, a grid input for a table of data will probably have a fixed number of columns. The column widths will probably be fixed and so will the row heights. There won’t be any doubt as to what goes in any cell: an integer, a real, or a text string. You may not need to be able to select a column, but you will almost certainly need to select a row because it may be necessary to delete it, to edit it or to insert a new blank row at that position, pushing all the later rows down one position in the table. There may be comparatively few columns, but lots of rows.
The first simple strategy is not to allow editing directly in the table grid but outside it in a set of data input boxes corresponding to each column. You will still need the accept and reject buttons and you will need a mechanism to pick a row for editing.
If you permit in-grid editing, then you have a choice as to whether or not to provide the out of grid editing facility as an alternative or not to provide it at all.
If you only have out of grid editing, then the job in hand is basically to provide an appropriate %rd, %rf or %rs data input box for each cell in a particular row and, apart from the ability to select a row, the table that is displayed only has to be updated with numbers when the contents of those input boxes are accepted. To make matters simpler it may even be possible not to need a response from the table itself to a click selecting a row but instead to have the row selected in its own data input box which could have a spin wheel control.
Displaying the table is particularly easy if done in an entirely graphics mode using a %gr drawing surface, since that simplifies drawing the grid lines, using colour, adjusting the font so that numbers fit in the allotted space, and if the table is longer than can be drawn in the visible part of the %gr field, simply draw the lot and rely on the inbuilt cropping facilities to not display the parts outside the actual display area. Scroll bar use is also simplified, and scrolling can be quite smooth. The issue with scrolling is to remember how much the grid had scrolled so that rows can still be picked by mouse. This approach is even simpler as the entire dataset will fit in the dialog.
Figure 14.2 illustrates this approach in a program where coordinates are associated with some points identified by a single letter. In practice, there are typically about six points, and sometimes as many as eight, but inconceivably using the whole of the alphabet. In the first version of this program the letters had to be sequential but now, gaps can be left in the sequence by leaving out some letters, particularly I and O, which can be mistaken for 1 (one) and 0 (zero). It is the only place in the program where table editing and entry is used.
Data interchange between programs originated by the same person or organisation is probably best done using unformatted files, but between different programs then a formatted and hence person readable file is probably better and almost certainly if drawn by a third party program, will be in a sort of comma-separated format.
It is possible to launch another application such as Excel using START_PROCESS@ or START_PPROCESS@, which can be done with a filename on the ‘command line’ that can later be picked up easily by the originating program. Files up to a certain length may even still be in a disk cache and therefore be readable at much greater speed than reading from the physical device. On return, a file saved in that launched application may be read by your program, but with all the caveats concerning its acceptability when it comes to reading it.
In-grid editing is possible in ClearWin+ by means of the %lv (listview) control, an array of data input boxes, and because listview is primarily a way of displaying data, it is rather easier to program out-of-grid editing, where you have the benefits of %il and %fl, and do not have to monitor every keystroke.
Essentially, however (and in my view), %lv is an abomination that will have you screaming at the computer in absolute frustration, swearing that you’ll never touch it, or sometimes even ClearWin+ ever again! The fundamental problem is that the listview control is reasonably good at displaying lists, but interacting with it in a spreadsheet mode requires a great deal of contortion to be done at a very low level in its callback function.
Basically, listview displays a set or array of character strings, one string per row of the control. Hence, if you want 10 rows for numeric input, and a row for the headings, you need an array of 11 strings. What goes in each of those strings will control the way the information is displayed, and as well as everything else, the length of text and a few control codes in the first row will control the widths in the columns. Suppose that we want those 11 rows, and have declared:
CHARACTER*(80) ROWS(11)
Row 1 will sort out the widths of the columns and the column headings, so taking an example of a handful of survey stations with x,y,z coordinates, we might have:
ROWS(1) = '|Station|Easting (m)|Northing (m)|Altitude (m)'
we will get a particular spacing based on the length of text between the separators (| symbol). In each substring, leading spaces are significant (but not very) and trailing spaces seem to be ignored, particularly in the width calculation, so
ROWS(1) = '| Station| Easting (m)| Northing (m)| Altitude (m)'
will get you slightly wider columns. That still won’t be enough to get you what you want, so perhaps you need to use the mechanism for setting those column widths, which is to follow the text in any of the cells with an underscore and an integer, then that integer will define the width of the column in pixels.
ROWS(1) = '|Station_65|Easting (m)_105|Northing (m)_105|Altitude' & //' (m)_105'
(Long layouts like this make it easy to run over column 72 and need to be continued). One other thing is that a sign character after the underscore defines how the heading is arranged: + makes the heading centred, - makes it right justified. Finally, I have used the | symbol as the separator between fields. The first character in the string defines what the separator character is and so it could be some other character if you wished.
Now, we need to judge the width and height of the control to feed into the %lv control:
IW = WINIO@ ('%^lv[options]&' iWIDTH, iHEIGHT, parameters, KB_FOR_LV_GRID)
When developing the grid, you will not be able to judge the size required, so anything will have to do for the moment. Just make it large enough – say for a start 400 across and 200 down. On my PC, provided that I don’t change the default font, each row seems to take 15 pixels, but minus 1 from the total, so 11x15-1 = 164 works perfectly – unless you use a manifest. The equivalent that worked for me was 196 – 17 for each row plus 9. Don’t ask me why. Now, for some secrets. Firstly, about those 15-pixel row heights. I measured them with an on-screen ruler! The one I use is ‘Ruler by George‘, but there are others.
There are 16 options, many of which are cosmetic. The ones typically of most use are edit_cells and go_down_on_return. So, to evolve a grid that has column headers and labels down the lefthand side that are not editable, the %^lv command format becomes:
IW = WINIO@ ('%^lv[edit_cells,go_down_on_return]&' ... to be followed by the parameters.
The parameters, in order after the pixel sizes are:
WINAPP OPTIONS (INTL, DREAL) PROGRAM LV_EXAMPLEC INCLUDE <WINDOWS.INS> CHARACTER*(80) ROWS(11) COMMON /INPUT_GRID/ ROWS COMMON /COORDINATES/ X(10), Y(10), Z(10) INTEGER, EXTERNAL:: KB_FOR_LV_GRID DIMENSION ISEL(10) NROWS = 11 iSEL = 0 iVIEW = 1 ROWS = '||||' ! explained later in the text ROWS(1) = '|Station_65|Easting (m)_105|Northing (m)_105|Altitude' & //' (m)_105' IW = WINIO@ ('%ca[ListView example]&') IW = WINIO@ ('%^lv[edit_cells,go_down_on_return]&', & 400, 164, ROWS, NROWS, iSEL, iVIEW, KB_FOR_LV_GRID) IW = WINIO@ ('%lw%sf', LW) END INTEGER FUNCTION KB_FOR_LV_GRID() C ============== CHARACTER*(80) ROWS(11) COMMON /INPUT_GRID/ ROWS KB_FOR_LV_GRID = 2 END
Now suppose that each row that follows comprises a capital letter and 3 REAL values, quoted to 3 decimal places, then:
SUBROUTINE SET_ROWS (I) C ============== CHARACTER*(80) ROWS(11) COMMON /INPUT_GRID/ ROWS COMMON /COORDINATES/ X(10), Y(10), Z(10) WRITE(ROWS(I-1),900) CHAR(I+63), X(I-1), Y(I-1), Z(I-1) 900 FORMAT ('|', A, '|', F11.3, '|', F11.3, '|', F8.3) END
I used F12.3 because in the UK, using National grid coordinates, the x and y values could (just) get into thousands of kilometres but no more, and are never negative, thus requiring at most, 7 places before the decimal point, the decimal point and 3 decimal places afterwards, making 11 in total. For many locations the string will contain blanks before the numeric values, but they don’t matter. Elevations range from 0 to slightly over 1000 metres, and may include some negatives but of much smaller magnitude, so with 4 characters before the point, the decimal point itself, and 3 characters after, I got to F8.3. SET_ROWS is fine if I have start values x, y and z, but if I do not, then I will need ‘||||’ as my string.
A completely blank row needs to have:
ROWS(11) = '||||'
as an absolutely blank string won’t show at all.
If we supposed that there was an array of character strings in the background, each one formatted as per the elements of ROWS, then scrolling up would be effectively to replace ROWS(2) to ROWS(11) with the character strings from the background, and then CALL WINDOW_UPDATE@(ROWS). The simplicity of this must be one of the strengths of listview.
A further characteristic of the %lv control is that it can take a pivot and is therefore resizeable. Unlike a drawing surface where the contents need to be redrawn when there is a resize event, list view appears to understand resizing and if shrunk below the size where everything is properly displayed (either or both horizontally and vertically) then scroll bars are drawn automatically and are acted upon without programmer intervention.
The cells displayed in a list view are individually selectable, and indeed may be edited. The problem is that the changes are not persistent and once that cell loses the focus then it returns to its original contents. What must happen is that the particular row character string must be updated and once that is done it must be re-shown in the list view with CALL WINDOW_UPDATE@ (ROWS).
The first step in the callback function is to find what cell we are in. Paradoxically, the numbering begins at the second row, but, the row and column numbers are found by two calls to the CLEARWIN_INFO@ function:
IROW = CLEARWIN_INFO@ ('ROW_NUMBER') ICOL = CLEARWIN_INFO@ ('COLUMN_NUMBER')
Armed with that positional information, then if the first column is not editable, it is a matter of setting the return value to 2 and returning.
For any other cell, the next step is to examine the reason for the callback being invoked. This is done using the function CLEARWIN_STRING@ putting the result into an appropriately long CHARACTER variable CBR:
CBR = CLEARWIN_STRING@ ('CALLBACK_REASON')
Now, you can test CBR. If it contains ‘BEGIN_EDIT’ then the callback function can return with value 2. If it contains ‘EDIT_KEY_DOWN’, then you can find the ASCII code for that key NK with:
NK = CLEARWIN_INFO@('KEYBOARD_KEY')
If the cell is to contain an unsigned INTEGER, then the only valid keystrokes are the numerals 0 ... 9 inclusive, so if it is anything else, return from the callback, but this time with return value 4, which will discard that keypress. The ASCII numbers for the digits are 48 to 57 inclusive.
Similarly, if the cell is to contain an unsigned REAL, then once again, only the numerals 0 ... 9 are immediately acceptable with an immediate return with value 2.
However, in the case of a REAL number, a decimal point is acceptable, but only if there is a single decimal point in the cell. The same goes for + and -, which not only should only occur once, but should be at the beginning of the cell contents, regardless of whether the cell contains an INTEGER or REAL.
If you allow + and -, then they have ASCII numbers 43 and 45, and the decimal point is ASCII 46 (comma is ASCII 44 for Europeans) but you first need to obtain the currently edited text from the function:
CBE = CLEARWIN_STRING@ ('EDITED_TEXT')
where CBE is a CHARACTER variable of an appropriate length. You can check for duplicated decimal points and correct, and similarly for the occurrence and position of positive and negative signs.
L = LEN_TRIM (CBE) K = 0 DO 10 M=1,L IF (CBE(M:M) .EQ. '.') K = 1 ! decimal point exists 10 CONTINUE IF (NK .GE. 48 .AND. NK .LE. 57) RETURN IF (NK .EQ. 46 .AND. K .EQ. 0) RETURN KB_FOR_LV_GRID = 4 ! reject this key RETURN
If the callback reason is given as ‘END_EDIT’, then the edited text has to be (a) converted into a value, and (b) the character string for the appropriate row has to be edited appropriately and then the listview box updated using:
CALL WINDOW_UPDATE@ (ROWS)
The numeric value of the character string in CBE can be obtained by
READ (CBE,*) variable name
And the text to be spliced into the CHARACTER string of the appropriate row is probably not CBE, but rather the variable written into a different character string using an appropriate FORMAT. Such a procedure may not be necessary for an INTEGER, where CBE will already be the right format, but writing a REAL variable will give, for example, the correct number of decimal places. You cannot do the READ if the string is ‘’, so that would create an error if you entered a cell and skipped out without putting anything in, so:
ELSE IF (CBR .EQ. 'END_EDIT') THEN L = LEN_TRIM (ROWS(IROW+1)) INDEX = L ! initialise whole INDEX array BAND = ROWS (IROW + 1) KHAR = BAND(1:1) ! find the separator INDEX(1) = 1 ! INDEX is an integer array long enough K = 2 DO 20 M=2,L IF (BAND(M:M) .EQ. KHAR) THEN INDEX(K) = M K = K + 1 ENDIF 20 CONTINUE IF (CBE .EQ. '') THEN TEXT = ' ' GO TO 30 ENDIF READ (CBE,*) VALUE WRITE (TEXT,'(F12.3)') VALUE 30 CONTINUE
Finally, you need to update ROWS:
ROWS(IROW + 1) = BAND(1:INDEX(ICOL))//TEXT//BAND(INDEX(ICOL+1):L) CALL WINDOW_UPDATE@ (ROWS) KB_FOR_LV_GRID = 2 RETURN
A program using %lv should be assemble-able from the above fragments, See Appendix I.
The first column in a listview control can be enhanced with an icon or a tick box, and the appearance changed to any one of the 4 view modes. As well as INTEGER and REAL cell contents, you can enter text or even formulae, but parsing formulae is well beyond the intentions of this book. However, those enhancements are dealt with in the online help files and if you understood the foregoing, then adding those enhancements should be plain sailing.
A grid type of input can also be arranged by placing the requisite number of %rs, %rd and %rf boxes in a rectangular array of %ob .. %cb boxes, and making the input cells have no border, for example:
IW = WINIO@ ('%4.11ob[thin_margin]&')
It is not particularly obvious that removing the data border should be part of %co:
IW = WINIO@ ('%co[no_data_border,check_on_focus_loss]&')
Each cell requires its own %cb: 44 in all in the example above. It is therefore best to define the input cell and its %cb in a loop or loops. Each variable does require an initial value, and an appropriate initial value can be specified in such a way that it triggers ‘initially_blank’, as for example, in the case of UK National Grid coordinates which are never negative, -1.0 might do. However, as some altitudes might be negative, a larger negative number has to be specified for altitude, and the same value might do for Easting and Northing, perhaps -1000.0.
Cells that contain headings should use %`rs and would therefore not be editable.
DO 100 J=2,11 IW = WINIO@ (%`rs&', CHAR(J+63)) IW = WINIO@ (%rf// TESTER(X(J-1),-1000.0)//'%cb&', X(J-1)) IW = WINIO@ (%rf// TESTER(Y(J-1),-1000.0)//'%cb&', Y(J-1)) IW = WINIO@ (%rf// TESTER(Z(J-1),-1000.0)//'%cb&', Z(J-1)) 100 CONTINUE
With:
CHARACTER*(18) FUNCTION TESTER (VALUE, CRITERION) IF (VALUE .LE. CRITERION) THEN TESTER = '[initially_blank]' ELSE TESTER = '' ENDIF END
Unlike %lv where a mechanism is needed to initialize the ROWS character strings, this grid approach may deal with the variables directly. Equally, it may be the case that using surrogate variables is the approach of choice, because that simplifies the response to Cancel (Section 16.3).
FORTRAN and the ART of Windows Programming, Copyright © Eddie Bromhead, 2023.