drivers.pas 144 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624
  1. {********[ SOURCE FILE OF GRAPHICAL FREE VISION ]**********}
  2. { }
  3. { System independent clone of DRIVERS.PAS }
  4. { }
  5. { Interface Copyright (c) 1992 Borland International }
  6. { }
  7. { Copyright (c) 1996, 1997, 1998, 1999 by Leon de Boer }
  8. { [email protected] - primary e-mail addr }
  9. { [email protected] - backup e-mail addr }
  10. { }
  11. { Original FormatStr kindly donated by Marco Schmidt }
  12. { }
  13. { Mouse callback hook under FPC with kind assistance of }
  14. { Pierre Muller, Gertjan Schouten & Florian Klaempfl. }
  15. { }
  16. {****************[ THIS CODE IS FREEWARE ]*****************}
  17. { }
  18. { This sourcecode is released for the purpose to }
  19. { promote the pascal language on all platforms. You may }
  20. { redistribute it and/or modify with the following }
  21. { DISCLAIMER. }
  22. { }
  23. { This SOURCE CODE is distributed "AS IS" WITHOUT }
  24. { WARRANTIES AS TO PERFORMANCE OF MERCHANTABILITY OR }
  25. { ANY OTHER WARRANTIES WHETHER EXPRESSED OR IMPLIED. }
  26. { }
  27. {*****************[ SUPPORTED PLATFORMS ]******************}
  28. { 16 and 32 Bit compilers }
  29. { DOS - Turbo Pascal 7.0 + (16 Bit) }
  30. { DPMI - Turbo Pascal 7.0 + (16 Bit) }
  31. { - FPC 0.9912+ (GO32V2) (32 Bit) }
  32. { WINDOWS - Turbo Pascal 7.0 + (16 Bit) }
  33. { - Delphi 1.0+ (16 Bit) }
  34. { WIN95/NT - Delphi 2.0+ (32 Bit) }
  35. { - Virtual Pascal 2.0+ (32 Bit) }
  36. { - Speedsoft Sybil 2.0+ (32 Bit) }
  37. { - FPC 0.9912+ (32 Bit) }
  38. { OS2 - Virtual Pascal 1.0+ (32 Bit) }
  39. { - Speed Pascal 1.0+ (32 Bit) }
  40. { }
  41. {******************[ REVISION HISTORY ]********************}
  42. { Version Date Fix }
  43. { ------- --------- --------------------------------- }
  44. { 1.00 26 Jul 96 First DOS/DPMI platform release }
  45. { 1.10 18 Nov 97 Windows conversion added. }
  46. { 1.20 29 Aug 97 Platform.inc sort added. }
  47. { 1.30 10 Jun 98 Virtual pascal 2.0 code added. }
  48. { 1.40 13 Jul 98 Added FormatStr by Marco Schmidt. }
  49. { 1.50 14 Jul 98 Fixed width = 0 in FormatStr. }
  50. { 1.60 13 Aug 98 Complete rewrite of FormatStr. }
  51. { 1.70 10 Sep 98 Added mouse int hook for FPC. }
  52. { 1.80 10 Sep 98 Checks run & commenting added. }
  53. { 1.90 15 Oct 98 Fixed for FPC version 0.998 }
  54. { 1.91 18 Feb 99 Added PrintStr functions }
  55. { 1.92 18 Feb 99 FormatStr literal '%' fix added }
  56. { 1.93 10 Jul 99 Sybil 2.0 code added }
  57. { 1.94 15 Jul 99 Fixed for FPC 0.9912 release }
  58. { 1.95 26 Jul 99 Windows..Scales to GFV system font }
  59. { 1.96 30 Jul 99 Fixed Ctrl+F1..F10 in GetKeyEvent }
  60. { 1.97 07 Sep 99 InitEvent, DoneEvent fixed for OS2 }
  61. { 1.98 09 Sep 99 GetMouseEvent fixed for OS2. }
  62. { 1.99 03 Nov 99 FPC windows support added. }
  63. { 2.00 26 Nov 99 Graphics stuff moved to GFVGraph }
  64. {**********************************************************}
  65. UNIT Drivers;
  66. {<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>}
  67. INTERFACE
  68. {<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>}
  69. {====Include file to sort compiler platform out =====================}
  70. {$I Platform.inc}
  71. {====================================================================}
  72. {==== Compiler directives ===========================================}
  73. {$IFNDEF PPC_FPC} { FPC doesn't support these switches }
  74. {$F+} { Force far calls - Used because of the ShowMouseProc etc... }
  75. {$A+} { Word Align Data }
  76. {$B-} { Allow short circuit boolean evaluations }
  77. {$O-} { This unit may >>> NOT <<< be overlaid }
  78. {$G+} { 286 Code optimization - if you're on an 8088 get a real computer }
  79. {$P-} { Normal string variables }
  80. {$N-} { No 80x87 code generation }
  81. {$E+} { Emulation is on }
  82. {$ENDIF}
  83. {$X+} { Extended syntax is ok }
  84. {$R-} { Disable range checking }
  85. {$S-} { Disable Stack Checking }
  86. {$I-} { Disable IO Checking }
  87. {$Q-} { Disable Overflow Checking }
  88. {$V-} { Turn off strict VAR strings }
  89. {====================================================================}
  90. USES
  91. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  92. {$IFNDEF PPC_SPEED} { NON SPEED COMPILER }
  93. {$IFDEF PPC_FPC} { FPC WINDOWS COMPILER }
  94. Windows, { Standard unit }
  95. {$ELSE} { OTHER COMPILERS }
  96. WinTypes, WinProcs, { Standard units }
  97. {$ENDIF}
  98. {$IFDEF PPC_DELPHI} { DELPHI3+ COMPILER }
  99. SysUtils, Messages, { Standard unit }
  100. {$ENDIF}
  101. {$ELSE} { SYBIL2+ COMPILER }
  102. WinBase, WinDef, WinUser, WinGDI, { Standard units }
  103. {$ENDIF}
  104. {$ENDIF}
  105. {$IFDEF OS_OS2} { OS2 CODE }
  106. {$IFDEF PPC_Virtual} { VIRTUAL PASCAL UNITS }
  107. OS2Def, OS2Base, OS2PMAPI, { Standard units }
  108. {$ENDIF}
  109. {$IFDEF PPC_Speed} { SPEED PASCAL UNITS }
  110. BseDos, Os2Def, { Standard units }
  111. {$ENDIF}
  112. {$ENDIF}
  113. Common, GFVGraph, Objects; { GFV standard units }
  114. {***************************************************************************}
  115. { PUBLIC CONSTANTS }
  116. {***************************************************************************}
  117. {---------------------------------------------------------------------------}
  118. { EVENT TYPE MASKS }
  119. {---------------------------------------------------------------------------}
  120. CONST
  121. evMouseDown = $0001; { Mouse down event }
  122. evMouseUp = $0002; { Mouse up event }
  123. evMouseMove = $0004; { Mouse move event }
  124. evMouseAuto = $0008; { Mouse auto event }
  125. evKeyDown = $0010; { Key down event }
  126. evCommand = $0100; { Command event }
  127. evBroadcast = $0200; { Broadcast event }
  128. {---------------------------------------------------------------------------}
  129. { EVENT CODE MASKS }
  130. {---------------------------------------------------------------------------}
  131. CONST
  132. evNothing = $0000; { Empty event }
  133. evMouse = $000F; { Mouse event }
  134. evKeyboard = $0010; { Keyboard event }
  135. evMessage = $FF00; { Message event }
  136. {---------------------------------------------------------------------------}
  137. { EXTENDED KEY CODES }
  138. {---------------------------------------------------------------------------}
  139. CONST
  140. kbNoKey = $0000; kbAltEsc = $0100; kbEsc = $011B;
  141. kbAltSpace = $0200; kbCtrlIns = $0400; kbShiftIns = $0500;
  142. kbCtrlDel = $0600; kbShiftDel = $0700; kbAltBack = $0800;
  143. kbAltShiftBack= $0900; kbBack = $0E08; kbCtrlBack = $0E7F;
  144. kbShiftTab = $0F00; kbTab = $0F09; kbAltQ = $1000;
  145. kbCtrlQ = $1011; kbAltW = $1100; kbCtrlW = $1117;
  146. kbAltE = $1200; kbCtrlE = $1205; kbAltR = $1300;
  147. kbCtrlR = $1312; kbAltT = $1400; kbCtrlT = $1414;
  148. kbAltY = $1500; kbCtrlY = $1519; kbAltU = $1600;
  149. kbCtrlU = $1615; kbAltI = $1700; kbCtrlI = $1709;
  150. kbAltO = $1800; kbCtrlO = $180F; kbAltP = $1900;
  151. kbCtrlP = $1910; kbAltLftBrack = $1A00; kbAltRgtBrack = $1B00;
  152. kbCtrlEnter = $1C0A; kbEnter = $1C0D; kbAltA = $1E00;
  153. kbCtrlA = $1E01; kbAltS = $1F00; kbCtrlS = $1F13;
  154. kbAltD = $2000; kbCtrlD = $2004; kbAltF = $2100;
  155. kbCtrlF = $2106; kbAltG = $2200; kbCtrlG = $2207;
  156. kbAltH = $2300; kbCtrlH = $2308; kbAltJ = $2400;
  157. kbCtrlJ = $240A; kbAltK = $2500; kbCtrlK = $250B;
  158. kbAltL = $2600; kbCtrlL = $260C; kbAltSemiCol = $2700;
  159. kbAltQuote = $2800; kbAltOpQuote = $2900; kbAltBkSlash = $2B00;
  160. kbAltZ = $2C00; kbCtrlZ = $2C1A; kbAltX = $2D00;
  161. kbCtrlX = $2D18; kbAltC = $2E00; kbCtrlC = $2E03;
  162. kbAltV = $2F00; kbCtrlV = $2F16; kbAltB = $3000;
  163. kbCtrlB = $3002; kbAltN = $3100; kbCtrlN = $310E;
  164. kbAltM = $3200; kbCtrlM = $320D; kbAltComma = $3300;
  165. kbAltPeriod = $3400; kbAltSlash = $3500; kbAltGreyAst = $3700;
  166. kbSpaceBar = $3920; kbF1 = $3B00; kbF2 = $3C00;
  167. kbF3 = $3D00; kbF4 = $3E00; kbF5 = $3F00;
  168. kbF6 = $4000; kbF7 = $4100; kbF8 = $4200;
  169. kbF9 = $4300; kbF10 = $4400; kbHome = $4700;
  170. kbUp = $4800; kbPgUp = $4900; kbGrayMinus = $4A2D;
  171. kbLeft = $4B00; kbCenter = $4C00; kbRight = $4D00;
  172. kbAltGrayPlus = $4E00; kbGrayPlus = $4E2B; kbEnd = $4F00;
  173. kbDown = $5000; kbPgDn = $5100; kbIns = $5200;
  174. kbDel = $5300; kbShiftF1 = $5400; kbShiftF2 = $5500;
  175. kbShiftF3 = $5600; kbShiftF4 = $5700; kbShiftF5 = $5800;
  176. kbShiftF6 = $5900; kbShiftF7 = $5A00; kbShiftF8 = $5B00;
  177. kbShiftF9 = $5C00; kbShiftF10 = $5D00; kbCtrlF1 = $5E00;
  178. kbCtrlF2 = $5F00; kbCtrlF3 = $6000; kbCtrlF4 = $6100;
  179. kbCtrlF5 = $6200; kbCtrlF6 = $6300; kbCtrlF7 = $6400;
  180. kbCtrlF8 = $6500; kbCtrlF9 = $6600; kbCtrlF10 = $6700;
  181. kbAltF1 = $6800; kbAltF2 = $6900; kbAltF3 = $6A00;
  182. kbAltF4 = $6B00; kbAltF5 = $6C00; kbAltF6 = $6D00;
  183. kbAltF7 = $6E00; kbAltF8 = $6F00; kbAltF9 = $7000;
  184. kbAltF10 = $7100; kbCtrlPrtSc = $7200; kbCtrlLeft = $7300;
  185. kbCtrlRight = $7400; kbCtrlEnd = $7500; kbCtrlPgDn = $7600;
  186. kbCtrlHome = $7700; kbAlt1 = $7800; kbAlt2 = $7900;
  187. kbAlt3 = $7A00; kbAlt4 = $7B00; kbAlt5 = $7C00;
  188. kbAlt6 = $7D00; kbAlt7 = $7E00; kbAlt8 = $7F00;
  189. kbAlt9 = $8000; kbAlt0 = $8100; kbAltMinus = $8200;
  190. kbAltEqual = $8300; kbCtrlPgUp = $8400; kbF11 = $8500;
  191. kbF12 = $8600; kbShiftF11 = $8700; kbShiftF12 = $8800;
  192. kbCtrlF11 = $8900; kbCtrlF12 = $8A00; kbAltF11 = $8B00;
  193. kbAltF12 = $8C00; kbCtrlUp = $8D00; kbCtrlMinus = $8E00;
  194. kbCtrlCenter = $8F00; kbCtrlGreyPlus= $9000; kbCtrlDown = $9100;
  195. kbCtrlTab = $9400; kbAltHome = $9700; kbAltUp = $9800;
  196. kbAltPgUp = $9900; kbAltLeft = $9B00; kbAltRight = $9D00;
  197. kbAltEnd = $9F00; kbAltDown = $A000; kbAltPgDn = $A100;
  198. kbAltIns = $A200; kbAltDel = $A300; kbAltTab = $A500;
  199. { ------------------------------- REMARK ------------------------------ }
  200. { New keys not initially defined by Borland in their unit interface. }
  201. { ------------------------------ END REMARK --- Leon de Boer, 15May96 - }
  202. kbFullStop = $342E; kbComma = $332C; kbBackSlash = $352F;
  203. kbApostrophe = $2827; kbSemiColon = $273B; kbEqual = $0D3D;
  204. kbGreaterThan = $343E; kbLessThan = $333C; kbQuestion = $353F;
  205. kbQuote = $2822; kbColon = $273A; kbPlus = $0D2B;
  206. kbPipe = $2B7C; kbSlash = $2B5C; kbExclaim = $0221;
  207. kbAt = $0340; kbNumber = $0423; kbPercent = $0625;
  208. kbCaret = $075E; kbAmpersand = $0826; kbAsterix = $092A;
  209. kbLeftBracket = $0A28; kbRightBracket= $0B29; kbApprox = $2960;
  210. kbTilde = $297E; kbDollar = $0524; kbMinus = $0C2D;
  211. kbUnderline = $0C5F; kbLeftSqBr = $1A5B; kbRightSqBr = $1B5D;
  212. kbLeftCurlyBr = $1A7B; kbRightCurlyBr= $1B7D;
  213. {---------------------------------------------------------------------------}
  214. { KEYBOARD STATE AND SHIFT MASKS }
  215. {---------------------------------------------------------------------------}
  216. CONST
  217. kbRightShift = $0001; { Right shift key }
  218. kbLeftShift = $0002; { Left shift key }
  219. kbCtrlShift = $0004; { Control key down }
  220. kbAltShift = $0008; { Alt key down }
  221. kbScrollState = $0010; { Scroll lock on }
  222. kbNumState = $0020; { Number lock on }
  223. kbCapsState = $0040; { Caps lock on }
  224. kbInsState = $0080; { Insert mode on }
  225. kbBothShifts = kbRightShift + kbLeftShift; { Right & Left shifts }
  226. {---------------------------------------------------------------------------}
  227. { MOUSE BUTTON STATE MASKS }
  228. {---------------------------------------------------------------------------}
  229. CONST
  230. mbLeftButton = $01; { Left mouse button }
  231. mbRightButton = $02; { Right mouse button }
  232. mbMiddleButton = $04; { Middle mouse button }
  233. {---------------------------------------------------------------------------}
  234. { SCREEN CRT MODE CONSTANTS }
  235. {---------------------------------------------------------------------------}
  236. CONST
  237. smBW80 = $0002; { Black and white }
  238. smCO80 = $0003; { Colour mode }
  239. smMono = $0007; { Monochrome mode }
  240. smFont8x8 = $0100; { 8x8 font mode }
  241. {***************************************************************************}
  242. { PUBLIC TYPE DEFINITIONS }
  243. {***************************************************************************}
  244. { ******************************* REMARK ****************************** }
  245. { The TEvent definition is completely compatable with all existing }
  246. { code but adds two new fields ID and Data into the message record }
  247. { which helps with WIN/NT and OS2 message processing. }
  248. { ****************************** END REMARK *** Leon de Boer, 11Sep97 * }
  249. {---------------------------------------------------------------------------}
  250. { EVENT RECORD DEFINITION }
  251. {---------------------------------------------------------------------------}
  252. TYPE
  253. TEvent = PACKED RECORD
  254. What: Word; { Event type }
  255. Case Word Of
  256. evNothing: (); { ** NO EVENT ** }
  257. evMouse: (
  258. Buttons: Byte; { Mouse buttons }
  259. Double: Boolean; { Double click state }
  260. Where: TPoint); { Mouse position }
  261. evKeyDown: ( { ** KEY EVENT ** }
  262. Case Integer Of
  263. 0: (KeyCode: Word); { Full key code }
  264. 1: (CharCode: Char; { Char code }
  265. ScanCode: Byte)); { Scan code }
  266. evMessage: ( { ** MESSAGE EVENT ** }
  267. Command: Word; { Message command }
  268. Id : Word; { Message id }
  269. Data : Real; { Message data }
  270. Case Word Of
  271. 0: (InfoPtr: Pointer); { Message pointer }
  272. 1: (InfoLong: Longint); { Message longint }
  273. 2: (InfoWord: Word); { Message word }
  274. 3: (InfoInt: Integer); { Message integer }
  275. 4: (InfoByte: Byte); { Message byte }
  276. 5: (InfoChar: Char)); { Message character }
  277. END;
  278. PEvent = ^TEvent;
  279. {---------------------------------------------------------------------------}
  280. { ERROR HANDLER FUNCTION DEFINITION }
  281. {---------------------------------------------------------------------------}
  282. TYPE
  283. TSysErrorFunc = FUNCTION (ErrorCode: Integer; Drive: Byte): Integer;
  284. {***************************************************************************}
  285. { INTERFACE ROUTINES }
  286. {***************************************************************************}
  287. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  288. { BUFFER MOVE ROUTINES }
  289. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  290. {-CStrLen------------------------------------------------------------
  291. Returns the length of string S, where S is a control string using tilde
  292. characters ('~') to designate shortcut characters. The tildes are
  293. excluded from the length of the string, as they will not appear on
  294. the screen. For example, given the string '~B~roccoli' as its
  295. parameter, CStrLen returns 8.
  296. 25May96 LdB
  297. ---------------------------------------------------------------------}
  298. FUNCTION CStrLen (Const S: String): Integer;
  299. {-MoveStr------------------------------------------------------------
  300. Moves a string into a buffer for use with a view's WriteBuf or WriteLine.
  301. Dest must be a TDrawBuffer (or an equivalent array of words). The
  302. characters in Str are moved into the low bytes of corresponding words
  303. in Dest. The high bytes of the words are set to Attr, or remain
  304. unchanged if Attr is zero.
  305. 25May96 LdB
  306. ---------------------------------------------------------------------}
  307. PROCEDURE MoveStr (Var Dest; Const Str: String; Attr: Byte);
  308. {-MoveCStr-----------------------------------------------------------
  309. The characters in Str are moved into the low bytes of corresponding
  310. words in Dest. The high bytes of the words are set to Lo(Attr) or
  311. Hi(Attr). Tilde characters (~) in the string toggle between the two
  312. attribute bytes passed in the Attr word.
  313. 25May96 LdB
  314. ---------------------------------------------------------------------}
  315. PROCEDURE MoveCStr (Var Dest; Const Str: String; Attrs: Word);
  316. {-MoveBuf------------------------------------------------------------
  317. Count bytes are moved from Source into the low bytes of corresponding
  318. words in Dest. The high bytes of the words in Dest are set to Attr,
  319. or remain unchanged if Attr is zero.
  320. 25May96 LdB
  321. ---------------------------------------------------------------------}
  322. PROCEDURE MoveBuf (Var Dest, Source; Attr: Byte; Count: Word);
  323. {-MoveChar------------------------------------------------------------
  324. Moves characters into a buffer for use with a view's WriteBuf or
  325. WriteLine. Dest must be a TDrawBuffer (or an equivalent array of words).
  326. The low bytes of the first Count words of Dest are set to C, or
  327. remain unchanged if Ord(C) is zero. The high bytes of the words are
  328. set to Attr, or remain unchanged if Attr is zero.
  329. 25May96 LdB
  330. ---------------------------------------------------------------------}
  331. PROCEDURE MoveChar (Var Dest; C: Char; Attr: Byte; Count: Word);
  332. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  333. { KEYBOARD SUPPORT ROUTINES }
  334. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  335. {-GetAltCode---------------------------------------------------------
  336. Returns the scancode corresponding to Alt+Ch key that is given.
  337. 25May96 LdB
  338. ---------------------------------------------------------------------}
  339. FUNCTION GetAltCode (Ch: Char): Word;
  340. {-GetCtrlCode--------------------------------------------------------
  341. Returns the scancode corresponding to Alt+Ch key that is given.
  342. 25May96 LdB
  343. ---------------------------------------------------------------------}
  344. FUNCTION GetCtrlCode (Ch: Char): Word;
  345. {-GetAltChar---------------------------------------------------------
  346. Returns the ascii character for the Alt+Key scancode that was given.
  347. 25May96 LdB
  348. ---------------------------------------------------------------------}
  349. FUNCTION GetAltChar (KeyCode: Word): Char;
  350. {-GetCtrlChar--------------------------------------------------------
  351. Returns the ascii character for the Ctrl+Key scancode that was given.
  352. 25May96 LdB
  353. ---------------------------------------------------------------------}
  354. FUNCTION GetCtrlChar (KeyCode: Word): Char;
  355. {-CtrlToArrow--------------------------------------------------------
  356. Converts a WordStar-compatible control key code to the corresponding
  357. cursor key code.
  358. 25May96 LdB
  359. ---------------------------------------------------------------------}
  360. FUNCTION CtrlToArrow (KeyCode: Word): Word;
  361. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  362. { KEYBOARD CONTROL ROUTINES }
  363. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  364. {-GetShiftState------------------------------------------------------
  365. Returns a byte containing the current Shift key state. The return
  366. value contains a combination of the kbXXXX constants for shift states.
  367. 08Jul96 LdB
  368. ---------------------------------------------------------------------}
  369. FUNCTION GetShiftState: Byte;
  370. {-GetKeyEvent--------------------------------------------------------
  371. Checks whether a keyboard event is available. If a key has been pressed,
  372. Event.What is set to evKeyDown and Event.KeyCode is set to the scan
  373. code of the key. Otherwise, Event.What is set to evNothing.
  374. 19May98 LdB
  375. ---------------------------------------------------------------------}
  376. PROCEDURE GetKeyEvent (Var Event: TEvent);
  377. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  378. { MOUSE CONTROL ROUTINES }
  379. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  380. {-ShowMouse----------------------------------------------------------
  381. Decrements the hide counter and if zero the mouse is shown on screen.
  382. 30Jun98 LdB
  383. ---------------------------------------------------------------------}
  384. PROCEDURE ShowMouse;
  385. {-HideMouse----------------------------------------------------------
  386. If mouse hide counter is zero it removes the cursor from the screen.
  387. The hide counter is then incremented by one count.
  388. 30Jun98 LdB
  389. ---------------------------------------------------------------------}
  390. PROCEDURE HideMouse;
  391. {-GetMouseEvent------------------------------------------------------
  392. Checks whether a mouse event is available. If a mouse event has occurred,
  393. Event.What is set to evMouseDown, evMouseUp, evMouseMove, or evMouseAuto
  394. and the button and double click variables are set appropriately.
  395. 06Jan97 LdB
  396. ---------------------------------------------------------------------}
  397. PROCEDURE GetMouseEvent (Var Event: TEvent);
  398. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  399. { EVENT HANDLER CONTROL ROUTINES }
  400. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  401. {-InitEvents---------------------------------------------------------
  402. Initializes the event manager, enabling the mouse handler routine and
  403. under DOS/DPMI shows the mouse on screen. It is called automatically
  404. by TApplication.Init.
  405. 02May98 LdB
  406. ---------------------------------------------------------------------}
  407. PROCEDURE InitEvents;
  408. {-DoneEvents---------------------------------------------------------
  409. Terminates event manager and disables the mouse and under DOS hides
  410. the mouse. It is called automatically by TApplication.Done.
  411. 02May98 LdB
  412. ---------------------------------------------------------------------}
  413. PROCEDURE DoneEvents;
  414. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  415. { VIDEO CONTROL ROUTINES }
  416. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  417. {-InitVideo---------------------------------------------------------
  418. Initializes the video manager, Saves the current screen mode in
  419. StartupMode, and switches to the mode indicated by ScreenMode.
  420. 19May98 LdB
  421. ---------------------------------------------------------------------}
  422. PROCEDURE InitVideo;
  423. {-DoneVideo---------------------------------------------------------
  424. Terminates the video manager by restoring the initial screen mode
  425. (given by StartupMode), clearing the screen, and restoring the cursor.
  426. Called automatically by TApplication.Done.
  427. 03Jan97 LdB
  428. ---------------------------------------------------------------------}
  429. PROCEDURE DoneVideo;
  430. {-ClearScreen--------------------------------------------------------
  431. Does nothing provided for compatability purposes only.
  432. 04Jan97 LdB
  433. ---------------------------------------------------------------------}
  434. PROCEDURE ClearScreen;
  435. {-SetVideoMode-------------------------------------------------------
  436. Does nothing provided for compatability purposes only.
  437. 04Jan97 LdB
  438. ---------------------------------------------------------------------}
  439. PROCEDURE SetVideoMode (Mode: Word);
  440. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  441. { ERROR CONTROL ROUTINES }
  442. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  443. {-InitSysError-------------------------------------------------------
  444. Error handling is not yet implemented so this simply sets
  445. SysErrActive=True (ie it lies) and exits.
  446. 20May98 LdB
  447. ---------------------------------------------------------------------}
  448. PROCEDURE InitSysError;
  449. {-DoneSysError-------------------------------------------------------
  450. Error handling is not yet implemented so this simply sets
  451. SysErrActive=False and exits.
  452. 20May98 LdB
  453. ---------------------------------------------------------------------}
  454. PROCEDURE DoneSysError;
  455. {-SystemError---------------------------------------------------------
  456. Error handling is not yet implemented so this simply drops through.
  457. 20May98 LdB
  458. ---------------------------------------------------------------------}
  459. FUNCTION SystemError (ErrorCode: Integer; Drive: Byte): Integer;
  460. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  461. { STRING FORMAT ROUTINES }
  462. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  463. {-PrintStr-----------------------------------------------------------
  464. Does nothing provided for compatability purposes only.
  465. 30Jun98 LdB
  466. ---------------------------------------------------------------------}
  467. PROCEDURE PrintStr (CONST S: String);
  468. {-FormatStr----------------------------------------------------------
  469. A string formatting routine that given a string that includes format
  470. specifiers and a list of parameters in Params, FormatStr produces a
  471. formatted output string in Result.
  472. 18Feb99 LdB
  473. ---------------------------------------------------------------------}
  474. PROCEDURE FormatStr (Var Result: String; CONST Format: String; Var Params);
  475. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  476. { >> NEW QUEUED EVENT HANDLER ROUTINES << }
  477. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  478. {-PutEventInQueue-----------------------------------------------------
  479. If there is room in the queue the event is placed in the next vacant
  480. position in the queue manager.
  481. 17Mar98 LdB
  482. ---------------------------------------------------------------------}
  483. FUNCTION PutEventInQueue (Var Event: TEvent): Boolean;
  484. {-NextQueuedEvent----------------------------------------------------
  485. If there are queued events the next event is loaded into event else
  486. evNothing is returned.
  487. 17Mar98 LdB
  488. ---------------------------------------------------------------------}
  489. PROCEDURE NextQueuedEvent(Var Event: TEvent);
  490. {***************************************************************************}
  491. { INITIALIZED PUBLIC VARIABLES }
  492. {***************************************************************************}
  493. {---------------------------------------------------------------------------}
  494. { INITIALIZED DOS/DPMI VARIABLES }
  495. {---------------------------------------------------------------------------}
  496. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  497. { ******************************* REMARK ****************************** }
  498. { In Hi-Res graphics modes where the usual mouse handler will not }
  499. { work these can be set so the user can provide his own hide, show }
  500. { and redraw on move routines, otherwise leave them set as nil. }
  501. { ****************************** END REMARK *** Leon de Boer, 20Jul98 * }
  502. TYPE DrawProc = PROCEDURE;
  503. CONST
  504. HideMouseProc: DrawProc = Nil; { Hide mouse procedure }
  505. ShowMouseProc: DrawProc = Nil; { Show mouse procedure }
  506. MouseMoveProc: DrawProc = Nil; { Mouse moved procedure }
  507. PROCEDURE HideMouseCursor;
  508. PROCEDURE ShowMouseCursor;
  509. {$ENDIF}
  510. {---------------------------------------------------------------------------}
  511. { INITIALIZED WIN/NT VARIABLES }
  512. {---------------------------------------------------------------------------}
  513. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  514. CONST
  515. AppWindow : HWnd = 0; { Application window }
  516. DefGfvFont : HFont = 0; { Default GFV font }
  517. DefFontWeight: Integer = fw_Normal; { Default font weight }
  518. DefFontStyle : String = 'Times New Roman'; { Default font style }
  519. {$ENDIF}
  520. {---------------------------------------------------------------------------}
  521. { INITIALIZED OS2 VARIABLES }
  522. {---------------------------------------------------------------------------}
  523. {$IFDEF OS_OS2} { OS2 CODE }
  524. CONST
  525. AppWindow : HWnd = 0; { Application window }
  526. Anchor : HAB = 0; { Anchor block }
  527. MsgQue : HMq = 0; { Message queue }
  528. DefGFVFont : LongInt = 0; { Default font style }
  529. DefPointer : HPointer = 0; { Default pointer }
  530. DefFontStyle: String = 'Times'; { Default font style }
  531. {$ENDIF}
  532. {---------------------------------------------------------------------------}
  533. { INITIALIZED DOS/DPMI/WIN/NT/OS2 VARIABLES }
  534. {---------------------------------------------------------------------------}
  535. CONST
  536. CheckSnow : Boolean = False; { Compatability only }
  537. MouseEvents : Boolean = False; { Mouse event state }
  538. MouseReverse : Boolean = False; { Mouse reversed }
  539. HiResScreen : Boolean = False; { Compatability only }
  540. CtrlBreakHit : Boolean = False; { Compatability only }
  541. SaveCtrlBreak: Boolean = False; { Compatability only }
  542. SysErrActive : Boolean = False; { Compatability only }
  543. FailSysErrors: Boolean = False; { Compatability only }
  544. ButtonCount : Byte = 0; { Mouse button count }
  545. DoubleDelay : Word = 8; { Double click delay }
  546. RepeatDelay : Word = 8; { Auto mouse delay }
  547. SysColorAttr : Word = $4E4F; { System colour attr }
  548. SysMonoAttr : Word = $7070; { System mono attr }
  549. StartupMode : Word = $FFFF; { Compatability only }
  550. CursorLines : Word = $FFFF; { Compatability only }
  551. ScreenBuffer : Pointer = Nil; { Compatability only }
  552. SaveInt09 : Pointer = Nil; { Compatability only }
  553. SysErrorFunc : TSysErrorFunc = SystemError; { System error ptr }
  554. {---------------------------------------------------------------------------}
  555. { >>> NEW INITIALIZED DOS/DPMI/WIN/NT/OS2 VARIABLES <<< }
  556. {---------------------------------------------------------------------------}
  557. CONST
  558. DefLineNum : Integer = 25; { Default line number }
  559. DefFontHeight : Integer = 0; { Default font height }
  560. SysFontWidth : Integer = 8; { System font width }
  561. SysFontHeight : Integer = 16; { System font height }
  562. {***************************************************************************}
  563. { UNINITIALIZED PUBLIC VARIABLES }
  564. {***************************************************************************}
  565. {---------------------------------------------------------------------------}
  566. { UNINITIALIZED DOS/DPMI/WIN/NT/OS2 VARIABLES }
  567. {---------------------------------------------------------------------------}
  568. VAR
  569. MouseIntFlag: Byte; { Mouse in int flag }
  570. MouseButtons: Byte; { Mouse button state }
  571. ScreenWidth : Byte; { Screen text width }
  572. ScreenHeight: Byte; { Screen text height }
  573. ScreenMode : Word; { Screen mode }
  574. MouseWhere : TPoint; { Mouse position }
  575. {<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>}
  576. IMPLEMENTATION
  577. {<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>}
  578. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  579. {$IFDEF PPC_FPC} { FPC DOS COMPILER }
  580. USES Go32; { Standard unit }
  581. {$ENDIF}
  582. {$ENDIF}
  583. {***************************************************************************}
  584. { PRIVATE INTERNAL CONSTANTS }
  585. {***************************************************************************}
  586. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  587. {---------------------------------------------------------------------------}
  588. { DOS/DPMI MOUSE INTERRUPT EVENT QUEUE SIZE }
  589. {---------------------------------------------------------------------------}
  590. CONST EventQSize = 16; { Default irq bufsize }
  591. {$ENDIF}
  592. {---------------------------------------------------------------------------}
  593. { DOS/DPMI/WIN/NT/OS2 NEW EVENT QUEUE MAX SIZE }
  594. {---------------------------------------------------------------------------}
  595. CONST QueueMax = 64; { Max new queue size }
  596. {***************************************************************************}
  597. { PRIVATE INTERNAL TYPES }
  598. {***************************************************************************}
  599. {$IFDEF OS_WINDOWS} { DOS/DPMI CODE }
  600. {---------------------------------------------------------------------------}
  601. { SYBIL2+ WIN/NT COMPILER TYPE FIX UPS }
  602. {---------------------------------------------------------------------------}
  603. {$IFDEF PPC_SPEED} { SYBIL2+ COMPILER }
  604. TYPE TLogFont = LogFont; { Type fix up }
  605. TYPE TMsg = Msg; { Type fix up }
  606. TYPE TTextMetric = TextMetric; { Type fix up }
  607. {$ENDIF}
  608. {$ENDIF}
  609. {***************************************************************************}
  610. { PRIVATE INTERNAL INITIALIZED VARIABLES }
  611. {***************************************************************************}
  612. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  613. {---------------------------------------------------------------------------}
  614. { DOS/DPMI INITIALIZED VARIABLES }
  615. {---------------------------------------------------------------------------}
  616. {$IFDEF GO32V2} { GO32V2 needs these }
  617. CONST
  618. RealSeg: Word = 0; { Real mode segment }
  619. RealOfs: Word = 0; { Real mode offset }
  620. MouseCallback: Pointer = Nil; { Mouse call back ptr }
  621. {$ENDIF}
  622. {$ENDIF}
  623. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  624. {---------------------------------------------------------------------------}
  625. { WIN/NT TABLE OF ALT + ASCII CODES FROM VIRTUAL CODES }
  626. {---------------------------------------------------------------------------}
  627. CONST AltVirtualToAscii: Array [0..127] Of Word =
  628. ($00, $00, $00, $00, $00, $00, $00, $00,
  629. kbAltBack, kbAltTab, $00, $00, $00, kbEnter, $00, $00,
  630. {10H} $00, $00, $00, $00, $00, $00, $00, $00,
  631. $00, $00, $00, kbEsc, $00, $00, $00, $00,
  632. {20H} kbAltSpace, kbAltPgUp, kbAltPgDn, kbAltEnd, kbAltHome,
  633. kbAltLeft, kbAltUp, kbAltRight,
  634. kbAltDown, $00, $00, $00, $00, kbAltIns, kbAltDel, $00,
  635. {30H} kbAlt0, kbAlt1, kbAlt2, kbAlt3, kbAlt4, kbAlt5, kbAlt6, kbAlt7,
  636. kbAlt8, kbAlt9, $00, $00, $00, $00, $00, $00,
  637. {40H} $00, kbAltA, kbAltB, kbAltC, kbAltD, kbAltE, kbAltF, kbAltG,
  638. kbAltH, kbAltI, kbAltJ, kbAltK, kbAltL, kbAltM, kbAltN, kbAltO,
  639. {50H} kbAltP, kbAltQ, kbAltR, kbAltS, kbAltT, kbAltU, kbAltV, kbAltW,
  640. kbAltX, kbAltY, kbAltZ, $00, $00, $00, $00, $00,
  641. {60H} $00, $00, $00, $00, $00, $00, $00, $00,
  642. $00, $00, $372A, $4E2B, $00, $4A2D, $00, $352F,
  643. {70H} kbAltF1, kbAltF2, kbAltF3, kbAltF4, kbAltF5, kbAltF6, kbAltF7, kbAltF8,
  644. kbAltF9, kbAltF10, $00, $00, $00, $00, $00, $00);
  645. {---------------------------------------------------------------------------}
  646. { WIN/NT TABLE OF WINDOWS ASCII TO INTERNATIONAL ASCII }
  647. {---------------------------------------------------------------------------}
  648. CONST WinAsciiToIntAscii: Array [128..255] Of Byte = (
  649. {80H} $00, $00, $00, $00, $00, $00, $00, $00,
  650. $00, $00, $00, $00, $00, $00, $00, $00,
  651. {90H} $00, $00, $00, $00, $00, $00, $00, $00,
  652. $00, $00, $00, $00, $00, $00, $00, $00,
  653. {A0H} $00, $AD, $BD, $9C, $CF, $BE, $B3, $F5,
  654. $00, $B8, $A6, $AE, $AA, $B0, $A9, $00,
  655. {B0H} $F8, $F1, $FD, $00, $EF, $E6, $F4, $00,
  656. $3C, $3E, $A7, $AF, $AC, $AB, $F3, $A8,
  657. {C0H} $B7, $B5, $B6, $C7, $8E, $8F, $92, $80,
  658. $D4, $90, $D2, $D3, $DE, $D6, $D7, $D8,
  659. {D0H} $D1, $A5, $E3, $E0, $E2, $E5, $99, $00,
  660. $9D, $EB, $E9, $EA, $9A, $ED, $E7, $E1,
  661. {E0H} $85, $A0, $83, $C6, $84, $86, $91, $87,
  662. $8A, $82, $88, $89, $8D, $A1, $8C, $8B,
  663. {F0H} $D0, $A4, $95, $A2, $93, $E4, $94, $F6,
  664. $9B, $97, $A3, $96, $81, $EC, $E8, $98);
  665. {$ENDIF}
  666. {---------------------------------------------------------------------------}
  667. { DOS/DPMI/WIN/NT/OS2 ALT KEY SCANCODES FROM KEYS (0-127) }
  668. {---------------------------------------------------------------------------}
  669. CONST AltCodes: Array [0..127] Of Byte = (
  670. $00, $00, $00, $00, $00, $00, $00, $00, { $00 - $07 }
  671. $00, $00, $00, $00, $00, $00, $00, $00, { $08 - $0F }
  672. $00, $00, $00, $00, $00, $00, $00, $00, { $10 - $17 }
  673. $00, $00, $00, $00, $00, $00, $00, $00, { $18 - $1F }
  674. $00, $00, $00, $00, $00, $00, $00, $00, { $20 - $27 }
  675. $00, $00, $00, $00, $00, $82, $00, $00, { $28 - $2F }
  676. $81, $78, $79, $7A, $7B, $7C, $7D, $7E, { $30 - $37 }
  677. $7F, $80, $00, $00, $00, $83, $00, $00, { $38 - $3F }
  678. $00, $1E, $30, $2E, $20, $12, $21, $22, { $40 - $47 }
  679. $23, $17, $24, $25, $26, $32, $31, $18, { $48 - $4F }
  680. $19, $10, $13, $1F, $14, $16, $2F, $11, { $50 - $57 }
  681. $2D, $15, $2C, $00, $00, $00, $00, $00, { $58 - $5F }
  682. $00, $00, $00, $00, $00, $00, $00, $00, { $60 - $67 }
  683. $00, $00, $00, $00, $00, $00, $00, $00, { $68 - $6F }
  684. $00, $00, $00, $00, $00, $00, $00, $00, { $70 - $77 }
  685. $00, $00, $00, $00, $00, $00, $00, $00); { $78 - $7F }
  686. {***************************************************************************}
  687. { PRIVATE INTERNAL INITIALIZED VARIABLES }
  688. {***************************************************************************}
  689. {---------------------------------------------------------------------------}
  690. { NEW CONTROL VARIABLES }
  691. {---------------------------------------------------------------------------}
  692. CONST
  693. HideCount : Integer = 0; { Cursor hide count }
  694. QueueCount: Word = 0; { Queued message count }
  695. QueueHead : Word = 0; { Queue head pointer }
  696. QueueTail : Word = 0; { Queue tail pointer }
  697. {***************************************************************************}
  698. { PRIVATE INTERNAL UNINITIALIZED VARIABLES }
  699. {***************************************************************************}
  700. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  701. {---------------------------------------------------------------------------}
  702. { UNINITIALIZED DOS/DPMI VARIABLES }
  703. {---------------------------------------------------------------------------}
  704. VAR
  705. LastDouble : Boolean; { Last double buttons }
  706. LastButtons: Byte; { Last button state }
  707. DownButtons: Byte; { Last down buttons }
  708. EventCount : Word; { Events in queue }
  709. AutoDelay : Word; { Delay time count }
  710. DownTicks : Word; { Down key tick count }
  711. AutoTicks : Word; { Held key tick count }
  712. LastWhereX : Word; { Last x position }
  713. LastWhereY : Word; { Last y position }
  714. DownWhereX : Word; { Last x position }
  715. DownWhereY : Word; { Last y position }
  716. EventQHead : Pointer; { Head of queue }
  717. EventQTail : Pointer; { Tail of queue }
  718. EventQueue : Array [0..EventQSize - 1] Of TEvent; { Event queue }
  719. EventQLast : RECORD END; { Simple end marker }
  720. {---------------------------------------------------------------------------}
  721. { ABSOLUTE PRIVATE DOS/DPMI ADDRESS VARIABLES }
  722. {---------------------------------------------------------------------------}
  723. VAR
  724. {$IFNDEF GO32V1}
  725. ShiftState: Byte Absolute $40:$17; { Shift state mask }
  726. Ticks: Word Absolute $40:$6C; { DOS tick counter }
  727. {$ENDIF}
  728. {$IFDEF GO32V2} { GO32V2 registers }
  729. ActionRegs: TRealRegs; { Real mode registers }
  730. {$ENDIF}
  731. {$ENDIF}
  732. {---------------------------------------------------------------------------}
  733. { UNINITIALIZED DOS/DPMI/WIN/NT/OS2 VARIABLES }
  734. {---------------------------------------------------------------------------}
  735. VAR
  736. SaveExit: Pointer; { Saved exit pointer }
  737. Queue : Array [0..QueueMax-1] Of TEvent; { New message queue }
  738. {***************************************************************************}
  739. { PRIVATE INTERNAL ROUTINES }
  740. {***************************************************************************}
  741. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  742. { DOS/DPMI ONLY PRIVATE INTERNAL ROUTINES }
  743. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  744. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  745. {$IFDEF GO32V2} { GO32V2 CODE }
  746. {---------------------------------------------------------------------------}
  747. { MouseTrap -> Platforms GO32V2 - FPC COMPILER Updated 10Sep98 LdB }
  748. {---------------------------------------------------------------------------}
  749. PROCEDURE Mouse_Trap; FAR; ASSEMBLER;
  750. ASM
  751. PUSH %ES; { Save ES register }
  752. PUSH %DS; { Save DS register }
  753. PUSHL %EDI; { Save register }
  754. PUSHL %ESI; { Save register }
  755. ;{ caution : ds is not the selector for our data !! }
  756. PUSH %ES; { Push data seg }
  757. POP %DS; { Load data seg }
  758. PUSHL %EDI; { Actionregs address }
  759. MOVL MOUSECALLBACK, %EAX; { Fetch callback addr }
  760. CMPL $0, %EAX; { Check for nil ptr }
  761. JS .L_NoCallBack; { Ignore if nil }
  762. POPL %EAX; { %EAX = @actionregs }
  763. MOVL (%EAX), %EDI; { EDI from actionregs }
  764. MOVL 4(%EAX), %ESI; { ESI from actionregs }
  765. MOVL 16(%EAX), %EBX; { EBX from actionregs }
  766. MOVL 20(%EAX), %EDX; { EDX from actionregs }
  767. MOVL 24(%EAX), %ECX; { ECX from actionregs }
  768. MOVL 28(%EAX), %EAX; { EAX from actionregs }
  769. CALL *MOUSECALLBACK; { Call callback proc }
  770. .L_NoCallBack:
  771. POPL %ESI; { Recover register }
  772. POPL %EDI; { Recover register }
  773. POP %DS; { Restore DS register }
  774. POP %ES; { Restore ES register }
  775. MOVL (%ESI), %EAX;
  776. MOVL %EAX, %ES:42(%EDI); { Set as return addr }
  777. ADDW $4, %ES:46(%EDI); { adjust stack }
  778. IRET; { Interrupt return }
  779. END;
  780. {$ENDIF}
  781. {$IFDEF PPC_FPC} { FPC COMPILER CODE }
  782. {---------------------------------------------------------------------------}
  783. { Mouse_Action -> Platforms DPMI - FPC COMPILER Updated 10Sep98 LdB }
  784. {---------------------------------------------------------------------------}
  785. PROCEDURE Mouse_Action (Mask : Integer; P : Pointer);
  786. VAR Error: Word; ErrStr: String; {$IFDEF GO32V2} Rg: TRealRegs; {$ENDIF}
  787. BEGIN
  788. {$IFDEF GO32V1} { GO32V1 CODE }
  789. ErrStr := 'GO32V1 mouse handler set failed !!'; { Set error string }
  790. ASM
  791. MOVL $0xFF, %EAX; { GO32v1 special id }
  792. MOVL P, %ECX; { Fuction to chain }
  793. MOVL $0x20, %EBX; { Event queue size > 0 }
  794. MOVL $0x12345678, %EDX; { Test for version ? }
  795. INT $0x33; { Call special wrapper }
  796. CMPW $0xFF0, %AX; { AX=$FF0 if success }
  797. JNZ .L_GO32V1Err;
  798. MOVW $0, %AX; { Zero register }
  799. JMP .LGO32V1Ok; { Now jump over }
  800. .L_GO32V1Err:
  801. MOVW $0xFFFF, %AX; { -1 to register }
  802. .L_GO32V1Ok:
  803. MOVW %AX, Error; { Set error result }
  804. END;
  805. {$ENDIF}
  806. {$IFDEF GO32V2} { GO32V2 CODE }
  807. Error := 0; { Preset no error }
  808. ErrStr := 'GO32V2 mouse handler set failed !!'; { Set error string }
  809. If (P <> MouseCallBack) Then Begin { Check func different }
  810. If (RealSeg <> 0) Then Begin { Remove old calback }
  811. Rg.AX := 12; { Function id }
  812. Rg.CX := 0; { Zero mask register }
  813. Rg.ES := 0; { Zero proc seg }
  814. Rg.DX := 0; { Zero proc ofs }
  815. RealIntr($33, Rg); { Stop INT 33 callback }
  816. ASM
  817. MOVW $0x304, %AX; { Set function id }
  818. MOVW REALSEG, %CX; { Bridged real seg }
  819. MOVW REALOFS, %DX; { Bridged real ofs }
  820. INT $0x31; { Release bridge }
  821. END;
  822. End;
  823. MouseCallback := P; { Set call back addr }
  824. If (P <> Nil) Then Begin { Check non nil proc }
  825. ASM
  826. LEAL ACTIONREGS, %EDI; { Addr of actionregs }
  827. LEAL MOUSE_TRAP, %ESI; { Procedure address }
  828. PUSH %DS; { Save DS segment }
  829. PUSH %ES; { Save ES segment }
  830. PUSH %DS;
  831. POP %ES; { ES now has dataseg }
  832. PUSH %CS;
  833. POP %DS; { DS now has codeseg }
  834. MOVW $0x303, %AX; { Function id }
  835. INT $0x31; { Call DPMI bridge }
  836. POP %ES; { Restore ES segment }
  837. POP %DS; { Restore DS segment }
  838. MOVW %CX, REALSEG; { Transfer real seg }
  839. MOVW %DX, REALOFS; { Transfer real ofs }
  840. MOVW $0, %AX; { Preset zero error }
  841. JNC .L_call_ok; { Branch if ok }
  842. MOVW $0xFFFF, %AX; { Force a -1 error }
  843. .L_call_ok:
  844. MOVW %AX, ERROR; { Return error state }
  845. END;
  846. Rg.CX := Mask; { Set mask register }
  847. End Else Begin
  848. Rg.EDI := 0; { Zero proc register }
  849. Rg.CX := 0; { Zero mask register }
  850. End;
  851. If (Error = 0) Then Begin { If no error }
  852. Rg.AX := 12; { Set function id }
  853. Rg.ES := RealSeg; { Real mode segment }
  854. Rg.DX := RealOfs; { Real mode offset }
  855. RealIntr($33, Rg); { Set interrupt 33 }
  856. End Else Begin
  857. RealSeg := 0; { Zero real mode seg }
  858. RealOfs := 0; { Zero real mode ofs }
  859. End;
  860. End;
  861. {$ENDIF}
  862. If (Error <> 0) Then Begin { Error encountered }
  863. WriteLn(ErrStr); { Write error }
  864. ReadLn; { Wait for user to see }
  865. End;
  866. END;
  867. {$ENDIF}
  868. {---------------------------------------------------------------------------}
  869. { MouseInt -> Platforms DOS/DPMI - Updated 30Jun98 LdB }
  870. {---------------------------------------------------------------------------}
  871. PROCEDURE MouseInt; FAR; ASSEMBLER;
  872. {$IFDEF ASM_BP} { BP COMPATABLE ASM }
  873. ASM
  874. MOV SI, SEG @DATA; { Fetch data segment }
  875. MOV DS, SI; { Fix data segment }
  876. MOV SI, CX; { Transfer x position }
  877. MOV MouseButtons, BL; { Update mouse buttons }
  878. MOV MouseWhere.X, SI; { Update x position }
  879. MOV MouseWhere.Y, DX; { Update y position }
  880. CMP EventCount, EventQSize; { Check if queue full }
  881. JZ @@QueueFull; { Queue is full exit }
  882. MOV ES, Seg0040; { Fetch DOS segment }
  883. MOV AX, ES:Ticks; { Fetch dos tick count }
  884. MOV DI, WORD PTR EventQTail; { Address of tail }
  885. PUSH DS; { Push to stack }
  886. POP ES; { ES to data segment }
  887. CLD; { Store forward }
  888. STOSW; { Store tick count }
  889. XCHG AX, BX; { Transfer register }
  890. STOSW; { Store button state }
  891. XCHG AX, SI; { Transfer register }
  892. STOSW; { Store x position }
  893. XCHG AX, DX; { Transfer register }
  894. STOSW; { Store y position }
  895. CMP DI, OFFSET EventQLast; { Roll if at queue end }
  896. JNE @@NoRollNeeded; { Not at queue end }
  897. MOV DI, OFFSET EventQueue; { Roll back to start }
  898. @@NoRollNeeded:
  899. MOV WORD PTR EventQTail, DI; { Update queue tail }
  900. INC EventCount; { One message added }
  901. @@QueueFull:
  902. MOV MouseIntFlag, 1; { Set interrupt flag }
  903. MOV SI, WORD PTR MouseMoveProc; { Low address word }
  904. OR SI, WORD PTR MouseMoveProc+2; { "OR" high word }
  905. JZ @@Exit; { No move call so exit }
  906. DB $66; PUSH AX; { Store EAX }
  907. DB $66; PUSH BX; { Store EBX }
  908. DB $66; PUSH CX; { Store ECX }
  909. DB $66; PUSH DX; { Store EDX }
  910. DB $66; PUSH SI; { Store ESI }
  911. DB $66; PUSH DI; { Store EDI }
  912. DB $66; PUSH BP; { Store EBP }
  913. PUSH ES; { Store ES }
  914. PUSH BP; { Standard BP push }
  915. MOV BP, SP; { Transfer stack ptr }
  916. CALL MouseMoveProc; { Standard procedure }
  917. POP BP; { Standard BP recover }
  918. POP ES; { Recover ES }
  919. DB $66; POP BP; { Recover EBP }
  920. DB $66; POP DI; { Recover EDI }
  921. DB $66; POP SI; { Recover ESI }
  922. DB $66; POP DX; { Recover EDX }
  923. DB $66; POP CX; { Recover ECX }
  924. DB $66; POP BX; { Recover EBX }
  925. DB $66; POP AX; { Recover EAX }
  926. @@Exit:
  927. END;
  928. {$ENDIF}
  929. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  930. ASM
  931. MOVW %CX, %SI; { Transfer x position }
  932. MOVB %BL, MOUSEBUTTONS; { Update mouse buttons }
  933. MOVW %SI, MOUSEWHERE; { Update x position }
  934. MOVW %DX, MOUSEWHERE+2; { Update y position }
  935. CMPW $16, EVENTCOUNT; { Check if queue full }
  936. JZ .L_QueueFull; { Queue is full exit }
  937. PUSH %ES; { Save segment }
  938. MOVW $0x40, %AX; { Fetch DOS segment }
  939. MOVW %AX, %ES; { Transfer to segment }
  940. MOVL $0x6C, %EDI; { Address of ticks }
  941. MOVW %ES:(%EDI), %AX; { Fetch dos tick count }
  942. POP %ES; { Recover segment }
  943. MOVL EVENTQTAIL, %EDI; { Queue tail address }
  944. CLD; { Store forward }
  945. STOSW; { Store tick count }
  946. XCHGW %BX, %AX; { Transfer register }
  947. STOSW; { Store button state }
  948. XCHGW %SI, %AX; { Transfer register }
  949. STOSW; { Store x position }
  950. XCHGW %DX, %AX; { Transfer register }
  951. STOSW; { Store y position }
  952. LEAL EVENTQLAST, %EAX; { Roll point address }
  953. CMPL %EAX, %EDI; { Roll if at queue end }
  954. JNE .L_NoRollNeeded; { Not at queue end }
  955. LEAL EVENTQUEUE, %EDI; { Roll back to start }
  956. .L_NoRollNeeded:
  957. MOVL %EDI, EVENTQTAIL; { Update queue tail }
  958. INCW EVENTCOUNT; { One message added }
  959. .L_QueueFull:
  960. MOVB $1, MOUSEINTFLAG; { Set interrupt flag }
  961. MOVL MOUSEMOVEPROC, %EAX; { Load proc address }
  962. CMPL $0, %EAX; { Check for nil ptr }
  963. JZ .L_Exit; { No move call so exit }
  964. PUSHL %EAX; { Store EAX }
  965. PUSHL %EBX; { Store EBX }
  966. PUSHL %ECX; { Store ECX }
  967. PUSHL %EDX; { Store EDX }
  968. PUSHL %ESI; { Store ESI }
  969. PUSHL %EDI; { Store EDI }
  970. PUSHL %EBP; { Store EBP }
  971. PUSH %ES; { Store ES }
  972. CALL %EAX; { Standard procedure }
  973. POP %ES; { Recover ES }
  974. POPL %EBP; { Recover EBP }
  975. POPL %EDI; { Recover EDI }
  976. POPL %ESI; { Recover ESI }
  977. POPL %EDX; { Recover EDX }
  978. POPL %ECX; { Recover ECX }
  979. POPL %EBX; { Recover EBX }
  980. POPL %EAX; { Recover EAX }
  981. .L_Exit:
  982. END;
  983. {$ENDIF}
  984. {---------------------------------------------------------------------------}
  985. { HideMouseCursor -> Platforms DOS/DPMI - Updated 10Sep98 LdB }
  986. {---------------------------------------------------------------------------}
  987. PROCEDURE HideMouseCursor; ASSEMBLER;
  988. {$IFDEF ASM_BP} { BP COMPATABLE ASM }
  989. ASM
  990. CMP MouseEvents, 0; { Check mouse system }
  991. JZ @@Exit; { Branch if not active }
  992. MOV AX, WORD PTR [HideMouseProc]; { Fetch offset of addr }
  993. OR AX, WORD PTR [HideMouseProc+2]; { Check for nil ptr }
  994. JZ @@UseMouseInt; { Branch if nil }
  995. CALL FAR PTR [HideMouseProc]; { Call hide mouse }
  996. JMP @@Exit; { Now exit }
  997. @@UseMouseInt:
  998. MOV AX, $2; { Load function id }
  999. PUSH BP; { Safety!! save reg }
  1000. INT $33; { Hide the mouse }
  1001. POP BP; { Restore register }
  1002. @@Exit:
  1003. END;
  1004. {$ENDIF}
  1005. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  1006. ASM
  1007. CMPB $0, MouseEvents; { Check mouse system }
  1008. JZ .L_Exit; { Branch if not active }
  1009. MOVL HideMouseProc, %EAX; { Fetch address }
  1010. ORL %EAX, %EAX; { Check for nil ptr }
  1011. JZ .L_UseMouseInt; { Branch if nil }
  1012. CALL HideMouseProc; { Call show mouse }
  1013. JMP .L_Exit; { Now exit }
  1014. .L_UseMouseInt:
  1015. MOVW $2, %AX; { Load function id }
  1016. PUSHL %EBP; { Save regigister }
  1017. INT $0x33; { Hide the mouse }
  1018. POPL %EBP; { Restore register }
  1019. .L_Exit:
  1020. END;
  1021. {$ENDIF}
  1022. {---------------------------------------------------------------------------}
  1023. { ShowMouseCursor -> Platforms DOS/DPMI - Updated 10Sep98 LdB }
  1024. {---------------------------------------------------------------------------}
  1025. PROCEDURE ShowMouseCursor; ASSEMBLER;
  1026. {$IFDEF ASM_BP} { BP COMPATABLE ASM }
  1027. ASM
  1028. CMP MouseEvents, 0; { Check mouse system }
  1029. JZ @@Exit; { Branch if not active }
  1030. MOV AX, WORD PTR [ShowMouseProc]; { Fetch offset of addr }
  1031. OR AX, WORD PTR [ShowMouseProc+2]; { Check for nil ptr }
  1032. JZ @@UseMouseInt; { Branch if nil }
  1033. CALL FAR PTR [ShowMouseProc]; { Call show mouse }
  1034. JMP @@Exit; { Now exit }
  1035. @@UseMouseInt:
  1036. MOV AX, $1; { Load function id }
  1037. PUSH BP; { Safety!! save reg }
  1038. INT $33; { Show the mouse }
  1039. POP BP; { Restore register }
  1040. @@Exit:
  1041. END;
  1042. {$ENDIF}
  1043. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  1044. ASM
  1045. CMPB $0, MouseEvents; { Check mouse system }
  1046. JZ .L_Exit; { Branch if not active }
  1047. MOVL ShowMouseProc, %EAX; { Fetch address }
  1048. ORL %EAX, %EAX; { Check for nil ptr }
  1049. JZ .L_UseMouseInt; { Branch if nil }
  1050. CALL ShowMouseProc; { Call show mouse }
  1051. JMP .L_Exit; { Now exit }
  1052. .L_UseMouseInt:
  1053. MOVW $1, %AX; { Load function id }
  1054. PUSHL %EBP; { Save regigister }
  1055. INT $0x33; { Hide the mouse }
  1056. POPL %EBP; { Restore register }
  1057. .L_Exit:
  1058. END;
  1059. {$ENDIF}
  1060. {---------------------------------------------------------------------------}
  1061. { HookMouse -> Platforms DOS/DPMI - Updated 27Aug98 LdB }
  1062. {---------------------------------------------------------------------------}
  1063. PROCEDURE HookMouse;
  1064. BEGIN
  1065. {$IFDEF ASM_BP} { BP COMPTABABLE ASM }
  1066. ASM
  1067. MOV AX, $000C; { Set user interrupt }
  1068. MOV CX, $FFFF; { For all event masks }
  1069. MOV DX, OFFSET CS:MouseInt; { Mouse int is hook }
  1070. PUSH CS; { Push code segment }
  1071. POP ES; { ES:DX -> MouseInt }
  1072. PUSH BP; { Safety!! save reg }
  1073. INT $33; { Hook the routine }
  1074. POP BP; { Restore register }
  1075. END;
  1076. {$ENDIF}
  1077. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  1078. {$IFDEF GO32V2} { GO32V2 CODE }
  1079. Lock_Code(Pointer(@Mouse_Trap), 400); { Lock trap code }
  1080. Lock_Data(ActionRegs, SizeOf(ActionRegs)); { Lock registers }
  1081. {$ENDIF}
  1082. Mouse_Action(-1, @MouseInt); { Set masks/interrupt }
  1083. {$ENDIF}
  1084. END;
  1085. {---------------------------------------------------------------------------}
  1086. { UnHookMouse -> Platforms DOS/DPMI - Updated 27Aug98 LdB }
  1087. {---------------------------------------------------------------------------}
  1088. PROCEDURE UnHookMouse;
  1089. BEGIN
  1090. {$IFDEF ASM_BP} { BP COMPATABLE ASM }
  1091. ASM
  1092. MOV AX, $000C; { Set user interrupt }
  1093. XOR CX, CX; { Clear all masks }
  1094. XOR DX, DX; { Clear register }
  1095. MOV ES, CX; { ES:DX -> Nil }
  1096. PUSH BP; { Safety!! save reg }
  1097. INT $33; { Release mouse hook }
  1098. POP BP; { Restore register }
  1099. END;
  1100. {$ENDIF}
  1101. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  1102. Mouse_Action(0, Nil); { Clear mask/interrupt }
  1103. {$IFDEF GO32V2} { GO32V2 CODE }
  1104. Unlock_Code(Pointer(@Mouse_Trap), 400); { Release trap code }
  1105. Unlock_Data(ActionRegs, SizeOf(TRealRegs)); { Release registers }
  1106. {$ENDIF}
  1107. {$ENDIF}
  1108. END;
  1109. {---------------------------------------------------------------------------}
  1110. { GetMousePosition -> Platforms DOS/DPMI - Updated 19May98 LdB }
  1111. {---------------------------------------------------------------------------}
  1112. PROCEDURE GetMousePosition (Var X, Y: Integer); ASSEMBLER;
  1113. {$IFDEF ASM_BP} { BP COMPATABLE ASM }
  1114. ASM
  1115. MOV AX, $3; { Set function id }
  1116. PUSH BP; { Safety!! save reg }
  1117. INT $33; { Get button data }
  1118. POP BP; { Restore register }
  1119. LES DI, X; { Adress of x }
  1120. MOV ES:[DI], CX; { Return x position }
  1121. LES DI, Y; { Adress of y }
  1122. MOV ES:[DI], DX; { Return y position }
  1123. END;
  1124. {$ENDIF}
  1125. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  1126. ASM
  1127. MOVW $3, %AX; { Set function id }
  1128. PUSHL %EBP; { Save register }
  1129. INT $0x33; { Get button data }
  1130. POPL %EBP; { Restore register }
  1131. MOVL X, %EDI; { Adress of x }
  1132. MOVW %CX, (%EDI); { Return x position }
  1133. MOVL Y, %EDI; { Adress of y }
  1134. MOVW %DX, (%EDI); { Return y position }
  1135. END;
  1136. {$ENDIF}
  1137. {$ENDIF}
  1138. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  1139. { DOS/DPMI/WIN/NT/OS2 PRIVATE INTERNAL ROUTINES }
  1140. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  1141. {---------------------------------------------------------------------------}
  1142. { ExitDrivers -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 08Jun98 LdB }
  1143. {---------------------------------------------------------------------------}
  1144. PROCEDURE ExitDrivers; FAR;
  1145. BEGIN
  1146. DoneSysError; { Relase error trap }
  1147. DoneEvents; { Close event driver }
  1148. ExitProc := SaveExit; { Restore old exit }
  1149. END;
  1150. {---------------------------------------------------------------------------}
  1151. { DetectVideo -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 19May98 LdB }
  1152. {---------------------------------------------------------------------------}
  1153. PROCEDURE DetectVideo;
  1154. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  1155. ASSEMBLER;
  1156. {$IFDEF ASM_BP} { BP COMPATABLE ASM }
  1157. ASM
  1158. MOV AH, $0F; { Set function id }
  1159. PUSH BP; { Safety!! save reg }
  1160. INT $10; { Get current crt mode }
  1161. POP BP; { Restore register }
  1162. PUSH AX; { Hold result }
  1163. MOV AX, $1130; { Set function id }
  1164. MOV BH, 0; { Zero register }
  1165. MOV DL, 0; { Zero register }
  1166. PUSH BP; { Safety!! save reg }
  1167. INT $10; { Get ext-video mode }
  1168. POP BP; { Restore register }
  1169. POP AX; { Recover held value }
  1170. MOV DH, AH; { Transfer high mode }
  1171. CMP DL, 25; { Check screen ht }
  1172. SBB AH, AH; { Subtract borrow }
  1173. INC AH; { Make #1 if in high }
  1174. CMP AL, smMono; { Is screen mono }
  1175. JZ @@Exit1; { Exit of mono }
  1176. CMP AL, smBW80; { Is screen B&W }
  1177. JZ @@Exit1; { Exit if B&W }
  1178. MOV AX, smCO80; { Else set to colour }
  1179. @@Exit1:
  1180. MOV ScreenMode, AX; { Hold screen mode }
  1181. END;
  1182. {$ENDIF}
  1183. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  1184. ASM
  1185. MOVB $0x0F, %AH; { Set function id }
  1186. PUSHL %EBP; { Save register }
  1187. INT $0x10; { Get current crt mode }
  1188. POPL %EBP; { Restore register }
  1189. PUSHL %EAX; { Hold result }
  1190. MOVW $0x1130, %AX; { Set function id }
  1191. MOVB $0, %BH; { Zero register }
  1192. MOVB $0, %DL; { Zero register }
  1193. PUSHL %EBP; { Safety!! save reg }
  1194. INT $0x10; { Get ext-video mode }
  1195. POPL %EBP; { Restore register }
  1196. POPL %EAX; { Recover held value }
  1197. MOVB %AH, %DH; { Transfer high mode }
  1198. CMPB $25, %DL; { Check screen ht }
  1199. SBB %AH, %AH; { Subtract borrow }
  1200. INCB %AH; { Make #1 if in high }
  1201. CMPB $07, %AL; { Is screen mono }
  1202. JZ .L_Exit1; { Exit of mono }
  1203. CMPB $02, %AL; { Is screen B&W }
  1204. JZ .L_Exit1; { Exit if B&W }
  1205. MOVW $03, %AX; { Else set to colour }
  1206. .L_Exit1:
  1207. MOVW %AX, SCREENMODE; { Hold screen mode }
  1208. END;
  1209. {$ENDIF}
  1210. {$ENDIF}
  1211. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  1212. VAR Dc: HDC;
  1213. BEGIN
  1214. Dc := GetDc(0); { Get screen context }
  1215. If ((GetDeviceCaps(Dc, BitsPixel) > 1) OR { Colour capacity }
  1216. (GetDeviceCaps(Dc, Planes) > 1)) Then { Colour capacity }
  1217. ScreenMode := smCO80 Else ScreenMode := smMono; { Screen mode }
  1218. ReleaseDc(0, Dc); { Release context }
  1219. END;
  1220. {$ENDIF}
  1221. {$IFDEF OS_OS2} { OS2 CODE }
  1222. VAR Ps: Hps; Dc: Hdc; Colours: LongInt;
  1223. BEGIN
  1224. Ps := WinGetPS(HWND_Desktop); { Get desktop PS }
  1225. Dc := GpiQueryDevice(Ps); { Get gpi context }
  1226. DevQueryCaps(Dc, Caps_Phys_Colors, 1, Colours); { Colour capacity }
  1227. If (Colours> 2) Then ScreenMode := smCO80 { Colour screen }
  1228. Else ScreenMode := smMono; { Mono screen }
  1229. WinReleasePS(Ps); { Release desktop PS }
  1230. END;
  1231. {$ENDIF}
  1232. {---------------------------------------------------------------------------}
  1233. { DetectMouse -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 19May98 LdB }
  1234. {---------------------------------------------------------------------------}
  1235. FUNCTION DetectMouse: Byte;
  1236. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  1237. {$IFDEF ASM_BP} { BP COMPATABLE ASM }
  1238. ASSEMBLER;
  1239. ASM
  1240. MOV AX, $3533; { Set function id }
  1241. PUSH BP; { Safety!! save reg }
  1242. INT $21; { Get mouse interrupt }
  1243. POP BP; { Restore register }
  1244. MOV AX, ES; { Transfer register }
  1245. OR AX, BX; { Check for nil ptr }
  1246. JZ @@Exit2; { Jump no mouse driver }
  1247. XOR AX, AX; { Set function id }
  1248. PUSH BP; { Safety!! save reg }
  1249. INT $33; { Reset mouse }
  1250. POP BP; { Restore register }
  1251. OR AX, AX; { Check for success }
  1252. JZ @@Exit2; { Reset mouse failed }
  1253. MOV AX, BX; { Return button count }
  1254. @@Exit2:
  1255. END;
  1256. {$ENDIF}
  1257. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  1258. ASSEMBLER;
  1259. ASM
  1260. MOVW $0x200, %AX; { Get real mode int }
  1261. MOVW $0x33, %BX; { Vector 33H }
  1262. PUSHL %EBP; { Save register }
  1263. INT $0x31; { Get the address }
  1264. POPL %EBP; { Restore register }
  1265. MOVW %CX, %AX; { Transfer register }
  1266. ORW %DX, %AX; { Check for nil ptr }
  1267. JZ .L_Exit2; { Jump no mouse driver }
  1268. XORW %AX, %AX; { Set function id }
  1269. PUSHL %EBP; { Save register }
  1270. INT $0x33; { Reset mouse driver }
  1271. POPL %EBP; { Restore register }
  1272. ORW %AX, %AX; { Check for success }
  1273. JZ .L_Exit2; { Reset mouse failed }
  1274. MOVW %BX, %AX; { Return button count }
  1275. .L_Exit2:
  1276. END;
  1277. {$ENDIF}
  1278. {$ENDIF}
  1279. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  1280. BEGIN
  1281. If (GetSystemMetrics(sm_MousePresent) <> 0) Then
  1282. DetectMouse := 2 Else DetectMouse := 0; { Buttons present }
  1283. END;
  1284. {$ENDIF}
  1285. {$IFDEF OS_OS2} { OS2 CODE }
  1286. BEGIN
  1287. DetectMouse := WinQuerySysValue(HWND_Desktop,
  1288. SV_CMouseButtons); { Buttons present }
  1289. END;
  1290. {$ENDIF}
  1291. {***************************************************************************}
  1292. { INTERFACE ROUTINES }
  1293. {***************************************************************************}
  1294. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  1295. { BUFFER MOVE ROUTINES }
  1296. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  1297. {---------------------------------------------------------------------------}
  1298. { CStrLen -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 25May96 LdB }
  1299. {---------------------------------------------------------------------------}
  1300. FUNCTION CStrLen (Const S: String): Integer;
  1301. VAR I, J: Integer;
  1302. BEGIN
  1303. J := 0; { Set result to zero }
  1304. For I := 1 To Length(S) Do
  1305. If (S[I] <> '~') Then Inc(J); { Inc count if not ~ }
  1306. CStrLen := J; { Return length }
  1307. END;
  1308. {---------------------------------------------------------------------------}
  1309. { MoveStr -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 10Jul99 LdB }
  1310. {---------------------------------------------------------------------------}
  1311. PROCEDURE MoveStr (Var Dest; Const Str: String; Attr: Byte);
  1312. VAR I: Word; P: PWord;
  1313. BEGIN
  1314. For I := 1 To Length(Str) Do Begin { For each character }
  1315. P := @TWordArray(Dest)[I-1]; { Pointer to word }
  1316. If (Attr <> 0) Then WordRec(P^).Hi := Attr; { Copy attribute }
  1317. WordRec(P^).Lo := Byte(Str[I]); { Copy string char }
  1318. End;
  1319. END;
  1320. {---------------------------------------------------------------------------}
  1321. { MoveCStr -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 10Jul99 LdB }
  1322. {---------------------------------------------------------------------------}
  1323. PROCEDURE MoveCStr (Var Dest; Const Str: String; Attrs: Word);
  1324. VAR B: Byte; I, J: Word; P: PWord;
  1325. BEGIN
  1326. J := 0; { Start position }
  1327. For I := 1 To Length(Str) Do Begin { For each character }
  1328. If (Str[I] <> '~') Then Begin { Not tilde character }
  1329. P := @TWordArray(Dest)[J]; { Pointer to word }
  1330. If (Lo(Attrs) <> 0) Then
  1331. WordRec(P^).Hi := Lo(Attrs); { Copy attribute }
  1332. WordRec(P^).Lo := Byte(Str[I]); { Copy string char }
  1333. Inc(J); { Next position }
  1334. End Else Begin
  1335. B := Hi(Attrs); { Hold attribute }
  1336. WordRec(Attrs).Hi := Lo(Attrs); { Copy low to high }
  1337. WordRec(Attrs).Lo := B; { Complete exchange }
  1338. End;
  1339. End;
  1340. END;
  1341. {---------------------------------------------------------------------------}
  1342. { MoveBuf -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 10Jul99 LdB }
  1343. {---------------------------------------------------------------------------}
  1344. PROCEDURE MoveBuf (Var Dest, Source; Attr: Byte; Count: Word);
  1345. VAR I: Word; P: PWord;
  1346. BEGIN
  1347. For I := 1 To Count Do Begin
  1348. P := @TWordArray(Dest)[I-1]; { Pointer to word }
  1349. If (Attr <> 0) Then WordRec(P^).Hi := Attr; { Copy attribute }
  1350. WordRec(P^).Lo := TByteArray(Source)[I-1]; { Copy source data }
  1351. End;
  1352. END;
  1353. {---------------------------------------------------------------------------}
  1354. { MoveChar -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 10Jul99 LdB }
  1355. {---------------------------------------------------------------------------}
  1356. PROCEDURE MoveChar (Var Dest; C: Char; Attr: Byte; Count: Word);
  1357. VAR I: Word; P: PWord;
  1358. BEGIN
  1359. For I := 1 To Count Do Begin
  1360. P := @TWordArray(Dest)[I-1]; { Pointer to word }
  1361. If (Attr <> 0) Then WordRec(P^).Hi := Attr; { Copy attribute }
  1362. If (Ord(C) <> 0) Then WordRec(P^).Lo := Byte(C); { Copy character }
  1363. End;
  1364. END;
  1365. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  1366. { KEYBOARD SUPPORT ROUTINES }
  1367. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  1368. {---------------------------------------------------------------------------}
  1369. { GetAltCode -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 25May96 LdB }
  1370. {---------------------------------------------------------------------------}
  1371. FUNCTION GetAltCode (Ch: Char): Word;
  1372. BEGIN
  1373. GetAltCode := 0; { Preset zero return }
  1374. Ch := UpCase(Ch); { Convert upper case }
  1375. If (Ch < #128) Then
  1376. GetAltCode := AltCodes[Ord(Ch)] SHL 8 { Return code }
  1377. Else If (Ch = #240) Then GetAltCode := $0200 { Return code }
  1378. Else GetAltCode := 0; { Return zero }
  1379. END;
  1380. {---------------------------------------------------------------------------}
  1381. { GetCtrlCode -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 25May96 LdB }
  1382. {---------------------------------------------------------------------------}
  1383. FUNCTION GetCtrlCode (Ch: Char): Word;
  1384. BEGIN
  1385. GetCtrlCode := GetAltCode(Ch) OR (Ord(Ch) - $40); { Ctrl+key code }
  1386. END;
  1387. {---------------------------------------------------------------------------}
  1388. { GetAltChar -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 25May96 LdB }
  1389. {---------------------------------------------------------------------------}
  1390. FUNCTION GetAltChar (KeyCode: Word): Char;
  1391. VAR I: Integer;
  1392. BEGIN
  1393. GetAltChar := #0; { Preset fail return }
  1394. If (Lo(KeyCode) = 0) Then Begin { Extended key }
  1395. If (Hi(KeyCode) < 128) Then Begin { Key between 0-127 }
  1396. I := 0; { Start at first }
  1397. While (I < 128) AND (Hi(KeyCode) <> AltCodes[I])
  1398. Do Inc(I); { Search for match }
  1399. If (I < 128) Then GetAltChar := Chr(I); { Return character }
  1400. End Else
  1401. If (Hi(KeyCode)=$02) Then GetAltChar := #240; { Return char }
  1402. End;
  1403. END;
  1404. {---------------------------------------------------------------------------}
  1405. { GetCtrlChar -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 25May96 LdB }
  1406. {---------------------------------------------------------------------------}
  1407. FUNCTION GetCtrlChar (KeyCode: Word): Char;
  1408. VAR C: Char;
  1409. BEGIN
  1410. C := #0; { Preset #0 return }
  1411. If (Lo(KeyCode) > 0) AND (Lo(KeyCode) <= 26) Then { Between 1-26 }
  1412. C := Chr(Lo(KeyCode) + $40); { Return char A-Z }
  1413. GetCtrlChar := C; { Return result }
  1414. END;
  1415. {---------------------------------------------------------------------------}
  1416. { CtrlToArrow -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 25May96 LdB }
  1417. {---------------------------------------------------------------------------}
  1418. FUNCTION CtrlToArrow (KeyCode: Word): Word;
  1419. CONST NumCodes = 11;
  1420. CtrlCodes : Array [0..NumCodes-1] Of Char =
  1421. (#19, #4, #5, #24, #1, #6, #7, #22, #18, #3, #8);
  1422. ArrowCodes: Array [0..NumCodes-1] Of Word =
  1423. (kbLeft, kbRight, kbUp, kbDown, kbHome, kbEnd, kbDel, kbIns,
  1424. kbPgUp, kbPgDn, kbBack);
  1425. VAR I: Integer;
  1426. BEGIN
  1427. CtrlToArrow := KeyCode; { Preset key return }
  1428. For I := 0 To NumCodes - 1 Do
  1429. If WordRec(KeyCode).Lo = Byte(CtrlCodes[I]) { Matches a code }
  1430. Then Begin
  1431. CtrlToArrow := ArrowCodes[I]; { Return key stroke }
  1432. Exit; { Now exit }
  1433. End;
  1434. END;
  1435. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  1436. { KEYBOARD CONTROL ROUTINES }
  1437. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  1438. {---------------------------------------------------------------------------}
  1439. { GetShiftState -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 08Jul96 LdB }
  1440. {---------------------------------------------------------------------------}
  1441. FUNCTION GetShiftState: Byte;
  1442. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  1443. {$IFDEF ASM_BP} { BP COMPATABLE ASM }
  1444. ASSEMBLER;
  1445. ASM
  1446. MOV ES, Seg0040; { Load DOS segment }
  1447. XOR AX, AX;
  1448. MOV DX, AX; { Clear registers }
  1449. MOV AL, ES:[$0017]; { Read shift state }
  1450. END;
  1451. {$ENDIF}
  1452. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  1453. BEGIN
  1454. ASM
  1455. MOVW $0x0200, %AX; { Set function id }
  1456. PUSHL %EBP; { Save register }
  1457. INT $0x16; { Get shift status }
  1458. POPL %EBP; { Restore register }
  1459. END;
  1460. END;
  1461. {$ENDIF}
  1462. {$ENDIF}
  1463. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  1464. CONST vk_Scroll = $91; { Borland forgot this! }
  1465. VAR B: Byte;
  1466. BEGIN
  1467. B := 0; { Clear all masks }
  1468. If (GetKeyState(vk_Shift) AND $80 <> 0) Then
  1469. B := B OR kbBothShifts; { Set both shifts }
  1470. If (GetKeyState(vk_Control) AND $80 <> 0) Then
  1471. B := B OR kbCtrlShift; { Set control mask }
  1472. If (GetKeyState(vk_Menu) AND $80 <> 0) Then
  1473. B := B OR kbAltShift; { Set alt mask }
  1474. If (GetKeyState(vk_Scroll) AND $81 <> 0) Then
  1475. B := B OR kbScrollState; { Set scroll lock mask }
  1476. If (GetKeyState(vk_NumLock) AND $81 <> 0) Then
  1477. B := B OR kbNumState; { Set number lock mask }
  1478. If (GetKeyState(vk_Capital) AND $81 <> 0) Then
  1479. B := B OR kbCapsState; { Set caps lock mask }
  1480. If (GetKeyState(vk_Insert) AND $81 <> 0) Then
  1481. B := B OR kbInsState; { Set insert mask }
  1482. GetShiftState := B; { Return masks }
  1483. END;
  1484. {$ENDIF}
  1485. {$IFDEF OS_OS2} { OS2 CODE }
  1486. VAR Key: KbdInfo;
  1487. BEGIN
  1488. Key.cb := SizeOf(Key); { Keyboard size }
  1489. If KbdGetStatus(Key, 0) = 0 Then { Get key status }
  1490. GetShiftState := Key.fsState Else { Return shift state }
  1491. GetShiftState := 0; { Failed so return 0 }
  1492. END;
  1493. {$ENDIF}
  1494. {---------------------------------------------------------------------------}
  1495. { GetKeyEvent -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 30Jul99 LdB }
  1496. {---------------------------------------------------------------------------}
  1497. PROCEDURE GetKeyEvent (Var Event: TEvent);
  1498. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  1499. ASSEMBLER;
  1500. {$IFDEF ASM_BP} { BP COMPATABLE ASM }
  1501. ASM
  1502. MOV AH, $1; { Set function id }
  1503. PUSH BP; { Safety!! save reg }
  1504. INT $16; { Check for keypress }
  1505. POP BP; { Restore register }
  1506. MOV AX, $0; { Zero register AX }
  1507. MOV BX, AX; { Zero register BX }
  1508. JZ @@Exit3; { No keypress jump }
  1509. MOV AH, $00; { Set function id }
  1510. PUSH BP; { Safety!! save reg }
  1511. INT $16; { Read the key }
  1512. POP BP; { Restore register }
  1513. XCHG AX, BX; { Exchange registers }
  1514. MOV AX, evKeyDown; { Set keydown event }
  1515. @@Exit3:
  1516. LES DI, Event; { ES:DI -> Event }
  1517. MOV ES:[DI].TEvent.What, AX; { Store event mask }
  1518. MOV ES:[DI].TEvent.KeyCode, BX; { Store key code }
  1519. END;
  1520. {$ENDIF}
  1521. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  1522. ASM
  1523. MOVB $1, %AH; { Set function id }
  1524. PUSHL %EBP; { Save register }
  1525. INT $0x16; { Check for keypress }
  1526. POPL %EBP; { Restore register }
  1527. MOVW $0x0, %AX; { Zero register AX }
  1528. MOVW %AX, %BX; { Zero register BX }
  1529. JZ .L_Exit3; { No keypress jump }
  1530. MOVB $0, %AH; { Set function id }
  1531. PUSHL %EBP; { Save register }
  1532. INT $0x16; { Read the key }
  1533. POPL %EBP; { Restore register }
  1534. XCHGW %BX, %AX; { Exchange registers }
  1535. MOVW $0x10, %AX; { Set keydown event }
  1536. .L_Exit3:
  1537. MOVL Event, %EDI; { EDI -> Event }
  1538. CLD;
  1539. STOSW; { Store event mask }
  1540. XCHGW %BX, %AX; { Transfer key code }
  1541. STOSW; { Store key code }
  1542. END;
  1543. {$ENDIF}
  1544. {$ENDIF}
  1545. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  1546. CONST NumPos: Byte = 0; Numeric: Byte = 0;
  1547. VAR Handled: Boolean; B: Byte; Msg: TMsg;
  1548. BEGIN
  1549. Event.What := evNothing; { Preset no event }
  1550. {$IFDEF PPC_FPC} { FPC WINDOWS COMPILER }
  1551. If (PeekMessage(@Msg, 0, 0, WM_MouseFirst-1, pm_Remove)
  1552. OR PeekMessage(@Msg, 0, WM_MouseLast+1, $FFFF, pm_Remove))
  1553. {$ELSE} { OTHER COMPILERS }
  1554. If (PeekMessage(Msg, 0, 0, WM_MouseFirst-1, pm_Remove)
  1555. OR PeekMessage(Msg, 0, WM_MouseLast+1, $FFFF, pm_Remove))
  1556. {$ENDIF}
  1557. Then Begin { Non mouse message }
  1558. Handled := False; { Preset not handled }
  1559. Case Msg.Message Of
  1560. WM_Char: Begin { CHARACTER KEY }
  1561. NumPos := 0; { Zero number position }
  1562. Event.CharCode := Char(Msg.wParam); { Transfer character }
  1563. If (Event.CharCode > #127) Then
  1564. Event.CharCode := Chr(WinAsciiToIntAscii[
  1565. Ord(Event.CharCode)]); { Convert to ascii }
  1566. Event.ScanCode := Lo(HiWord(Msg.lParam)); { Transfer scan code }
  1567. If (Event.CharCode <> #0) Then Begin { If valid key then }
  1568. Event.What := evKeyDown; { Key down event }
  1569. Handled := True; { Message was handled }
  1570. If (Event.KeyCode = kbTab) AND { Tab key is special }
  1571. (GetShiftState AND kbBothShifts <> 0) Then { Check shift state }
  1572. Event.KeyCode := kbShiftTab; { If set make shifttab }
  1573. End;
  1574. End;
  1575. WM_SysKeyDown: Begin { SYSTEM KEY DOWN }
  1576. If (NumPos > 0) Then Begin { Numerics entry op }
  1577. Case Msg.wParam Of
  1578. VK_Insert: B := 0; { Key value = 0 }
  1579. VK_End: B := 1; { Key value = 1 }
  1580. VK_Down: B := 2; { Key value = 2 }
  1581. VK_Next: B := 3; { Key value = 3 }
  1582. VK_Left: B := 4; { Key value = 4 }
  1583. VK_Clear: B := 5; { Key value = 5 }
  1584. VK_Right: B := 6; { Key value = 6 }
  1585. VK_Home: B := 7; { Key value = 7 }
  1586. VK_Up: B := 8; { Key value = 8 }
  1587. VK_Prior: B := 9; { Key value = 9 }
  1588. VK_NumPad0..VK_NumPad9: B := Msg.wParam
  1589. - $60; { Numbic key pad }
  1590. Else NumPos := 0; { Invalid key }
  1591. End;
  1592. If ((NumPos > 0) AND (NumPos < 4)) AND { Valid position }
  1593. ((B >= $0) AND (B <= $9)) Then Begin { Valid key }
  1594. Numeric := Numeric*10 + B; { Adjust numeric }
  1595. Inc(NumPos); { Next position }
  1596. If (NumPos = 4) Then Begin { We have three keys }
  1597. Event.What := evKeyDown; { Set keydown event }
  1598. Event.CharCode := Chr(Numeric); { Transfer code }
  1599. NumPos := 0; { Zero number position }
  1600. End;
  1601. Handled := True; { Message was handled }
  1602. End Else NumPos := 0; { Zero number position }
  1603. End;
  1604. If (Msg.WParam = vk_Menu) Then Begin { ALT key down }
  1605. Numeric := 0; { Zero numeric }
  1606. NumPos := 1; { Set to start point }
  1607. Handled := True; { Message was handled }
  1608. End;
  1609. If NOT Handled Then Begin { Key press not handled }
  1610. If (Lo(Msg.wParam) < 128) Then Begin { Ignore if above 128 }
  1611. If (Msg.wParam = vk_F10) Then Begin { F10 reports oddly }
  1612. If (GetKeyState(vk_Shift) AND $80 <> 0)
  1613. Then Event.KeyCode := kbShiftF10 Else{ Shift F10 }
  1614. If (GetKeyState(vk_Menu) AND $80 <> 0)
  1615. Then Event.KeyCode := kbAltF10 Else { Alt F10 }
  1616. If (GetKeyState(vk_Control) AND $80 <> 0)
  1617. Then Event.KeyCode := kbCtrlF10 { Ctrl F10 }
  1618. Else Event.KeyCode := kbF10; { Normal F10 }
  1619. End Else Event.KeyCode :=
  1620. AltVirtualToAscii[Lo(Msg.wParam)]; { Convert key code }
  1621. End Else Event.KeyCode := 0; { Clear Event.keycode }
  1622. If (Event.KeyCode <> 0) Then Begin { If valid key then }
  1623. Event.What := evKeyDown; { Key down event }
  1624. Handled := True; { Message was handled }
  1625. End;
  1626. End;
  1627. End;
  1628. WM_KeyDown: Begin { ARROWS/F1..F12 KEYS }
  1629. If (((Msg.WParam >= Vk_F1) AND (Msg.WParam <= Vk_F12)) OR
  1630. ((Msg.WParam >= Vk_Prior) AND (Msg.WParam <= Vk_Delete)))
  1631. Then Begin { Special key press }
  1632. Event.CharCode := #0; { Clear char code }
  1633. Event.ScanCode := Lo(HiWord(Msg.LParam)); { Create scan code }
  1634. If (GetKeyState(vk_Shift) AND $80 <> 0)
  1635. Then Begin { Shift key down }
  1636. Case Msg.wParam Of
  1637. vk_F1..vk_F9: Event.KeyCode :=
  1638. Event.KeyCode + $1900; { Shift F1..F9 keys }
  1639. vk_F11: Event.KeyCode := kbShiftF11; { Shift F11 key }
  1640. vk_F12: Event.KeyCode := kbShiftF12; { Shift F12 key }
  1641. End;
  1642. End Else If (GetKeyState(vk_Control) AND $80 <> 0)
  1643. Then Begin { Control key down }
  1644. Case Msg.wParam Of
  1645. vk_F1..vk_F9: Event.KeyCode :=
  1646. Event.KeyCode + $2300; { Ctrl F1..F9 keys }
  1647. vk_F11: Event.KeyCode := kbCtrlF11; { Ctrl F11 key }
  1648. vk_F12: Event.KeyCode := kbCtrlF12; { Ctrl F12 key }
  1649. End;
  1650. End;
  1651. If (Event.KeyCode <> 0) Then Begin { If valid key then }
  1652. Event.What := evKeyDown; { Key down event }
  1653. Handled := True; { Message was handled }
  1654. End;
  1655. End;
  1656. NumPos := 0; { Zero number position }
  1657. End;
  1658. End;
  1659. If NOT Handled Then Begin { Check we did not handle }
  1660. TranslateMessage(Msg); { Translate message }
  1661. DispatchMessage(Msg); { Dispatch message }
  1662. End;
  1663. End;
  1664. END;
  1665. {$ENDIF}
  1666. {$IFDEF OS_OS2} { OS2 CODE }
  1667. VAR Msg: QMsg;
  1668. BEGIN
  1669. Event.What := evNothing; { Preset no event }
  1670. If (WinPeekMsg(Anchor, Msg, 0, 0, WM_MouseFirst-1, pm_Remove)
  1671. OR WinPeekMsg(Anchor, Msg, 0, WM_MouseLast+1, $FFFFFFFF, pm_Remove))
  1672. Then Begin { Check for message }
  1673. If (Msg.Msg = WM_Char) AND { Character message }
  1674. (Msg.Mp1 AND KC_KeyUp <> 0) AND { Key released }
  1675. (Msg.Mp1 AND KC_Composite = 0) { Not composite key }
  1676. Then Begin
  1677. If (Short1FromMP(Msg.Mp1) AND KC_ScanCode <> 0 )
  1678. Then Begin
  1679. Event.ScanCode := Ord(Char4FromMP(Msg.Mp1)); { Return scan code }
  1680. Event.CharCode := Char1FromMP(Msg.Mp2); { Return char code }
  1681. If (Event.CharCode = Chr($E0)) Then Begin
  1682. Event.CharCode := #0;
  1683. Event.ScanCode := Byte(Char2FromMP(Msg.Mp2));
  1684. End;
  1685. If (Event.KeyCode <> 0) Then
  1686. Event.What := evKeyDown; { Key down event }
  1687. End;
  1688. End;
  1689. If (Event.What = evNothing) Then { Event not handled }
  1690. WinDispatchMsg(Anchor, Msg); { Disptach message }
  1691. End;
  1692. END;
  1693. {$ENDIF}
  1694. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  1695. { MOUSE CONTROL ROUTINES }
  1696. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  1697. {---------------------------------------------------------------------------}
  1698. { HideMouse -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 30Jun98 LdB }
  1699. {---------------------------------------------------------------------------}
  1700. PROCEDURE HideMouse;
  1701. BEGIN
  1702. If (HideCount = 0) Then Begin { Is mouse hidden yet? }
  1703. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  1704. HideMouseCursor; { Hide mouse cursor }
  1705. {$ENDIF}
  1706. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  1707. ShowCursor(False); { Hide mouse cursor }
  1708. {$ENDIF}
  1709. {$IFDEF OS_OS2} { OS2 CODE }
  1710. If (AppWindow <> 0) Then { Window valid }
  1711. WinShowCursor(AppWindow, False); { Hide mouse cursor }
  1712. {$ENDIF}
  1713. End;
  1714. Inc(HideCount); { Inc hide count }
  1715. END;
  1716. {---------------------------------------------------------------------------}
  1717. { ShowMouse -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 30Jun98 LdB }
  1718. {---------------------------------------------------------------------------}
  1719. PROCEDURE ShowMouse;
  1720. BEGIN
  1721. Dec(HideCount); { Dec hide count }
  1722. If (HideCount = 0) Then Begin { Is mouse visible? }
  1723. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  1724. ShowMouseCursor; { Show mouse cursor }
  1725. {$ENDIF}
  1726. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  1727. ShowCursor(True); { Show mouse cursor }
  1728. {$ENDIF}
  1729. {$IFDEF OS_OS2} { OS2 CODE }
  1730. If (AppWindow <> 0) Then { Window valid }
  1731. WinShowCursor(AppWindow, True); { Show mouse cursor }
  1732. {$ENDIF}
  1733. End;
  1734. END;
  1735. {---------------------------------------------------------------------------}
  1736. { GetMouseEvent -> Platforms DOS/DPMI/WINDOWS/OS2 - Updated 09Sep98 LdB }
  1737. {---------------------------------------------------------------------------}
  1738. PROCEDURE GetMouseEvent (Var Event: TEvent);
  1739. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  1740. ASSEMBLER;
  1741. {$IFDEF ASM_BP} { BP COMPATABLE ASM }
  1742. ASM
  1743. CMP MouseEvents, 0; { Any mouse events }
  1744. JNZ @@MouseOk; { Check mouse active }
  1745. JMP @@NoEventExit; { Mouse not active }
  1746. @@MouseOk:
  1747. CLI; { Disable interrupts }
  1748. CMP EventCount, 0; { Check event count }
  1749. JNE @@MouseEventInQueue; { If > 0 event avail }
  1750. MOV BL, MouseButtons; { Fetch mouse buttons }
  1751. MOV CX, MouseWhere.Word[0]; { Fetch mouse where.x }
  1752. MOV DX, MouseWhere.Word[2]; { Fetch mouse where.y }
  1753. MOV ES, Seg0040; { DOS DAT SEG }
  1754. MOV DI, ES:Ticks; { Fetch current time }
  1755. JMP @@NextMsgReady; { Now process }
  1756. @@MouseEventInQueue:
  1757. MOV SI, WORD PTR EventQHead; { Event queue head }
  1758. CLD; { Direction forward }
  1759. LODSW; { Fetch word 1 }
  1760. XCHG AX, DI; { Set timer ticks }
  1761. LODSW; { Fetch word 2 }
  1762. XCHG AX, BX; { Set button masks }
  1763. LODSW; { Fetch word 3 }
  1764. XCHG AX, CX; { Set mouse x position }
  1765. LODSW; { Fetch word 4 }
  1766. XCHG AX, DX; { Set mouse y position }
  1767. CMP SI, OFFSET EventQLast; { Check if roll needed }
  1768. JNE @@NoRoll;
  1769. MOV SI, OFFSET EventQueue; { Roll back to start }
  1770. @@NoRoll:
  1771. MOV WORD PTR EventQHead, SI; { Update queue head }
  1772. DEC EventCount; { One event cleared }
  1773. @@NextMsgReady:
  1774. STI; { Enable interrupts }
  1775. CMP MouseReverse, 0; { Check mouse reversed }
  1776. JE @@MouseNormal;
  1777. MOV BH, BL; { Transfer button mask }
  1778. AND BH, 3; { Clear others masks }
  1779. JE @@MouseNormal; { Neither set exit }
  1780. CMP BH, 3; { Check not all set }
  1781. JE @@MouseNormal; { Both set exit }
  1782. XOR BL, 3; { Invert button masks }
  1783. @@MouseNormal:
  1784. MOV BH, [LastDouble]; { Load last double }
  1785. MOV AL, [LastButtons]; { Load last buttons }
  1786. CMP AL, BL; { Are buttons same? }
  1787. JE @@SameButtonsDown;
  1788. OR AL, AL; { Any last buttons? }
  1789. JE @@ButtonsDown;
  1790. OR BL, BL; { Any buttons down? }
  1791. JE @@MouseUp;
  1792. MOV BL, AL; { Transfer new buttons }
  1793. @@SameButtonsDown:
  1794. CMP CX, [LastWhereX]; { Mouse moved from x }
  1795. JNE @@MouseMove;
  1796. CMP DX, [LastWhereY]; { Mouse moved from y }
  1797. JNE @@MouseMove;
  1798. OR BL, BL; { Any buttons pressed? }
  1799. JE @@NoButtonsDown;
  1800. MOV AX, DI; { Current tick count }
  1801. SUB AX, [AutoTicks]; { Subtract last count }
  1802. CMP AX, [AutoDelay]; { Greater than delay? }
  1803. JAE @@MouseAuto; { Mouse auto event }
  1804. @@NoButtonsDown:
  1805. JMP @@NoEventExit; { No event exit }
  1806. @@ButtonsDown:
  1807. MOV BH, 0; { Preset no dbl click }
  1808. CMP BL, [DownButtons]; { Check to last down }
  1809. JNE @@MouseDown;
  1810. CMP CX, [DownWhereX]; { Check x position }
  1811. JNE @@MouseDown;
  1812. CMP DX, [DownWhereY]; { Check y position }
  1813. JNE @@MouseDown;
  1814. MOV AX, DI; { Transfer tick count }
  1815. SUB AX, [DownTicks]; { Sub last down count }
  1816. CMP AX, [DoubleDelay]; { Greater than delay? }
  1817. JAE @@MouseDown;
  1818. MOV BH, 1; { Double click }
  1819. @@MouseDown:
  1820. MOV [DownButtons], BL; { Hold down buttons }
  1821. MOV [DownWhereX], CX; { Hold x down point }
  1822. MOV [DownWhereY], DX; { Hold y down point }
  1823. MOV [DownTicks], DI; { Hold tick value }
  1824. MOV [AutoTicks], DI; { Hold tick value }
  1825. MOV AX, [RepeatDelay]; { Load delay count }
  1826. MOV [AutoDelay], AX; { Set delay time }
  1827. MOV AX, evMouseDown; { Mouse down event }
  1828. JMP @@UpdateValues; { Update, svae & exit }
  1829. @@MouseUp:
  1830. MOV AX, evMouseUp; { Mouse button up }
  1831. JMP @@UpdateValues; { Update, save & exit }
  1832. @@MouseMove:
  1833. MOV AX, evMouseMove; { Mouse has moved }
  1834. JMP @@UpdateValues; { Update, save & exit }
  1835. @@MouseAuto:
  1836. MOV AX, evMouseAuto; { Mouse auto event }
  1837. MOV [AutoTicks], DI; { Reset auto ticks }
  1838. MOV [AutoDelay], 1; { Reset delay count }
  1839. @@UpdateValues:
  1840. MOV [LastButtons], BL; { Save last buttons }
  1841. MOV [LastDouble], BH; { Save double state }
  1842. MOV [LastWhereX], CX; { Save x position }
  1843. MOV [LastWhereY], DX; { Save y position }
  1844. JMP @@StoreAndExit; { Now store and exit }
  1845. @@NoEventExit:
  1846. XOR AX, AX; { Clear register }
  1847. MOV BX, AX; { Clear register }
  1848. MOV CX, AX; { Clear register }
  1849. MOV DX, AX; { Clear register }
  1850. @@StoreAndExit:
  1851. LES DI, Event; { Address of event }
  1852. CLD; { Set direction fwd }
  1853. STOSW; { Save 1st word }
  1854. XCHG AX, BX; { Transfer register }
  1855. STOSW; { Save 2nd word }
  1856. XCHG AX, CX; { Transfer register }
  1857. STOSW; { Save 3rd word }
  1858. XCHG AX, DX; { Transfer register }
  1859. STOSW; { Save 4th word }
  1860. END;
  1861. {$ENDIF}
  1862. {$IFDEF ASM_FPC} { FPC COMPATABLE ASM }
  1863. ASM
  1864. CMPB $0, MOUSEEVENTS; { Any mouse events }
  1865. JNZ .L_MouseOk; { Check mouse active }
  1866. JMP .L_NoEventExit; { Mouse not active }
  1867. .L_MouseOk:
  1868. CLI;
  1869. CMPW $0, EVENTCOUNT; { Check event count }
  1870. JNE .L_MouseEventInQueue; { If > 0 event avail }
  1871. MOVB MOUSEBUTTONS, %BL; { Fetch mouse buttons }
  1872. MOVW MOUSEWHERE, %CX; { Fetch mouse where.x }
  1873. MOVW MOUSEWHERE+2, %DX; { Fetch mouse where.y }
  1874. PUSH %ES; { Save segment }
  1875. MOVW $0x40, %AX; { Fetch DOS segment }
  1876. MOVW %AX, %ES; { Transfer to segment }
  1877. MOVL $0x6C, %EDI; { Tick address }
  1878. MOVW %ES:(%EDI), %DI; { Fetch dos tick count }
  1879. POP %ES; { Recover segment }
  1880. JMP .L_NextMsgReady; { Now process }
  1881. .L_MouseEventInQueue:
  1882. MOVL EVENTQHEAD, %ESI; { Event queue head }
  1883. CLD; { Direction forward }
  1884. LODSW; { Fetch word 1 }
  1885. XCHGW %DI, %AX; { Set timer ticks }
  1886. LODSW; { Fetch word 2 }
  1887. XCHGW %BX, %AX; { Set button masks }
  1888. LODSW; { Fetch word 3 }
  1889. XCHGW %CX, %AX; { Set mouse x position }
  1890. LODSW; { Fetch word 4 }
  1891. XCHGW %DX, %AX; { Set mouse y position }
  1892. LEAL EVENTQLAST, %EAX; { Address of roll pt }
  1893. CMPL %EAX, %ESI; { Check if roll needed }
  1894. JNE .L_NoHeadRoll;
  1895. LEAL EVENTQUEUE, %ESI; { Roll back to start }
  1896. .L_NoHeadRoll:
  1897. MOVL %ESI, EVENTQHEAD; { Update queue head }
  1898. DECW EVENTCOUNT; { One event cleared }
  1899. .L_NextMsgReady:
  1900. STI; { Enable interrupts }
  1901. CMPB $0, MOUSEREVERSE; { Check mouse reversed }
  1902. JE .L_MouseNormal;
  1903. MOVB %BL, %BH; { Transfer button mask }
  1904. ANDB $3, %BH; { Clear others masks }
  1905. JE .L_MouseNormal; { Neither set exit }
  1906. CMPB $3, %BH; { Check not all set }
  1907. JE .L_MouseNormal; { Both set exit }
  1908. XORB $3, %BL; { Invert button masks }
  1909. .L_MouseNormal:
  1910. MOVB LASTDOUBLE, %BH; { Load last double }
  1911. MOVB LASTBUTTONS, %AL; { Load last buttons }
  1912. CMPB %BL, %AL; { Are buttons same? }
  1913. JE .L_SameButtonsDown;
  1914. ORB %AL, %AL; { Any last buttons? }
  1915. JE .L_ButtonsDown;
  1916. ORB %BL, %BL; { Any buttons down? }
  1917. JE .L_MouseUp;
  1918. MOVB %AL, %BL; { Transfer new buttons }
  1919. .L_SameButtonsDown:
  1920. CMPW LASTWHEREX, %CX; { Mouse moved from x }
  1921. JNE .L_MouseMove;
  1922. CMPW LASTWHEREY, %DX; { Mouse moved from y }
  1923. JNE .L_MouseMove;
  1924. ORB %BL, %BL; { Any buttons pressed? }
  1925. JE .L_NoButtonsDown;
  1926. MOVW %DI, %AX; { Current tick count }
  1927. SUBW AUTOTICKS, %AX; { Subtract last count }
  1928. CMPW AUTODELAY, %AX; { Greater than delay? }
  1929. JAE .L_MouseAuto; { Mouse auto event }
  1930. .L_NoButtonsDown:
  1931. JMP .L_NoEventExit; { No event exit }
  1932. .L_ButtonsDown:
  1933. MOVB $0, %BH; { Preset no dbl click }
  1934. CMPB DOWNBUTTONS, %BL; { Check to last down }
  1935. JNE .L_MouseDown;
  1936. CMPW DOWNWHEREX, %CX; { Check x position }
  1937. JNE .L_MouseDown;
  1938. CMPW DOWNWHEREY, %DX; { Check y position }
  1939. JNE .L_MouseDown;
  1940. MOVW %DI, %AX; { Transfer tick count }
  1941. SUBW DOWNTICKS, %AX; { Sub last down count }
  1942. CMPW DOUBLEDELAY, %AX; { Greater than delay? }
  1943. JAE .L_MouseDown;
  1944. MOVB $1, %BH; { Double click }
  1945. .L_MouseDown:
  1946. MOVB %BL, DOWNBUTTONS; { Hold down buttons }
  1947. MOVW %CX, DOWNWHEREX; { Hold x down point }
  1948. MOVW %DX, DOWNWHEREY; { Hold y down point }
  1949. MOVW %DI, DOWNTICKS; { Hold tick value }
  1950. MOVW %DI, AUTOTICKS; { Hold tick value }
  1951. MOVW REPEATDELAY, %AX; { Load delay count }
  1952. MOVW %AX, AUTODELAY; { Set delay time }
  1953. MOVW $1, %AX; { Mouse down event }
  1954. JMP .L_UpdateValues; { Update, svae & exit }
  1955. .L_MouseUp:
  1956. MOVW $2, %AX; { Mouse button up }
  1957. JMP .L_UpdateValues; { Update, save & exit }
  1958. .L_MouseMove:
  1959. MOVW $4, %AX; { Mouse has moved }
  1960. JMP .L_UpdateValues; { Update, save & exit }
  1961. .L_MouseAuto:
  1962. MOVW $8, %AX; { Mouse auto event }
  1963. MOVW %DI, AUTOTICKS; { Reset auto ticks }
  1964. MOVW $1, AUTODELAY; { Reset delay count }
  1965. .L_UpdateValues:
  1966. MOVB %BL, LASTBUTTONS; { Save last buttons }
  1967. MOVB %BH, LASTDOUBLE; { Save double state }
  1968. MOVW %CX, LASTWHEREX; { Save x position }
  1969. MOVW %DX, LASTWHEREY; { Save y position }
  1970. JMP .L_StoreAndExit; { Now store and exit }
  1971. .L_NoEventExit:
  1972. XORW %AX, %AX; { Clear register }
  1973. MOVW %AX, %BX; { Clear register }
  1974. MOVW %AX, %CX; { Clear register }
  1975. MOVW %AX, %DX; { Clear register }
  1976. .L_StoreAndExit:
  1977. MOVL Event, %EDI; { Adress of event }
  1978. CLD; { Set direction fwd }
  1979. STOSW; { Save 1st word }
  1980. XCHGW %BX, %AX; { Transfer register }
  1981. STOSW; { Save 2nd word }
  1982. XCHGW %CX, %AX; { Transfer register }
  1983. STOSW; { Save 3rd word }
  1984. XCHGW %DX, %AX; { Transfer register }
  1985. STOSW; { Save 4th word }
  1986. END;
  1987. {$ENDIF}
  1988. {$ENDIF}
  1989. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  1990. VAR Msg: TMsg;
  1991. BEGIN
  1992. Event.What := evNothing; { Preset no event }
  1993. {$IFDEF PPC_FPC} { FPC WINDOWS COMPILER }
  1994. If PeekMessage(@Msg, 0, WM_MouseFirst,
  1995. WM_MouseLast, pm_Remove) Then Begin { Fetch mouse message }
  1996. {$ELSE} { OTHER COMPILERS }
  1997. If PeekMessage(Msg, 0, WM_MouseFirst,
  1998. WM_MouseLast, pm_Remove) Then Begin { Fetch mouse message }
  1999. {$ENDIF}
  2000. TranslateMessage(Msg); { Translate message }
  2001. DispatchMessage(Msg); { Dispatch message }
  2002. End;
  2003. END;
  2004. {$ENDIF}
  2005. {$IFDEF OS_OS2} { OS2 CODE }
  2006. VAR Msg: QMsg;
  2007. BEGIN
  2008. Event.What := evNothing; { Preset no event }
  2009. If WinPeekMsg(Anchor, Msg, 0, WM_MouseFirst,
  2010. WM_MouseLast, pm_Remove) Then Begin { Fetch mouse message }
  2011. WinDispatchMsg(Anchor, Msg); { Dispatch message }
  2012. End;
  2013. END;
  2014. {$ENDIF}
  2015. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  2016. { EVENT HANDLER CONTROL ROUTINES }
  2017. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  2018. {---------------------------------------------------------------------------}
  2019. { InitEvents -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 07Sep99 LdB }
  2020. {---------------------------------------------------------------------------}
  2021. PROCEDURE InitEvents;
  2022. BEGIN
  2023. If (ButtonCount <> 0) Then Begin { Mouse is available }
  2024. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  2025. EventQHead := @EventQueue; { Initialize head }
  2026. EventQtail := @EventQueue; { Initialize tail }
  2027. LastDouble := False; { Clear last double }
  2028. LastButtons := 0; { Clear last buttons }
  2029. DownButtons := 0; { Clear down buttons }
  2030. HookMouse; { Hook the mouse }
  2031. GetMousePosition(MouseWhere.X, MouseWhere.Y); { Get mouse position }
  2032. LastWhereX := MouseWhere.X; { Set last x position }
  2033. LastWhereY := MouseWhere.Y; { Set last y position }
  2034. MouseEvents := True; { Set initialized flag }
  2035. ShowMouseCursor; { Show the mouse }
  2036. {$ENDIF}
  2037. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  2038. MouseEvents := True; { Set initialized flag }
  2039. {$ENDIF}
  2040. {$IFDEF OS_OS2} { OS2 CODE }
  2041. If (Anchor=0) Then Anchor := WinInitialize(0); { Create anchor block }
  2042. If (MsgQue = 0) AND (Anchor <> 0) Then
  2043. MsgQue := WinCreateMsgQueue(Anchor, 0); { Initialize queue }
  2044. If (MsgQue = 0) Then Halt(254); { Check queue created }
  2045. MouseEvents := True; { Set initialized flag }
  2046. {$ENDIF}
  2047. End;
  2048. END;
  2049. {---------------------------------------------------------------------------}
  2050. { DoneEvents -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 30Jul99 LdB }
  2051. {---------------------------------------------------------------------------}
  2052. PROCEDURE DoneEvents;
  2053. BEGIN
  2054. If MouseEvents Then Begin { Initialized check }
  2055. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  2056. HideMouseCursor; { Hide the mouse }
  2057. MouseEvents := False; { Clear event flag }
  2058. UnHookMouse; { Unhook the mouse }
  2059. {$ENDIF}
  2060. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  2061. MouseEvents := False; { Clr initialized flag }
  2062. {$ENDIF}
  2063. {$IFDEF OS_OS2} { OS2 CODE }
  2064. If (MsgQue <> 0) Then WinDestroyMsgQueue(MsgQue);{ Destroy msg queue }
  2065. If (Anchor <> 0) Then WinTerminate(Anchor); { Destroy anchor block }
  2066. MsgQue := 0; { Zero msg queue handle }
  2067. Anchor := 0; { Zero anchor block }
  2068. MouseEvents := False; { Clr initialized flag }
  2069. {$ENDIF}
  2070. End;
  2071. END;
  2072. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  2073. { VIDEO CONTROL ROUTINES }
  2074. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  2075. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  2076. {$IFDEF PPC_FPC} { FPC COMPILER ONLY }
  2077. { ******************************* REMARK ****************************** }
  2078. { This is purely temporary for FPC because the Graph is SuperVGA you }
  2079. { have no mouse pointer on screen because the mouse drivers don't go }
  2080. { up to supporting SVGA modes. This simply makes a cross hair so you }
  2081. { can see the mouse for now..will be fixed soon. }
  2082. { ****************************** END REMARK *** Leon de Boer, 04Nov99 * }
  2083. VAR LastX, LastY: Integer;
  2084. PROCEDURE ShowTheMouse; FAR;
  2085. BEGIN
  2086. If (MouseEvents = True) AND (HideCount = 0) { Mouse visible }
  2087. Then Begin
  2088. SetWriteMode(XORPut); { XOR write mode }
  2089. SetColor(15); { Set color to white }
  2090. Line(LastX-5, LastY, LastX+5, LastY); { Remove horz line }
  2091. Line(LastX, LastY-5, LastX, LastY+5); { Remove vert line }
  2092. LastX := MouseWhere.X; { Update x position }
  2093. LastY := MouseWHere.Y; { Update y position }
  2094. Line(LastX-5, LastY, LastX+5, LastY); { Draw horz line }
  2095. Line(LastX, LastY-5, LastX, LastY+5); { Draw vert line }
  2096. SetWriteMode(NormalPut); { Write mode to normal }
  2097. End;
  2098. END;
  2099. {$ENDIF}
  2100. {$ENDIF}
  2101. {---------------------------------------------------------------------------}
  2102. { InitVideo -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 26Nov99 LdB }
  2103. {---------------------------------------------------------------------------}
  2104. PROCEDURE InitVideo;
  2105. VAR {$IFDEF OS_DOS} I, J: Integer; Ts: TextSettingsType; {$ENDIF}
  2106. {$IFDEF OS_WINDOWS} Dc, Mem: HDc; TempFont: TLogFont; Tm: TTextmetric; {$ENDIF}
  2107. {$IFDEF OS_OS2} Ts, Fs: Integer; Ps: HPs; Tm: FontMetrics; {$ENDIF}
  2108. BEGIN
  2109. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  2110. I := Detect; { Detect video card }
  2111. J := 0; { Zero select mode }
  2112. InitGraph(I, J, ''); { Initialize graphics }
  2113. I := GetMaxX; { Fetch max x size }
  2114. J := GetMaxY; { Fetch max y size }
  2115. {$IFDEF PPC_FPC} { FPC DOS COMPILER }
  2116. ASM
  2117. MOVW $7, %AX; { Set function id }
  2118. MOVW $0, %CX; { Clear register }
  2119. MOVW I, %DX; { Maximum x size }
  2120. INT $0x33; { Set mouse x movement }
  2121. MOVW $8, %AX; { Set function id }
  2122. MOVW $0, %CX; { Clear register }
  2123. MOVW J, %DX; { Maximum y size }
  2124. INT $0x33; { Set mouse y movement }
  2125. END;
  2126. Lock_Code(Pointer(@ShowTheMouse), 400); { Lock cursor code }
  2127. MouseMoveProc := ShowTheMouse; { Set move function }
  2128. ShowMouseProc := ShowTheMouse; { Set show function }
  2129. HideMouseProc := ShowTheMouse; { Set hide function }
  2130. {$ELSE} { OTHER DOS COMPILERS }
  2131. ASM
  2132. MOV AX, 7; { Set function id }
  2133. XOR CX, CX; { Clear register }
  2134. MOV DX, I; { Maximum x size }
  2135. INT 33H; { Set mouse x movement }
  2136. MOV AX, 8; { Set function id }
  2137. XOR CX, CX; { Clear register }
  2138. MOV DX, J; { Maximum y size }
  2139. INT 33H; { Set mouse y movement }
  2140. END;
  2141. {$ENDIF}
  2142. SysScreenWidth := GetMaxX+1; { Max screen width }
  2143. SysScreenHeight := GetMaxY+1; { Max screen height }
  2144. If (DefFontHeight = 0) Then { Font height not set }
  2145. J := SysScreenHeight DIV DefLineNum { Approx font height }
  2146. Else J := DefFontHeight; { Use set font height }
  2147. I := J DIV (TextHeight('H')+4); { Approx magnification }
  2148. If (I < 1) Then I := 1; { Must be 1 or above }
  2149. GetTextSettings(Ts); { Get text style }
  2150. SetTextStyle(Ts.Font, Ts.Direction, I); { Set new font settings }
  2151. SysFontWidth := TextWidth('H'); { Transfer font width }
  2152. SysFontHeight := TextHeight('H')+4; { Transfer font height }
  2153. {$ENDIF}
  2154. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  2155. SysScreenWidth := GetSystemMetrics(
  2156. SM_CXFullScreen)-GetSystemMetrics(SM_CXFrame); { Max screen width }
  2157. SysScreenHeight := GetSystemMetrics(
  2158. SM_CYFullScreen); { Max screen height }
  2159. With TempFont Do Begin
  2160. If (DefFontHeight = 0) Then Begin { Font height not set }
  2161. lfHeight := SysScreenHeight DIV DefLineNum; { Best guess height }
  2162. End Else lfHeight := -DefFontHeight; { Specific font height }
  2163. lfWidth := 0; { No specific width }
  2164. lfEscapement := 0; { No specifics }
  2165. lfOrientation := 0; { Normal orientation }
  2166. lfWeight := DefFontWeight; { Default font weight }
  2167. lfItalic := 0; { No italics }
  2168. lfUnderline := 0; { No underlines }
  2169. lfStrikeOut := 0; { No strikeouts }
  2170. lfCharSet := ANSI_CharSet; { ANSI font set }
  2171. lfOutPrecision := Out_Default_Precis; { Default precision out }
  2172. lfClipPrecision := Clip_Default_Precis; { Default clip precision }
  2173. lfQuality := Proof_Quality; { Proof quality }
  2174. lfPitchAndFamily:= Variable_Pitch OR
  2175. Fixed_Pitch; { Either fitch format }
  2176. FillChar(lfFaceName, SizeOf(lfFaceName), #0); { Clear memory area }
  2177. Move(DefFontStyle[1], lfFacename,
  2178. Length(DefFontStyle)); { Transfer style name }
  2179. End;
  2180. DefGFVFont := CreateFontIndirect(TempFont); { Create a default font }
  2181. Dc := GetDc(0); { Get screen context }
  2182. Mem := CreateCompatibleDC(Dc); { Compatable context }
  2183. SelectObject(Mem, DefGFVFont); { Select the font }
  2184. {$IFDEF PPC_FPC} { FPC WINDOWS COMPILER }
  2185. GetTextMetrics(Mem, @Tm); { Get text metrics }
  2186. {$ELSE} { OTHER COMPILERS }
  2187. GetTextMetrics(Mem, Tm); { Get text metrics }
  2188. {$ENDIF}
  2189. SysFontWidth := Tm.tmaveCharWidth+1; { Ave char font width }
  2190. SysFontHeight := Tm.tmHeight; { Ave char font height }
  2191. DeleteDc(Mem); { Destroy context }
  2192. ReleaseDc(0, Dc); { Release context }
  2193. {$ENDIF}
  2194. {$IFDEF OS_OS2} { OS2 CODE }
  2195. Ts := WinQuerySysValue(HWND_Desktop,
  2196. SV_CYTitleBar) + 2*WinQuerySysValue(HWND_Desktop,
  2197. SV_CYSizeBorder); { Title size }
  2198. Fs := 2*WinQuerySysValue(HWND_DeskTop,
  2199. SV_CXSizeBorder); { Frame size }
  2200. SysScreenWidth := WinQuerySysValue(HWND_Desktop,
  2201. SV_CXFullScreen) - Fs; { Max screen width }
  2202. SysScreenHeight := WinQuerySysValue(HWND_Desktop,
  2203. SV_CYFullScreen) - Ts; { Max screen height }
  2204. (*With DefGFVFont Do Begin
  2205. usRecordLength := SizeOf(fAttrs); { Structure size }
  2206. fsSelection := $20; { Uses default selection }
  2207. lMatch := 0; { Does not force match }
  2208. idRegistry := 0; { Uses default registry }
  2209. usCodePage := 850; { Code-page 850 }
  2210. If (DefFontHeight = 0) Then Begin { Font height not set }
  2211. lMaxBaselineExt := SysScreenHeight DIV DefLineNum; { Best guess height }
  2212. End Else lMaxBaselineExt := DefFontHeight; { Specific font height }
  2213. lAveCharWidth := 0; { Req font default width }
  2214. fsType := 0; { Uses default type }
  2215. fsFontUse := fAttr_FontUse_Nomix; { Doesn't mix with graphics }
  2216. FillChar(szFaceName, SizeOf(szFaceName), #0); { Clear memory area }
  2217. Move(DefFontStyle[1], szFacename,
  2218. Length(DefFontStyle)); { Transfer style name }
  2219. End;*)
  2220. Ps := WinGetPS(HWND_Desktop); { Get desktop PS }
  2221. (*GpiCreateLogFont(Ps, Nil, 1, DefGFVFont);*) { Create the font }
  2222. GpiQueryFontMetrics(Ps, SizeOf(Tm), Tm); { Get text metrics }
  2223. SysFontWidth := Tm.lAveCharWidth+1; { Transfer font width }
  2224. SysFontHeight := Tm.lMaxBaselineExt; { Transfer font height }
  2225. { SysFontheight := SysScreenheight DIV DefLineNum;}
  2226. WinReleasePS(Ps); { Release desktop PS }
  2227. DefPointer := WinQuerySysPointer(HWND_DESKTOP,
  2228. SPTR_ARROW, False); { Hold default pointer }
  2229. {$ENDIF}
  2230. ScreenWidth := SysScreenWidth DIV SysFontWidth; { Calc screen width }
  2231. ScreenHeight := SysScreenHeight DIV SysFontHeight; { Calc screen height }
  2232. SysScreenWidth := ScreenWidth * SysFontWidth; { Actual width }
  2233. SysScreenHeight := ScreenHeight * SysFontHeight; { Actual height }
  2234. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  2235. Inc(SysScreenWidth, 2*GetSystemMetrics(SM_CXFrame)); { Max screen width }
  2236. Inc(SysScreenHeight, GetSystemMetrics(SM_CYCaption)
  2237. + GetSystemMetrics(SM_CYFrame)); { Max screen height }
  2238. {$ENDIF}
  2239. {$IFDEF OS_OS2} { OS2 CODE }
  2240. Inc(SysScreenWidth, Fs); { Max screen width }
  2241. Inc(SysScreenHeight, Ts); { Max screen height }
  2242. {$ENDIF}
  2243. END;
  2244. {---------------------------------------------------------------------------}
  2245. { DoneVideo -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 19May98 LdB }
  2246. {---------------------------------------------------------------------------}
  2247. PROCEDURE DoneVideo;
  2248. BEGIN
  2249. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  2250. {$IFDEF PPC_FPC}
  2251. MouseMoveProc := Nil; { Clr mouse move ptr }
  2252. ShowMouseProc := Nil; { Clr show mouse ptr }
  2253. HideMouseProc := Nil; { Clr hide mouse ptr }
  2254. UnLock_Code(Pointer(@ShowTheMouse), 400); { Unlock cursor code }
  2255. {$ENDIF}
  2256. CloseGraph; { Close down graphics }
  2257. {$ENDIF}
  2258. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  2259. If (DefGFVFont <> 0) Then { Check font created }
  2260. DeleteObject(DefGFVFont); { Delete the font }
  2261. {$ENDIF}
  2262. END;
  2263. {---------------------------------------------------------------------------}
  2264. { ClearScreen -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 04Jan97 LdB }
  2265. {---------------------------------------------------------------------------}
  2266. PROCEDURE ClearScreen;
  2267. BEGIN
  2268. END;
  2269. {---------------------------------------------------------------------------}
  2270. { SetVideoMode -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 10Nov99 LdB }
  2271. {---------------------------------------------------------------------------}
  2272. PROCEDURE SetVideoMode (Mode: Word);
  2273. BEGIN
  2274. If (Mode > $100) Then DefLineNum := 50 { 50 line mode request }
  2275. Else DefLineNum := 24; { Normal 24 line mode }
  2276. END;
  2277. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  2278. { ERROR CONTROL ROUTINES }
  2279. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  2280. {---------------------------------------------------------------------------}
  2281. { InitSysError -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 20May98 LdB }
  2282. {---------------------------------------------------------------------------}
  2283. PROCEDURE InitSysError;
  2284. BEGIN
  2285. SysErrActive := True; { Set active flag }
  2286. END;
  2287. {---------------------------------------------------------------------------}
  2288. { DoneSysError -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 20May98 LdB }
  2289. {---------------------------------------------------------------------------}
  2290. PROCEDURE DoneSysError;
  2291. BEGIN
  2292. SysErrActive := False; { Clear active flag }
  2293. END;
  2294. {---------------------------------------------------------------------------}
  2295. { SystemError -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 20May98 LdB }
  2296. {---------------------------------------------------------------------------}
  2297. FUNCTION SystemError (ErrorCode: Integer; Drive: Byte): Integer;
  2298. BEGIN
  2299. If (FailSysErrors = False) Then Begin { Check error ignore }
  2300. End Else SystemError := 1; { Return 1 for ignored }
  2301. END;
  2302. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  2303. { STRING FORMAT ROUTINES }
  2304. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  2305. {---------------------------------------------------------------------------}
  2306. { PrintStr -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 18Feb99 LdB }
  2307. {---------------------------------------------------------------------------}
  2308. PROCEDURE PrintStr (CONST S: String);
  2309. {$IFNDEF OS_DOS} VAR Ts: String; {$ENDIF}
  2310. BEGIN
  2311. {$IFDEF OS_DOS} { DOS/DPMI CODE }
  2312. Write(S); { Write to screen }
  2313. {$ENDIF}
  2314. {$IFDEF OS_WINDOWS} { WIN/NT CODE }
  2315. Ts := S + #0; { Make asciiz }
  2316. {$IFNDEF PPC_SPEED} { NON SPEED COMPILER }
  2317. MessageBox(0, @Ts[1], Nil, mb_Ok OR mb_IconStop);{ Display to screen }
  2318. {$ELSE} { SYBIL 2+ COMPILER }
  2319. MessageBox(0, CString(@Ts[1]), Nil, mb_Ok OR
  2320. mb_IconStop); { Display to screen }
  2321. {$ENDIF}
  2322. {$ENDIF}
  2323. {$IFDEF OS_OS2} { OS2 CODE }
  2324. Ts := S + #0; { Make asciiz }
  2325. WinMessageBox(0, 0, @Ts[1], Nil, mb_Ok OR
  2326. 0, mb_IconHand); { Display to screen }
  2327. {$ENDIF}
  2328. END;
  2329. {---------------------------------------------------------------------------}
  2330. { FormatStr -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 13Jul99 LdB }
  2331. {---------------------------------------------------------------------------}
  2332. PROCEDURE FormatStr (Var Result: String; CONST Format: String; Var Params);
  2333. TYPE TLongArray = Array[0..0] Of LongInt;
  2334. VAR ResultLength, FormatIndex, Justify, Wth: Byte; Fill: Char; S: String;
  2335. FUNCTION LongToStr (L: Longint; Radix: Byte): String;
  2336. CONST HexChars: Array[0..15] Of Char =
  2337. ('0', '1', '2', '3', '4', '5', '6', '7',
  2338. '8', '9', 'A', 'B', 'C', 'D', 'E', 'F');
  2339. VAR I: LongInt; S: String; Sign: String[1];
  2340. BEGIN
  2341. LongToStr := ''; { Preset empty return }
  2342. If (L < 0) Then Begin { If L is negative }
  2343. Sign := '-'; { Sign is negative }
  2344. L := Abs(L); { Convert to positive }
  2345. End Else Sign := ''; { Sign is empty }
  2346. S := ''; { Preset empty string }
  2347. Repeat
  2348. I := L MOD Radix; { Radix mod of value }
  2349. S := HexChars[I] + S; { Add char to string }
  2350. L := L DIV Radix; { Divid by radix }
  2351. Until (L = 0); { Until no remainder }
  2352. LongToStr := Sign + S; { Return result }
  2353. END;
  2354. PROCEDURE HandleParameter (I : LongInt);
  2355. BEGIN
  2356. While (FormatIndex <= Length(Format)) Do Begin { While length valid }
  2357. While (Format[FormatIndex] <> '%') AND { Param char not found }
  2358. (FormatIndex <= Length(Format)) Do Begin { Length still valid }
  2359. Result[ResultLength+1] := Format[FormatIndex]; { Transfer character }
  2360. Inc(ResultLength); { One character added }
  2361. Inc(FormatIndex); { Next param char }
  2362. End;
  2363. If (FormatIndex < Length(Format)) AND { Not last char and }
  2364. (Format[FormatIndex] = '%') Then Begin { '%' char found }
  2365. Fill := ' '; { Default fill char }
  2366. Justify := 0; { Default justify }
  2367. Wth := 0; { Default 0=no width }
  2368. Inc(FormatIndex); { Next character }
  2369. If (Format[FormatIndex] = '0') Then
  2370. Fill := '0'; { Fill char to zero }
  2371. If (Format[FormatIndex] = '-') Then Begin { Optional just char }
  2372. Justify := 1; { Right justify }
  2373. Inc(FormatIndex); { Next character }
  2374. End;
  2375. While ((FormatIndex <= Length(Format)) AND { Length still valid }
  2376. (Format[FormatIndex] >= '0') AND
  2377. (Format[FormatIndex] <= '9')) Do Begin { Numeric character }
  2378. Wth := Wth * 10; { Multiply x10 }
  2379. Wth := Wth + Ord(Format[FormatIndex])-$30; { Add numeric value }
  2380. Inc(FormatIndex); { Next character }
  2381. End;
  2382. If ((FormatIndex <= Length(Format)) AND { Length still valid }
  2383. (Format[FormatIndex] = '#')) Then Begin { Parameter marker }
  2384. Inc(FormatIndex); { Next character }
  2385. HandleParameter(Wth); { Width is param idx }
  2386. End;
  2387. If (FormatIndex <= Length(Format)) Then Begin{ Length still valid }
  2388. Case Format[FormatIndex] Of
  2389. 'c': S := Char(TLongArray(Params)[I]); { Character parameter }
  2390. 'd': S := LongToStr(TLongArray(Params)[I],
  2391. 10); { Decimal parameter }
  2392. 's': S := PString(TLongArray(Params)[I])^;{ String parameter }
  2393. 'x': S := LongToStr(TLongArray(Params)[I],
  2394. 16); { Hex parameter }
  2395. '%': Begin { Literal % }
  2396. S := '%'; { Set string }
  2397. Inc(FormatIndex); { Next character }
  2398. Move(S[1], Result[ResultLength+1], 1); { '%' char to result }
  2399. Inc(ResultLength, Length(S)); { Inc result length }
  2400. Continue; { Now continue }
  2401. End;
  2402. End;
  2403. Inc(I); { Next parameter }
  2404. Inc(FormatIndex); { Next character }
  2405. If (Wth > 0) Then Begin { Width control active }
  2406. If (Length(S) > Wth) Then Begin { We must shorten S }
  2407. If (Justify=1) Then { Check right justify }
  2408. S := Copy(S, Length(S)-Wth+1, Wth) { Take right side data }
  2409. Else S := Copy(S, 1, Wth); { Take left side data }
  2410. End Else Begin { We must pad out S }
  2411. If (Justify=1) Then { Right justify }
  2412. While (Length(S) < Wth) Do
  2413. S := S+Fill Else { Right justify fill }
  2414. While (Length(S) < Wth) Do
  2415. S := Fill + S; { Left justify fill }
  2416. End;
  2417. End;
  2418. Move(S[1], Result[ResultLength+1],
  2419. Length(S)); { Move data to result }
  2420. ResultLength := ResultLength + Length(S); { Adj result length }
  2421. End;
  2422. End;
  2423. End;
  2424. END;
  2425. BEGIN
  2426. ResultLength := 0; { Zero result length }
  2427. FormatIndex := 1; { Format index to 1 }
  2428. HandleParameter(0); { Handle parameter }
  2429. {$IFDEF PPC_DELPHI3} { DELPHI 3+ COMPILER }
  2430. SetLength(Result, ResultLength); { Set string length }
  2431. {$ELSE} { OTHER COMPILERS }
  2432. Result[0] := Chr(ResultLength); { Set string length }
  2433. {$ENDIF}
  2434. END;
  2435. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  2436. { NEW QUEUED EVENT HANDLER ROUTINES }
  2437. {+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++}
  2438. {---------------------------------------------------------------------------}
  2439. { PutEventInQueue -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 17Mar98 LdB }
  2440. {---------------------------------------------------------------------------}
  2441. FUNCTION PutEventInQueue (Var Event: TEvent): Boolean;
  2442. BEGIN
  2443. If (QueueCount < QueueMax) Then Begin { Check room in queue }
  2444. Queue[QueueHead] := Event; { Store event }
  2445. Inc(QueueHead); { Inc head position }
  2446. If (QueueHead = QueueMax) Then QueueHead := 0; { Roll to start check }
  2447. Inc(QueueCount); { Inc queue count }
  2448. PutEventInQueue := True; { Return successful }
  2449. End Else PutEventInQueue := False; { Return failure }
  2450. END;
  2451. {---------------------------------------------------------------------------}
  2452. { NextQueuedEvent -> Platforms DOS/DPMI/WIN/NT/OS2 - Updated 17Mar98 LdB }
  2453. {---------------------------------------------------------------------------}
  2454. PROCEDURE NextQueuedEvent(Var Event: TEvent);
  2455. BEGIN
  2456. If (QueueCount > 0) Then Begin { Check queued event }
  2457. Event := Queue[QueueTail]; { Fetch next event }
  2458. Inc(QueueTail); { Inc tail position }
  2459. If (QueueTail = QueueMax) Then QueueTail := 0; { Roll to start check }
  2460. Dec(QueueCount); { Dec queue count }
  2461. End Else Event.What := evNothing; { Return empty event }
  2462. END;
  2463. {***************************************************************************}
  2464. { UNIT INITIALIZATION ROUTINE }
  2465. {***************************************************************************}
  2466. BEGIN
  2467. ButtonCount := DetectMouse; { Detect mouse }
  2468. DetectVideo; { Detect video }
  2469. SaveExit := ExitProc; { Save old exit }
  2470. ExitProc := @ExitDrivers; { Set new exit }
  2471. END.