=====Converting programs from RISC OS to Windows=====
//by Richard Russell, October 2009//\\ \\ The conversion of **WIMP-based** BBC BASIC programs (i.e. programs having a Graphical User Interface) from **RISC OS** to **Windows** is a very broad subject, and it would be virtually impossible to provide a comprehensive guide. Therefore this article takes the form of a Case Study, dealing specifically with the conversion of one program: **BasCalc**. BasCalc is a relatively simple application, but it is hoped that the techniques used in its conversion to Windows may be applicable to other programs.\\ \\ Sections of the program that require alteration will be listed in their 'before' and 'after' forms, with an explanation of the changes. Line numbers will be retained, partly because the original version had them and partly for the convenience of being able to refer to the program by that means. However they are not required in the final (Windows) version and may be removed using the Renumber utility if desired.\\ \\ Unfortunately this Wiki doesn't provide a means to annotate the **RISC OS** and **Windows** code differently (for example using different colours). Therefore you may find it more satisfactory to compare the two versions using a utility like [[http://www.microsoft.com/downloads/details.aspx?FamilyId=49AE8576-9BB9-4126-9761-BA8011FABF38&displaylang=en|WinDiff]], which highlights the differences. To enable you to do that the two programs are available for download as plain-text files as follows: [[http://www.rtr.myzen.co.uk/BASCalc_RISCOS.bas|BASCalc_RISCOS.bas]] and [[http://www.rtr.myzen.co.uk/BASCalc_Windows.bas|BASCalc_Windows.bas]]\\ \\ Firstly here are some screenshots of the RISC OS and Windows versions for comparison:\\ \\ {{bascalc1.png}}{{bascalc4.png}}\\ {{bascalc2.png}}{{bascalc5.png}}\\ \\ Now for the conversion. The first few lines require no alteration:
10 REM >BASCalc Source code
20 REM (C)1999 PAUL VIGAY
30 REM Inspired by !Calcula, by Iain Logan and thanks are due to him for the idea.
40 REM Please feel free to examine my rudimentary wimp library and experiment to
50 REM create your own applications. A limited number of comments have been inserted for you!
60 version$="1.05 (12th Mar 2006)" : ver$=LEFT$(version$,4)
The first line to require a change is this one:
70 DIM pb1% 256 : task%=&4B534154 : path$="."
70 DIM pb1% 256 : task%=&4B534154 : path$=@dir$
The RISC OS path specification isn't appropriate for Windows. Under Windows it could be set to an absolute or relative path, but for convenience here we simply set it to **@dir$**, which in BB4W is a pre-defined 'system' variable containing the directory from which the program was loaded. Neither **pb1%** nor **task%** are required in the Windows version (in fact, **pb1%** isn't used even in the RISC OS version!) and can be retained or deleted as preferred.
80 SYS "Wimp_Initialise",200,task%,"BASCalc" TO ,task_handle%
80 SYS "SetWindowText", @hwnd%, "BASCalc"
Windows has no direct equivalent to **Wimp_Initialise** so here we replace the call with one which sets the window title.
90 PROCinit
100 ON ERROR PROCerror
110 REPEAT
120 SYS "Wimp_Poll",1,pb% TO reason%
90 ON ERROR PROCerror
100 PROCinit
110 REPEAT
120 reason% = FNwimp_poll(pb%)
Because Windows is a //pre-emptive// OS rather than a //co-operative// OS it has no direct equivalent to Wimp_Poll. However, to maintain the structure of the program, and to reduce the extent of the alterations as far as possible, the call is replaced by the function **FNwimp_poll** which we shall see later. Lines 90 and 100 have been swapped so that any errors occurring in **PROCinit** will be properly reported.
130 CASE reason% OF
140 WHEN 2:PROCopen(pb%!0)
150 WHEN 3:PROCclose(pb%!0)
160 WHEN 6:PROCmouseclick
170 WHEN 8:PROCkeypressed
180 WHEN 9:PROCmenu
190 WHEN 17,18:CASE pb%!16 OF
200 WHEN 0:end%=TRUE
210 WHEN 2:REM This application dragged something elsewhere
220 f$=FNgetstring(pb%+44):VDU7:END:REMf$=LEFT$(f$,LEN(f$)-1):l$=FNleafname(f$)
230 IF pb%!12<>saveref% ERROR 1,FNmsg("err6")
240 CASE step% OF
250 WHEN 4:PROCcheckupgrade(f$,l$)
260 ENDCASE
270 oldpb%!12=oldpb%!8:oldpb%!16=3:SYS "Wimp_SendMessage",17,oldpb%,oldpb%!4
280 ENDCASE
290 ENDCASE
130 CASE reason% OF
160 WHEN 6:PROCmouseclick
170 WHEN 8:PROCkeypressed
180 WHEN 9:PROCmenu
190 WHEN 17: end%=TRUE
290 ENDCASE
There are no essential changes here, but for neatness lines 140-150 and 190-280 have been deleted. These perform operations needed to support RISC OS which are not required under Windows. It would do no harm, other than leaving 'dead' code, to retain them since the associated 'reason codes' won't be generated by **FNwimp_poll**.
300 UNTIL end%
310 PROCmsgend:@%=at%:SYS "Wimp_CloseDown",task_handle%,task%
320 END
300 UNTIL end%
310 QUIT
Here we can simply replace the calls to **Wimp_CloseDown** and **PROCmsgend** with QUIT. In a more complex program it may well be necessary to perform some 'cleanup' operations before exit, even in the Windows version.
330
340 DEF PROCmouseclick
350 LOCAL b%,w%,i%,pos%,reason%,nextmenu%
360 pos%=pb%!0 : ypos%=pb%!4 : b%=pb%!8 : w%=pb%!12 : i%=pb%!16
370 lastwindow%=w% : lasticon%=i%
330
340 DEF PROCmouseclick
350 LOCAL i%
360 i%=pb%!16
For the mouse-click processing the information required by the Windows version is much simpler than by the RISC OS version, basically it's a single value (here **i%**) to indicate what **control** (**icon** in RISC OS-speak) was clicked. Again, there's not actually any need to make changes but for clarity the unnecessary variables have been deleted.
380 CASE b% OF
390 WHEN 1,4:CASE w% OF
400 WHEN -2:IF open% THEN
410 PROCclose(window%(2))
420 ELSE
430 PROCfront(window%(2)):SYS "Wimp_SetCaretPosition",window%(2),2,0,0,-1,LEN($arg%)
440 ENDIF
450 WHEN window%(2):CASE i% OF
460 WHEN 4:PROCui(window%(2),2,arg%,""):PROCui(window%(2),3,res%,"")
470 SYS "Wimp_SetCaretPosition",window%(2),2,0,0,-1,LEN($arg%)
480 WHEN 11:PROCui(window%(2),8,m1%,""):mem(1)=0
490 WHEN 12:PROCui(window%(2),9,m2%,""):mem(2)=0
500 WHEN 13:PROCui(window%(2),10,m3%,""):mem(3)=0
510 WHEN 20:PROCui(window%(2),17,m4%,""):mem(4)=0
520 WHEN 21:PROCui(window%(2),18,m5%,""):mem(5)=0
530 WHEN 22:PROCui(window%(2),19,m6%,""):mem(6)=0
540 WHEN 23:PROCui(window%(2),8,m1%,""):mem(1)=0:PROCui(window%(2),9,m2%,""):mem(2)=0
550 PROCui(window%(2),10,m3%,""):mem(3)=0:PROCui(window%(2),17,m4%,""):mem(4)=0
560 PROCui(window%(2),18,m5%,""):mem(5)=0:PROCui(window%(2),19,m6%,""):mem(6)=0
570 PROCui(window%(2),2,arg%,""):PROCui(window%(2),3,res%,"")
580 SYS "Wimp_SetCaretPosition",window%(2),2,0,0,-1,LEN($arg%)
590 WHEN 24:PROCinsertcode($res%)
600 ENDCASE
610 ENDCASE
620 WHEN 2:CASE w% OF
630 WHEN -2:SYS "Wimp_CreateMenu",,menu1%,pos%-60,228
640 WHEN window%(2):SYS "Wimp_CreateMenu",,menu1%,pos%-60,ypos%
650 ENDCASE
660 ENDCASE
670 ENDPROC
450 CASE i% OF
460 WHEN 4:PROCui(window%(2),2,arg%,""):PROCui(window%(2),3,res%,"")
480 WHEN 11:PROCui(window%(2),8,m1%,""):mem(1)=0
490 WHEN 12:PROCui(window%(2),9,m2%,""):mem(2)=0
500 WHEN 13:PROCui(window%(2),10,m3%,""):mem(3)=0
510 WHEN 20:PROCui(window%(2),17,m4%,""):mem(4)=0
520 WHEN 21:PROCui(window%(2),18,m5%,""):mem(5)=0
530 WHEN 22:PROCui(window%(2),19,m6%,""):mem(6)=0
540 WHEN 23:PROCui(window%(2),8,m1%,""):mem(1)=0:PROCui(window%(2),9,m2%,""):mem(2)=0
550 PROCui(window%(2),10,m3%,""):mem(3)=0:PROCui(window%(2),17,m4%,""):mem(4)=0
560 PROCui(window%(2),18,m5%,""):mem(5)=0:PROCui(window%(2),19,m6%,""):mem(6)=0
570 PROCui(window%(2),2,arg%,""):PROCui(window%(2),3,res%,"")
590 WHEN 24:PROCinsertcode($res%)
600 ENDCASE
670 ENDPROC
The only necessary change to this code is the substitution of the calls to **Wimp_SetCaretPosition** (lines 470 and 580) with a suitable Windows equivalent. In fact, since these calls are non-essential, for convenience (and laziness!) they have here simply been deleted.\\ \\ However, in Windows, the outer **CASE b% OF** and **CASE w% OF** constructs aren't required, so for neatness and clarity lines 380-440 and 610-660 have been removed and in line 450 the **WHEN window%(2)** has been deleted. Again, it would do no harm to retain this 'dead' code (so long as the original lines 350 and 360 are kept) because **FNwimp_poll** does not return values which would cause it to be activated.
680
690 DEF PROCkeypressed
700 LOCAL key%,window%,icon%,k%,len%,process%
710 key%=pb%!24 : window%=pb%!0 : icon%=pb%!4 : process%=FALSE
680
690 DEF PROCkeypressed
700 LOCAL icon%
710 icon%=pb%!4
Here we have a very similar situation to **PROCmouseclick**; once again the information required by the Windows version is simpler than the RISC OS version: just the control (icon) into which input was entered (**icon%**). In Windows, handling of keyboard input to an 'edit box' does not require the attention of the main program. Checking that the key pressed was Return (Enter), which in the RISC OS version was done here, has been transferred into **FNwimp_poll** in the Windows version.
720 CASE window% OF
730 WHEN window%(2):CASE key% OF
740 WHEN 13:CASE icon% OF
750 WHEN 2:PROCcalc
760 WHEN 8:mem(1)=VAL($m1%)
770 WHEN 9:mem(2)=VAL($m2%)
780 WHEN 10:mem(3)=VAL($m3%)
790 WHEN 17:mem(4)=VAL($m4%)
800 WHEN 18:mem(5)=VAL($m5%)
810 WHEN 19:mem(6)=VAL($m6%)
820 ENDCASE
830 SYS "Wimp_SetCaretPosition",window%(2),2,0,0,-1,LEN($arg%)
840 OTHERWISE process%=TRUE
850 ENDCASE
860 OTHERWISE process%=TRUE
870 ENDCASE
880 IF process% SYS "Wimp_ProcessKey",key%
890 ENDPROC
740 CASE icon% OF
750 WHEN 2:PROCcalc
760 WHEN 8:mem(1)=VAL($m1%)
770 WHEN 9:mem(2)=VAL($m2%)
780 WHEN 10:mem(3)=VAL($m3%)
790 WHEN 17:mem(4)=VAL($m4%)
800 WHEN 18:mem(5)=VAL($m5%)
810 WHEN 19:mem(6)=VAL($m6%)
820 ENDCASE
890 ENDPROC
Once again the only essential change is the substitution or removal of **Wimp_SetCaretPosition** (line 830) but here the superfluous **CASE window% OF** and **CASE key% OF** constructs (lines 720-730 and 830-880) and the **WHEN 13** (in line 740) have been deleted. The **Wimp_ProcessKey** has no equivalent in Windows (it is related to the co-operative nature of RISC OS).
900
910 DEF PROCmenu
920 LOCAL menu%,b%,w%,ptr%,i%
930 w%=lastwindow%:i%=lasticon%
940 CASE w% OF
950 WHEN -2,window%(2):menu%=menu1%
960 ENDCASE
970 SYS "Wimp_DecodeMenu",,menu%,pb%,mbuffer%:SYS "Wimp_GetPointerInfo",,pb%
980 IF LEFT$($mbuffer%,9)="Decimals." decimals%=VAL(MID$($mbuffer%,10)):$mbuffer%="Decimals"
990 b%=pb%!8
1000 IF b% THEN
1010 CASE $mbuffer% OF
1020 WHEN "Quit":end%=TRUE
1030 WHEN "Decimals":@%=&01020000+(decimals%<<8):PROCsavechoices
1040 ENDCASE
1050 ENDIF
1060 PROCmenucreate
1070 IF b% AND 1 SYS "Wimp_CreateMenu",,menu%,pb%!0-60,228
1080 ENDPROC
900
910 DEF PROCmenu
920 LOCAL select%, pt{}
930 DIM pt{x%,y%} : SYS "GetCursorPos", pt{}
940 select% = FNtrackpopupmenu(menu1%, &100, pt.x%, pt.y%)
950 CASE select% OF
960 WHEN 100: SYS "ShellExecute", @hwnd%, 0, @dir$+"BASCalc.txt", 0, "", 1
970 WHEN 101:
980 showfull% = NOT showfull%
990 PROCsizewindow
1000 PROCsavechoices
1010 WHEN 109: end%=TRUE
1020 WHEN 200,201,202,203,204,205,206,207,208,209,210:
1030 SYS "CheckMenuItem", menu2%, 200+decimals%, 0
1040 SYS "CheckMenuItem", menu2%, select%, 8
1050 decimals% = select%-200 : @%=&01020000+(decimals%<<8)
1060 PROCsavechoices
1070 ENDCASE
1080 ENDPROC
This is the first major change. Since the means of displaying and processing a //context// menu is quite different, **PROCmenu** has been substantially re-written. In Windows the context menu is displayed by clicking the //right// mouse button.
1090
1100 DEF PROCinit
1110 DIM pb% 256,spr% 12,msgtext% 256,msgtrans% &500,inv% 28,outv% 28
1120 DIM ws% 1500 :REM workspace for icons in windows
1130 DIM wb% 1500 :REM window block (multi-purpose)
1140 DIM mb% 500:REM menu data block
1150 DIM md% 100:REM menu block containing in-directed data
1160 DIM mbuffer% 256,mem(6)
1170 nw%=2 :REM number of windows supported
1180 DIM window%(nw%):end%=FALSE:FOR N%=1 TO 6:mem(N%)=0:NEXT N%:
leavealone=FALSE:open%=FALSE:at%=@%:decimals%=8
1190 PROCmsgload(path$+"Messages"):PROCloadwindows:PROCloadchoices:PROCmenucreate
1200 @%=&01020000+(decimals%<<8)
1210 iconbar%=FNiconbar("!BASCalc"):PROCgetmodeinfo
1220 ENDPROC
1090
1100 DEF PROCinit
1110 DIM pb% 256,msgtext% 256
1160 DIM mbuffer% 256,mem(6)
1170 nw%=2
1180 DIM window%(nw%):end%=FALSE:at%=@%:decimals%=8:showfull%=FALSE
1190 PROCloadwindows:PROCloadchoices:PROCmenucreate:PROCsizewindow
1200 @%=&01020000+(decimals%<<8)
1210 SYS "CheckMenuItem", menu2%, 200+decimals%, 8
1220 ENDPROC
Here the changes consist mostly of (optional) deletions. The allocation of buffers in lines 1120 to 1150 is not required (along with most of line 1110); **PROCmsgload** can be dispensed with (as we will see later, the most convenient way of emulating the RISC OS **MessageTrans** facility does not require initialisation) and Windows looks after its iconbar (task bar) without user involvement.\\ \\ A couple of additions have also been made: **showfull%=FALSE** (line 1180) and **PROCsizewindow** (line 1190) initialise the size of the window, and **SYS "CheckMenuItem"** (line 1210) initialises the 'tick' against the default number of decimal places in the relevant sub-menu (**menu2%**).
1230
1240 DEF FNiconbar(A$)
1250 LOCAL sp$,ih%
1260 pb%!0=-1:pb%!4=0:pb%!8=0:pb%!12=68:pb%!16=68:pb%!20=&2102
1270 $spr%=A$:pb%!24=spr%:pb%!28=1:pb%!32=LEN(A$):SYS "Wimp_CreateIcon",,pb% TO ih%
1280 =ih%
1290
1300 DEF PROCgetmodeinfo
1310 inv%!0=4:inv%!4=5:inv%!8=6:inv%!12=7:inv%!16=11:inv%!20=12:
inv%!24=-1:SYS "OS_ReadVduVariables",inv%,outv%
1320 dx%=1<<(outv%!0):dy%=1<<(outv%!4):scrx%=(outv%!16+1)*dx%:scry%=(outv%!20+1)*dy%-2
1330 ENDPROC
1340
1350 DEF PROCopen(handle%)
1360 IF handle%<>-1 THEN pb%!0=handle%:SYS "Wimp_OpenWindow",,pb%
1370 CASE handle% OF
1380 WHEN window%(2):open%=TRUE
1390 ENDCASE
1400 ENDPROC
1410
1420 DEF PROCfront(handle%)
1430 LOCAL mx%,my%
1440 SYS "OS_Mouse" TO mx%,my%
1450 pb%!0=-1:pb%!4=iconbar%:SYS "Wimp_GetIconState",,pb%:mx%=pb%!8+46:
IF iconbar%=0 mx%=scrx%-250
1460 IF handle%<>-1 PROCgetw(handle%):pb%!28=-1
1470 CASE handle% OF
1480 WHEN window%(2):pb%!4=mx%-396:pb%!12=mx%+396:REM 792x300 (main window)
1490 ENDCASE
1500 PROCopen(handle%)
1510 ENDPROC
1520
1530 DEF PROCclose(handle%)
1540 !pb%=handle%:SYS "Wimp_CloseWindow",,pb%
1550 CASE handle% OF
1560 WHEN window%(2):open%=FALSE
1570 ENDCASE
1580 ENDPROC
1590
1600 DEF PROCgetw(handle%)
1610 !pb%=handle%:SYS "Wimp_GetWindowState",,pb%
1620 vminx%=pb%!4:vminy%=pb%!8:vmaxx%=pb%!12:vmaxy%=pb%!16
1630 scrollx%=pb%!20:scrolly%=pb%!24:bx%=vminx%-scrollx%:by%=vmaxy%-scrolly%
1640 bhandle%=pb%!28:flags%=pb%!32
1650 ENDPROC
This code is specific to RISC OS and can be **deleted in its entirety**. The program will still work correctly if it is left, but doing so will waste a substantial amount of memory and potentially cause confusion to somebody reading the code.
1660
1670 DEF PROCui(window%,icon%,ptr%,text$)
1680 pb%!0=window%:pb%!8=0:pb%!12=0:$ptr%=text$:pb%!4=icon%:SYS "Wimp_SetIconState",0,pb%
1690 ENDPROC
1660
1670 DEF PROCui(window%,icon%,ptr%,text$)
1680 SYS "SetDlgItemText", window%, icon%, text$
1690 ENDPROC
Here the original code has been substituted with a Windows near-equivalent.
1700
1710 REM Setup messagetrans
1720 DEF PROCmsgload(n$)
1730 SYS "MessageTrans_FileInfo",,n$
1740 SYS "OS_Module",6,,,17+LEN(n$) TO ,,msgdesc%
1750 $(msgdesc%+16)=n$:SYS "MessageTrans_OpenFile",msgdesc%,msgdesc%+16
1760 ENDPROC
1770
1780 REM Finish with messagetrans
1790 DEF PROCmsgend
1800 SYS "XMessageTrans_CloseFile",msgdesc%:SYS "XOS_Module",7,,msgdesc%
1810 ENDPROC
This code is not required and may be **deleted**.
1820
1830 REM decode tag$ into relevant text from Messages file
1840 DEF FNmsg(tag$)
1850 LOCAL F%,L%
1860 SYS "XMessageTrans_Lookup",msgdesc%,tag$,msgtext%,255,"","" TO ,,,L%;F%
1870 IF F% AND 1 L%=0
1880 msgtext%?L%=13
1890 =$msgtext%
1820
1830 REM decode tag$ into relevant text from Messages file
1840 DEF FNmsg(tag$)
1860 SYS "GetPrivateProfileString","messages",tag$,"",msgtext%,256,path$+"BASCalc.ini"
1890 = $$msgtext%
A convenient way of emulating the RISC OS **MessageTrans** facility is to use a file in the Windows '.INI' format. Here we are using such a file called **BASCalc.ini** in which the mesages are contained in a section called **messages** (the file is listed at the end of this article). Note the use of **$$** rather than **$** in line 1890.
1900
1910 DEF FNgetstring(a%)
1920 LOCAL a$
1930 WHILE ?a%<>0:a$+=CHR$(?a%):a%+=1:ENDWHILE
1940 =a$
1950
1960 DEF PROCstring0(a%,a$) $a%=a$:a%?LENa$=0:ENDPROC
These routines may be **deleted**. BBC BASIC for Windows supports NUL-terminated strings natively using the **$$** syntax.
1970
1980 DEF PROCerror
1990 LOCAL message$,action$,E%,err%,a%
2000 REMON ERROR OFF
2010 message$=REPORT$:err%=ERR:action$=FNmsg("e1"):a%=3:message$=message$+". "+action$:VDU7
2020 E%=FNerror(a%,err%,message$)
2030 IF E%=2 OR end%=TRUE end%=TRUE:@%=at%:PROCmsgend:
SYS "Wimp_CloseDown",task_handle%,task%:END
2040 ENDPROC
1970
1980 DEF PROCerror
1990 LOCAL message$,action$,E%,err%,a%
2000 REMON ERROR OFF
2010 message$=REPORT$:err%=ERR:action$=FNmsg("e1"):a%=3:message$=message$+". "+action$:VDU7
2020 E%=FNerror(a%,err%,message$)
2030 IF E%=2 OR end%=TRUE end%=TRUE:@%=at%:QUIT
2040 ENDPROC
The only alterations required here are to delete the **PROCmsgend:SYS "Wimp_CloseDown"** and change **END** to **QUIT**. In a more complex program it may be necessary to call a 'cleanup' routine before exit.
2050
2060 DEF FNerror(E%,N%,M$)
2070 SYS "Hourglass_Smash":SYS "Wimp_DragBox",,-1
2080 !pb%=N%:$(pb%+4)=M$+CHR$(0):SYS "Wimp_ReportError",pb%,E%,"!BASCalc" TO ,E%
2090 =E%
2050
2060 DEF FNerror(E%,N%,M$)
2070 SYS "MessageBox", @hwnd%, M$, "Error "+STR$(N%), 49 TO E%
2090 =E%
Here **SYS "Wimp_ReportError"** has been substituted with **SYS "MessageBox"**. Fortunately the codes returned for OK (1) and CANCEL (2) are the same as in RISC OS!
2100
2110 DEF PROCmenucreate
2120 LOCAL loop%,pass%,T$,C1%,C2%,C3%,C4%,W%,H%,V%,NI%,items%
2130 menu1%=0:menu2%=0
2140 FOR pass%=0 TO 2 STEP 2
2150 P%=mb%:indir%=md%
2160 menu1%=P%
2170 RESTORE 2540
2180 [ OPT pass%
2190 EQUS "BASCalc":EQUB 0:EQUD 0:EQUB 7:EQUB 2:EQUB 3:EQUB 0:EQUD 120:EQUD 44:EQUD 0
2200 ]
2210 FOR items%=1 TO 3
2220 READ t$
2230 nm%=-1:f4%=indir%:f6%=LEN(t$):$f4%=t$:indir%=indir%+f6%+1:f%=0
2240 CASE t$ OF
2250 WHEN "Info":f%=0:nm%=window%(1)
2260 WHEN "Decimals":nm%=menu2%
2270 WHEN "Quit":f%=&80
2280 ENDCASE
2290 [ OPT pass%
2300 EQUD f%:EQUD nm%:EQUB 49:EQUB 1:EQUB 0:EQUB 7:EQUD f4%:EQUD -1:EQUD f6%
2310 ]
2320 NEXT items%
2330 menu2%=P%
2340 RESTORE 2580
2350 [ OPT pass%
2360 EQUS "Decimals":EQUD 0:EQUB 7:EQUB 2:EQUB 3:EQUB 0:EQUD 120:EQUD 44:EQUD 0
2370 ]
2380 FOR items%=0 TO 10
2390 READ t$
2400 nm%=-1:f4%=indir%:f6%=LEN(t$):$f4%=t$:indir%=indir%+f6%+1:f%=0
2410 IF items%=decimals% f%+=1
2420 CASE t$ OF
2430 WHEN "10":f%=&80
2440 ENDCASE
2450 [ OPT pass%
2460 EQUD f%:EQUD nm%:EQUB 49:EQUB 1:EQUB 0:EQUB 7:EQUD f4%:EQUD -1:EQUD f6%
2470 ]
2480 NEXT items%
2490 NEXT pass%
2500 IF P%-mb%>450 ERROR 1,FNmsg("e2")+" ("+STR$(P%-mb%)+")"
2510 IF indir%-md%>50 ERROR 1,FNmsg("e3")+" ("+STR$(indir%-md%)+")"
2520 ENDPROC
2530
2540 REM data for menu 1
2550 DATA Info,Decimals,Quit
2560
2570 REM data for menu 2
2580 DATA 0,1,2,3,4,5,6,7,8,9,10
2590
2100
2110 DEF PROCmenucreate
2120 SYS "CreatePopupMenu" TO menu2%
2130 SYS "AppendMenu", menu2%, 0, 200, "0"
2140 SYS "AppendMenu", menu2%, 0, 201, "1"
2150 SYS "AppendMenu", menu2%, 0, 202, "2"
2160 SYS "AppendMenu", menu2%, 0, 203, "3"
2170 SYS "AppendMenu", menu2%, 0, 204, "4"
2180 SYS "AppendMenu", menu2%, 0, 205, "5"
2190 SYS "AppendMenu", menu2%, 0, 206, "6"
2200 SYS "AppendMenu", menu2%, 0, 207, "7"
2210 SYS "AppendMenu", menu2%, 0, 208, "8"
2220 SYS "AppendMenu", menu2%, 0, 209, "9"
2230 SYS "AppendMenu", menu2%, 0, 210, "10"
2240
2250 SYS "CreatePopupMenu" TO menu1%
2260 SYS "AppendMenu", menu1%, 0, 100, "&Info"
2270 SYS "AppendMenu", menu1%, 16, menu2%, "&Decimals"
2280 SYS "AppendMenu", menu1%, 0, 101, "&Show full"
2290 SYS "AppendMenu", menu1%, 0, 109, "&Quit"
2300 ENDPROC
This is another case where the RISC OS and Windows code is quite different. Also, an extra menu item (**Show full**) has been added to the Windows version as a more convenient way of switching between the two window sizes.
2590
2600 DEF PROCloadwindows
2610 LOCAL W%,window$,N%
2620 SYS "Wimp_OpenTemplate",,path$+"Templates":wm%=ws%+4500
2630 RESTORE 2760
2640 FOR W%=1 TO nw%
2650 READ window$:SYS "Wimp_LoadTemplate",,wb%,ws%,wm%,-1,window$,0 TO ,,ws%
2660 CASE window$ OF
2670 WHEN "info":$(wb%!(88+128+20))=version$
2680 WHEN "main":arg%=wb%!(88+(2*32)+20):res%=wb%!(88+(3*32)+20)
2690 m1%=wb%!(88+(8*32)+20):m2%=wb%!(88+(9*32)+20):m3%=wb%!(88+(10*32)+20)
2700 m4%=wb%!(88+(17*32)+20):m5%=wb%!(88+(18*32)+20):m6%=wb%!(88+(19*32)+20)
2710 ENDCASE
2720 SYS "Wimp_CreateWindow",,wb% TO window%(W%)
2730 NEXT W%
2740 SYS "Wimp_CloseTemplate"
2750 ENDPROC
2760 DATA info,main
2590
2600 DEF PROCloadwindows
2605 INSTALL @lib$+"WINLIB2A"
2610 window% = FN_newdialog("BASCalc",0,0,264,95,8,1000)
2615 window%!16=&508800C4 : REM remove title bar and make child window
2620 PROC_static(window%,"Argument:",0,0,6,60,16,2)
2625 PROC_static(window%,"Result:",1,0,24,60,16,2)
2630 PROC_editbox(window%,"",2,65,4,152,14,&81)
2635 PROC_editbox(window%,"",3,65,22,152,14,1)
2640 PROC_pushbutton(window%,"Clear",4,220,4,40,15,0)
2645 PROC_static(window%,"M1",5,0,44,19,16,2)
2650 PROC_static(window%,"M2",6,0,62,19,16,2)
2655 PROC_static(window%,"M3",7,0,80,19,16,2)
2660 PROC_editbox(window%,"",8,22,41,67,14,1)
2665 PROC_editbox(window%,"",9,22,59,67,14,1)
2670 PROC_editbox(window%,"",10,22,77,67,14,1)
2675 PROC_pushbutton(window%,"Clear",11,93,41,34,15,0)
2680 PROC_pushbutton(window%,"Clear",12,93,59,34,15,0)
2685 PROC_pushbutton(window%,"Clear",13,93,77,34,15,0)
2690 PROC_static(window%,"M4",14,132,44,19,16,2)
2695 PROC_static(window%,"M5",15,132,62,19,16,2)
2700 PROC_static(window%,"M6",16,132,80,19,16,2)
2705 PROC_editbox(window%,"",17,155,41,67,14,1)
2710 PROC_editbox(window%,"",18,155,59,67,14,1)
2715 PROC_editbox(window%,"",19,155,77,67,14,1)
2720 PROC_pushbutton(window%,"Clear",20,226,41,34,15,0)
2725 PROC_pushbutton(window%,"Clear",21,226,59,34,15,0)
2730 PROC_pushbutton(window%,"Clear",22,226,77,34,15,0)
2735 PROC_pushbutton(window%,"AC",23,234,22,26,15,0)
2740 PROC_pushbutton(window%,"»",24,220,22,12,15,0)
2745 PROC_showdialog(window%)
2750 DIM arg% 255,res% 255,m1% 255,m2% 255,m3% 255,m4% 255,m5% 255,m6% 255
2755 window%(2) = !window%
2760 ENDPROC
Once again the Windows version is quite different. Although it's possible in Windows to create a dialogue window using a template (which would be more similar to the RISC OS version) it's conventional in BB4W to incorporate the various controls using inline code as shown.
2780
2790 DEF PROCcalc
....
2990 DEF FNparse(a$)
....
3350 DEF FNconvtoupper(a$)
....
3410 DEF FNstripspaces(a$)
....
3510 DEF FNextractno(a$,x%,RETURN N%)
....
3590 DEF FNchr(a$)
....
3660 DEF FNvat(a$,t%)
....
3780 DEF FNgcol(c%)
....
3870 DEF FNconvtobin(a$)
....
3970 DEF FNcomma(a$)
....
4230 =a$
Lines 2780 to 4230 inclusive are the 'meat' of the BASCalc application and **require no changes**.
4240
4250 DEF PROCinsertcode(a$)
4260 LOCAL N%
4270 IF a$<>"" FOR N%=1 TO LEN(a$):SYS "OS_Byte",153,0,ASC(MID$(a$,N%,1)):NEXT N%
4280 ENDPROC
4240
4250 DEF PROCinsertcode(a$)
4260 LOCAL N%
4270 IF a$<>"" FOR N%=1 TO LEN(a$):SYS "PostMessage",Focus%,258,ASC(MID$(a$,N%,1)),0:NEXT N%
4280 ENDPROC
The **SYS "OS_Byte"** has been substituted with a Windows equivalent (258 is WM_CHAR).
4290
4300 DEF PROCloadchoices
4310 LOCAL t%,x%,v$
4320 SYS "OS_File",5,path$+"choices" TO t%
4330 IF t%=1 THEN
4340 x%=OPENIN(path$+"choices")
4350 IF x%>0 AND x%<256 THEN
4360 INPUT#x%,v$,decimals%
4370 CLOSE#x%
4380 ENDIF
4390 ELSE
4400 PROCsavechoices
4410 ENDIF
4420 ENDPROC
4290
4300 DEF PROCloadchoices
4310 SYS "GetPrivateProfileInt","choices","decimals",8,path$+"BASCalc.ini" TO decimals%
4320 SYS "GetPrivateProfileInt","choices","showfull",0,path$+"BASCalc.ini" TO showfull%
4330 ENDPROC
The Windows code is again quite different. In addition to loading the user's preference as regards the **decimals** setting, his most recent choice for the window size is also loaded.
4430
4440 DEF PROCsavechoices
4450 LOCAL x%,d%
4460 d%=0
4470 x%=OPENOUT(path$+"choices")
4480 IF x%>0 AND x%<256 THEN
4490 PRINT#x%,ver$,decimals%
4500 CLOSE#x%
4510 ENDIF
4520 ENDPROC
4430
4440 DEF PROCsavechoices
4450 LOCAL @%
4460 SYS "WritePrivateProfileString","choices","decimals",STR$decimals%,path$+"BASCalc.ini"
4470 SYS "WritePrivateProfileString","choices","showfull",STR$showfull%,path$+"BASCalc.ini"
4480 ENDPROC
Here the user's most recent window size setting is saved in addition to the number of decimal places.\\ \\ Finally here are the **FNwimp_poll**, **FNtrackpopupmenu** and **PROCsizewindow** routines which are specific to the Windows version:
DEF FNwimp_poll(pb%)
LOCAL click%, reason%, fg%, L%
PRIVATE icon%
Queue$ = "" : !^wParam$ = ^@wparam% : ?(^wParam$+4) = 4
ON SYS Queue$ += wParam$ : RETURN
ON CLOSE Queue$ += CHR$(0)+CHR$(0)+CHR$(3)+CHR$(0) : RETURN
REPEAT
WAIT 1
IF Queue$<>"" THEN
click% = !!^Queue$
Queue$ = MID$(Queue$,5)
ENDIF
UNTIL click% OR INKEY(-12)
SYS "GetDlgItem",window%(2),icon% TO Focus%
IF click% = 24 SYS "PostMessage",window%(2),40,Focus%,1
IF click% = 0 SYS "GetForegroundWindow" TO fg% : IF fg% = @hwnd% click% = &20000
IF click% = 1 click% = icon% OR &10000 ELSE icon% = click% AND &FFFF : REM return key
CASE click% >> 16 OF
WHEN 0: reason% = 6 : pb%!8 = 1 : pb%!12 = window%(2) : pb%!16 = icon% : REM mouse
WHEN 1,512: reason% = 8 : pb%!0 = window%(2) : pb%!4 = icon% : pb%!24 = 13 : REM key
WHEN 2: reason% = 9 : REM menu
WHEN 3: reason% = 17 : pb%!16 = 0 : REM close
ENDCASE
SYS "GetDlgItemText",window%(2),2,arg%,256 TO L% : arg%?L%=13
SYS "GetDlgItemText",window%(2),3,res%,256 TO L% : res%?L%=13
SYS "GetDlgItemText",window%(2),8, m1%,256 TO L% : m1%?L%=13
SYS "GetDlgItemText",window%(2),9, m2%,256 TO L% : m2%?L%=13
SYS "GetDlgItemText",window%(2),10,m3%,256 TO L% : m3%?L%=13
SYS "GetDlgItemText",window%(2),17,m4%,256 TO L% : m4%?L%=13
SYS "GetDlgItemText",window%(2),18,m5%,256 TO L% : m5%?L%=13
SYS "GetDlgItemText",window%(2),19,m6%,256 TO L% : m6%?L%=13
= reason%
DEF FNtrackpopupmenu(hmenu%,flags%,x%,y%)
LOCAL M%, O%, P%, T%
DIM P% LOCAL 54
SYS "GetWindowLong", @hwnd%, -4 TO O%
[OPT 2
.T%
push 0
push @hwnd%
push 0
push y%
push x%
push flags%
push hmenu%
call "TrackPopupMenu"
ret 16
.M% cmp dword [esp+8],&500 : jz T% : jmp O%
]
SYS "SetWindowLong", @hwnd%, -4, M%
SYS "SendMessage", @hwnd%, &500, 0, 0 TO T%
SYS "SetWindowLong", @hwnd%, -4, O%
SYS "SendMessage", @hwnd%, 0, 0, 0
= T%
DEF PROCsizewindow
LOCAL rc{}, style%
DIM rc{l%,t%,r%,b%}
rc.l% = 0 : rc.t% = 0 : rc.r% = 267
IF showfull% rc.b% = 97 ELSE rc.b% = 43
SYS "CheckMenuItem", menu1%, 101, showfull% AND 8
SYS "MapDialogRect", window%(2), rc{}
SYS "GetWindowLong", @hwnd%, -16 TO style%
SYS "AdjustWindowRect", rc{}, style%, 0
SYS "SetWindowPos", @hwnd%, 0, 0, 0, rc.r%-rc.l%, rc.b%-rc.t%, 6
ENDPROC
Here is what **BASCalc.ini** looks like; it may be compared with the equivalent RISC OS message file:
[messages]
# Messages file for !BASCalc
# ©1999 Paul Vigay
# Current VAT rate
vat=17.5
# If money is set to Yes then VAT calculations will be rounded to the nearest penny,
# otherwise left as a decimal value
money=Yes
# Application messages - DO NOT CHANGE!
e1=Click OK to continue or CANCEL to Quit application
e2=Menu data too long for mb% block
e3=Menu data too long for md% block
berr18=Division by zero
berr20=Number too big
berr21=Negative root
berr22=Log range!
berr23=Accuracy lost!
berr24=Exponent range!
berr26=Unknown variable
berr27=Missing bracket
berr28=Bad binary/hex
chr0/chr=null
chr1=send chr to prnt
chr2=Printer ON
chr3=Printer OFF
chr7=bell
chr8=cursor back
chr9=tab
chr10=LF
chr11=cursor up
chr12=cls
chr13=CR
chr30=home
chr32=space
chr127=delete
[choices]
decimals=8
showfull=-1