home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 9 / 09.iso / l / l180 / 1.ddi / INTRODEM.BAS < prev    next >
Encoding:
BASIC Source File  |  1989-02-07  |  47.5 KB  |  1,445 lines

  1.   ' ************************************************
  2.   ' **  Name:          INTRODEM                   **
  3.   ' **  Type:          Program                    **
  4.   ' **  Module:        INTRODEM.BAS               **
  5.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  6.   ' ************************************************
  7.   '
  8.   ' Quick introductory demonstration created from the
  9.   ' QuickBASIC Toolbox routines.
  10.   ' USAGE:  Run as a main program to see the demonstration.
  11.   ' REQUIREMENTS: CGA or VGA
  12.   ' .MAK FILE CONTENTS: (none)
  13.   ' FUNCTIONS:    Collision%            Date2Day%           Date2Julian&
  14.   '               Date2Month%           Date2Year%          DayOfTheWeek$
  15.   '               DOSVersion!           GetDirectory$       GetDrive$
  16.   '               GetVerifyState%       InKeyCode%          Julian2Date$
  17.   '               KeyCode%              MDY2Date$           MenuString%
  18.   '               MonthName$
  19.   ' SUBPROGRAMS:  Attrib                DrawBox             EditLine
  20.   '               Equipment             FindFirstFile       FindNextFile
  21.   '               GetDiskFreeSpace      GetFileData         GetShiftStates
  22.   '               Interrupt             InterruptX          Mouse
  23.   '               MouseMickey           OneMonthCalendar    SpaceWorms
  24.   '               TextGet               TextPut             VideoState
  25.   ' PARAMETERS:   (none)
  26.   ' VARIABLES:    disk          Structure of type DiskFreeSpaceType
  27.   '               equip         Structure of type EquipmentType
  28.   '               file          Structure of type FileDataType
  29.   '               shift         Structure of type ShiftType
  30.   '               wormship1%()  First image of UFO
  31.   '               i%            Looping index
  32.   '               h$            Hexadecimal notation character string
  33.   '               wormship2%()  Second image of UFO
  34.   '               title$()      Storage for title boxes
  35.   '               row%()        Row location of each title box
  36.   '               col%()        Column location of each title box
  37.   '               row2%         Row location at opposite corner of title box
  38.   '               col2%         Column location at opposite corner of title box
  39.   '               t0            Timer value at start of timing loop
  40.   '               path$         String for finding directory contents
  41.   '               result%       Returned code for finding each file name
  42.   '               dta$          Disk transfer area buffer string
  43.   '               mode%         Current video mode number
  44.   '               columns%      Current number of video columns
  45.   '               page%         Current video page number
  46.   '               drive$        Default drive string for getting disk
  47.   '                               information
  48.   '               quitFlag%     Indicates user wants to quit
  49.   
  50.   ' Constants
  51.     CONST FALSE = 0
  52.     CONST TRUE = NOT FALSE
  53.   
  54.   ' Define color constants
  55.     CONST BLACK = 0
  56.     CONST BLUE = 1
  57.     CONST GREEN = 2
  58.     CONST CYAN = 3
  59.     CONST RED = 4
  60.     CONST MAGENTA = 5
  61.     CONST BROWN = 6
  62.     CONST WHITE = 7
  63.     CONST BRIGHT = 8
  64.     CONST BLINK = 16
  65.     CONST YELLOW = BROWN + BRIGHT
  66.   
  67.   ' Key code numbers
  68.     CONST BACKSPACE = 8
  69.     CONST CTRLLEFTARROW = 29440
  70.     CONST CTRLRIGHTARROW = 29696
  71.     CONST CTRLY = 25
  72.     CONST CTRLQ = 17
  73.     CONST DELETE = 21248
  74.     CONST DOWNARROW = 20480
  75.     CONST ENDKEY = 20224
  76.     CONST ENTER = 13
  77.     CONST ESCAPE = 27
  78.     CONST HOME = 18176
  79.     CONST INSERTKEY = 20992
  80.     CONST LEFTARROW = 19200
  81.     CONST RIGHTARROW = 19712
  82.     CONST TABKEY = 9
  83.     CONST UPARROW = 18432
  84.   
  85.   ' File search attribute bits
  86.     CONST ISNORMAL = 0
  87.     CONST ISREADONLY = 1
  88.     CONST ISHIDDEN = 2
  89.     CONST ISSYSTEM = 4
  90.     CONST ISVOLUMELABEL = 8
  91.     CONST ISSUBDIRECTORY = 16
  92.     CONST ISARCHIVED = 32
  93.   
  94.   ' Here we'll search for normal files and subdirectories
  95.     CONST FILEATTRIBUTE = ISNORMAL + ISSUBDIRECTORY
  96.   
  97.   ' Declare the Type structures
  98.     TYPE RegType
  99.         ax    AS INTEGER
  100.         bx    AS INTEGER
  101.         cx    AS INTEGER
  102.         dx    AS INTEGER
  103.         Bp    AS INTEGER
  104.         si    AS INTEGER
  105.         di    AS INTEGER
  106.         flags AS INTEGER
  107.     END TYPE
  108.   
  109.     TYPE RegTypeX
  110.         ax    AS INTEGER
  111.         bx    AS INTEGER
  112.         cx    AS INTEGER
  113.         dx    AS INTEGER
  114.         Bp    AS INTEGER
  115.         si    AS INTEGER
  116.         di    AS INTEGER
  117.         flags AS INTEGER
  118.         ds    AS INTEGER
  119.         es    AS INTEGER
  120.     END TYPE
  121.   
  122.     TYPE DiskFreeSpaceType
  123.         sectorsPerCluster AS INTEGER
  124.         bytesPerSector AS INTEGER
  125.         clustersPerDrive AS LONG
  126.         availableClusters AS LONG
  127.         availableBytes AS LONG
  128.     END TYPE
  129.   
  130.     TYPE FileDataType
  131.         finame    AS STRING * 12
  132.         year      AS INTEGER
  133.         month     AS INTEGER
  134.         day       AS INTEGER
  135.         hour      AS INTEGER
  136.         minute    AS INTEGER
  137.         second    AS INTEGER
  138.         attribute AS INTEGER
  139.         size      AS LONG
  140.     END TYPE
  141.   
  142.     TYPE EquipmentType
  143.         printers     AS INTEGER
  144.         gameAdapter  AS INTEGER
  145.         serial       AS INTEGER
  146.         floppies     AS INTEGER
  147.         initialVideo AS INTEGER
  148.         coprocessor  AS INTEGER
  149.     END TYPE
  150.   
  151.     TYPE ShiftType
  152.         right           AS INTEGER
  153.         left            AS INTEGER
  154.         ctrl            AS INTEGER
  155.         alt             AS INTEGER
  156.         scrollLockState AS INTEGER
  157.         numLockState    AS INTEGER
  158.         capsLockState   AS INTEGER
  159.         insertState     AS INTEGER
  160.     END TYPE
  161.   
  162.   ' Functions
  163.     DECLARE FUNCTION Collision% (object%(), backGround%())
  164.     DECLARE FUNCTION Date2Day% (dat$)
  165.     DECLARE FUNCTION Date2Julian& (dat$)
  166.     DECLARE FUNCTION Date2Month% (dat$)
  167.     DECLARE FUNCTION Date2Year% (dat$)
  168.     DECLARE FUNCTION DayOfTheWeek$ (dat$)
  169.     DECLARE FUNCTION DOSVersion! ()
  170.     DECLARE FUNCTION GetDirectory$ (drive$)
  171.     DECLARE FUNCTION GetDrive$ ()
  172.     DECLARE FUNCTION GetVerifyState% ()
  173.     DECLARE FUNCTION InKeyCode% ()
  174.     DECLARE FUNCTION Julian2Date$ (julian&)
  175.     DECLARE FUNCTION KeyCode% ()
  176.     DECLARE FUNCTION MDY2Date$ (month%, day%, year%)
  177.     DECLARE FUNCTION MenuString% CDECL (row%, col%, a$)
  178.     DECLARE FUNCTION MonthName$ (dat$)
  179.   
  180.   ' Subprograms
  181.     DECLARE SUB Attrib ()
  182.     DECLARE SUB DrawBox (row1%, col1%, row2%, col2%)
  183.     DECLARE SUB EditLine (a$, exitCode%)
  184.     DECLARE SUB Equipment (equip AS ANY)
  185.     DECLARE SUB FindFirstFile (path$, dta$, result%)
  186.     DECLARE SUB FindNextFile (dta$, result%)
  187.     DECLARE SUB GetDiskFreeSpace (drive$, disk AS ANY)
  188.     DECLARE SUB GetFileData (dta$, file AS ANY)
  189.     DECLARE SUB GetShiftStates (shift AS ANY)
  190.     DECLARE SUB Interrupt (intnum%, inreg AS RegType, outreg AS RegType)
  191.     DECLARE SUB InterruptX (intnum%, inreg AS RegTypeX, outreg AS RegTypeX)
  192.     DECLARE SUB Mouse (m1%, m2%, m3%, m4%)
  193.     DECLARE SUB MouseMickey (horizontal%, vertical%)
  194.     DECLARE SUB OneMonthCalendar (dat$, row%, col%)
  195.     DECLARE SUB SpaceWorms ()
  196.     DECLARE SUB TextGet CDECL (r1%, c1%, r2%, c2%, a$)
  197.     DECLARE SUB TextPut CDECL (r1%, c1%, r2%, c2%, a$)
  198.     DECLARE SUB VideoState (mode%, columns%, page%)
  199.   
  200.   ' Data structures
  201.     DIM disk AS DiskFreeSpaceType
  202.     DIM equip AS EquipmentType
  203.     DIM file AS FileDataType
  204.     DIM shift AS ShiftType
  205.   
  206.   ' Worm ship image 1
  207.   ' (DRAW$) "c1u2e2r4f2d2 l8 be1p1,1 bf1br6 c2r6f3g3l20h3e3r18 "
  208.   ' (DRAW$) "bd3 p3,2 c2 l1 bl3 l1 bl3 l1 bl3 l1 bl3 l1 bd7 br1"
  209.   ' (DRAW$) " e3r10 f3"
  210.     DIM wormship1%(0 TO 54)
  211.     FOR i% = 0 TO 54
  212.         READ h$
  213.         wormship1%(i%) = VAL("&H" + h$)
  214.     NEXT i%
  215.     DATA 36,F,0,5501,0,0,500,4055,0,0,5515,50,0,1500,5055,0,AA02
  216.     DATA AAAA,AAAA,B00,FFFF,FFFF,80FF,FF2F,FFFF,FFFF,BFE0,AFAF
  217.     DATA AFAF,F8AF,FF2F,FFFF,FFFF,BE0,FFFF,FFFF,80FF,AA02,AAAA
  218.     DATA AAAA,0,AA00,A8AA,0,200,0,2,0,8,0,80,2000,0,2000,0
  219.   
  220.   ' Worm ship image 2
  221.   ' (DRAW$) "c1u2e2r4f2d2 l8 be1p1,1 bf1br6 c2r6f3g3l20h3e3r18 "
  222.   ' (DRAW$) "bd3 p3,2 c2 br2 l1 bl3 l1 bl3 l1 bl3 l1 bl3 l1 bl3"
  223.   ' (DRAW$) " l1 bd7 br3 e3r10 f3"
  224.     DIM wormship2%(0 TO 54)
  225.     FOR i% = 0 TO 54
  226.         READ h$
  227.         wormship2%(i%) = VAL("&H" + h$)
  228.     NEXT i%
  229.     DATA 36,F,0,5501,0,0,400,4000,0,0,10,10,0,1000,1000,0,AA02
  230.     DATA AAAA,AAAA,B00,FFFF,FFFF,80FF,FF2F,FFFF,FFFF,BAE0,FAFA
  231.     DATA FAFA,F8FA,FF2F,FFFF,FFFF,BE0,FFFF,FFFF,80FF,AA02,AAAA
  232.     DATA AAAA,0,AA00,A8AA,0,200,0,2,0,8,0,80,2000,0,2000,0
  233.   
  234.   ' Initialization
  235.     DIM title$(0 TO 17), row%(0 TO 17), col%(0 TO 17)
  236.     FOR i% = 0 TO 17
  237.         title$(i%) = SPACE$(432)
  238.     NEXT i%
  239.     COLOR BLACK, WHITE
  240.     CLS
  241.     menu$ = "  Files  Equipment  Game  Attributes  Calendar  Quit  "
  242.   
  243.   ' Main loop
  244.     DO
  245.       
  246.       ' Create main screen
  247.         COLOR BLACK, WHITE
  248.         DrawBox 1, 1, 25, 80
  249.         DrawBox 2, 3, 24, 78
  250.         DrawBox 3, 5, 23, 76
  251.       
  252.       ' Build the title box
  253.         COLOR YELLOW, BLUE
  254.         DrawBox 9, 29, 17, 52
  255.         COLOR BRIGHT + MAGENTA, BLUE
  256.         LOCATE 10, 35
  257.         PRINT "Introducing"
  258.         COLOR BRIGHT + GREEN, BLUE
  259.         LOCATE 12, 39
  260.         PRINT "THE"
  261.         LOCATE 13, 34
  262.         PRINT "QuickBASIC 4.0"
  263.         LOCATE 14, 37
  264.         PRINT "TOOLBOX"
  265.       
  266.       ' Grab a copy of the title box
  267.         TextGet 9, 29, 17, 52, title$(0)
  268.         row%(0) = 9
  269.         col%(0) = 29
  270.       
  271.       ' Randomly place 17 title boxes, saving the background
  272.         FOR i% = 1 TO 17
  273.             row%(i%) = INT(RND * 16) + 1
  274.             col%(i%) = INT(RND * 56) + 1
  275.             row2% = row%(i%) + 8
  276.             col2% = col%(i%) + 23
  277.             TextGet row%(i%), col%(i%), row2%, col2%, title$(i%)
  278.             TextPut row%(i%), col%(i%), row2%, col2%, title$(0)
  279.         NEXT i%
  280.       
  281.       ' Delay for half a second
  282.         t0 = TIMER
  283.         DO
  284.         LOOP WHILE TIMER - t0 < .5
  285.       
  286.       ' Replace the backgrounds
  287.         FOR i% = 17 TO 0 STEP -1
  288.             row2% = row%(i%) + 8
  289.             col2% = col%(i%) + 23
  290.             TextPut row%(i%), col%(i%), row2%, col2%, title$(i%)
  291.         NEXT i%
  292.       
  293.       ' Now for the main menu
  294.         LOCATE , , 0
  295.         SELECT CASE MenuString%(6, 12, menu$)
  296.           
  297.       ' Current drive, path, and directory listing
  298.         CASE 1
  299.             COLOR BRIGHT + WHITE, CYAN
  300.             CLS
  301.             PRINT "Enter path (or just press ";
  302.             PRINT CHR$(17); CHR$(196); CHR$(217); ")... "
  303.             path$ = GetDirectory$("")
  304.             IF RIGHT$(path$, 1) <> "\" THEN
  305.                 path$ = path$ + "\"
  306.             END IF
  307.             path$ = LEFT$(path$ + "*.*" + SPACE$(70), 70)
  308.             COLOR YELLOW, BLUE
  309.             LOCATE 2, 5
  310.             EditLine path$, exitCode%
  311.             path$ = LTRIM$(RTRIM$(path$))
  312.             IF path$ = "" OR RIGHT$(path$, 1) = "\" THEN
  313.                 path$ = path$ + "*.*"
  314.             END IF
  315.             COLOR BLACK, CYAN
  316.             PRINT
  317.             PRINT
  318.             FindFirstFile path$, dta$, result%
  319.             DO UNTIL result%
  320.                 GetFileData dta$, file
  321.                 IF file.attribute AND &H10 THEN
  322.                     PRINT "*"; file.finame,
  323.                 ELSE
  324.                     PRINT " "; file.finame,
  325.                 END IF
  326.                 FindNextFile dta$, result%
  327.             LOOP
  328.             PRINT
  329.             PRINT
  330.             COLOR BRIGHT + WHITE, CYAN
  331.             PRINT "Press any key to continue"
  332.             DO
  333.             LOOP WHILE INKEY$ = ""
  334.             COLOR BLACK, WHITE
  335.             CLS
  336.           
  337.       ' Equipment information
  338.         CASE 2
  339.             COLOR BRIGHT + GREEN, BLUE
  340.             CLS
  341.             PRINT "Information about your hardware and software..."
  342.             COLOR YELLOW, BLUE
  343.             PRINT
  344.             PRINT "Current version of DOS is "; DOSVersion!
  345.             Equipment equip
  346.             PRINT "Number of printers", equip.printers
  347.             PRINT "Game adapter", , equip.gameAdapter
  348.             PRINT "Serial I/O ports", equip.serial
  349.             PRINT "Floppy disk drives", equip.floppies
  350.             PRINT "Initial video state", equip.initialVideo
  351.             PRINT "Numerical coprocessor", equip.coprocessor
  352.             VideoState mode%, columns%, page%
  353.             PRINT "Video mode number", mode%
  354.             PRINT "Video width", , columns%
  355.             PRINT "Video page", , page%
  356.             GetShiftStates shift
  357.             PRINT "Scroll lock", , shift.scrollLockState
  358.             PRINT "Num lock", , shift.numLockState
  359.             PRINT "Caps lock", , shift.capsLockState
  360.             PRINT "Insert lock", , shift.insertState
  361.             PRINT "Disk verify state", GetVerifyState%
  362.             GetDiskFreeSpace drive$, disk
  363.             PRINT "Disk sectors per cluster", disk.sectorsPerCluster
  364.             PRINT "Disk bytes per sector", disk.bytesPerSector
  365.             PRINT "Disk clusters on drive", disk.clustersPerDrive
  366.             PRINT "Disk available clusters", disk.availableClusters
  367.             PRINT "Disk available bytes", disk.availableBytes
  368.             PRINT "Current complete path", " "; RTRIM$(GetDirectory$(""))
  369.             LOCATE 15, 45
  370.             COLOR BRIGHT + WHITE, BLACK
  371.             PRINT " Press any key to continue "
  372.             DO
  373.             LOOP WHILE INKEY$ = ""
  374.             COLOR WHITE, BLACK
  375.             CLS
  376.           
  377.       ' Worms from space
  378.         CASE 3
  379.             SpaceWorms
  380.           
  381.       ' Color attributes
  382.         CASE 4
  383.             Attrib
  384.             LOCATE 23, 1
  385.             PRINT "Press any key to continue"
  386.             DO
  387.             LOOP WHILE INKEY$ = ""
  388.             CLS
  389.           
  390.       ' Calendar sheet for this month
  391.         CASE 5
  392.             COLOR YELLOW, CYAN
  393.             CLS
  394.             COLOR YELLOW, BLUE
  395.             DrawBox 6, 21, 19, 60
  396.             COLOR BRIGHT + WHITE, BLUE
  397.             OneMonthCalendar DATE$, 9, 26
  398.             LOCATE 23, 27
  399.             PRINT "Press any key to continue"
  400.             DO
  401.             LOOP WHILE INKEY$ = ""
  402.             COLOR WHITE, BLACK
  403.             CLS
  404.           
  405.       ' Must be time to quit
  406.         CASE ELSE
  407.             quitFlag% = TRUE
  408.           
  409.         END SELECT
  410.       
  411.     LOOP UNTIL quitFlag%
  412.   
  413.   ' All done
  414.     COLOR WHITE, BLACK
  415.     CLS
  416.     END
  417.   
  418.   ' ************************************************
  419.   ' **  Name:          Attrib                     **
  420.   ' **  Type:          Subprogram                 **
  421.   ' **  Module:        ATTRIB.BAS                 **
  422.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  423.   ' ************************************************
  424.   '
  425.   ' Displays table of color attributes for text mode
  426.   '
  427.     SUB Attrib STATIC
  428.         SCREEN 0
  429.         CLS
  430.         PRINT "Attributes for the COLOR statement in text mode (SCREEN 0)."
  431.         PRINT "Add 16 to the foreground to cause the character to blink."
  432.         FOR bgd% = 0 TO 7
  433.             COLOR bgd% XOR 7, bgd%
  434.             PRINT
  435.             PRINT "Background%"; STR$(bgd%),
  436.             PRINT "Foreground% ..."; SPACE$(41)
  437.             FOR fgd% = 0 TO 15
  438.                 COLOR fgd%, bgd%
  439.                 PRINT STR$(fgd%); "  ";
  440.             NEXT fgd%
  441.         NEXT bgd%
  442.         COLOR 7, 0
  443.         PRINT
  444.     END SUB
  445.   
  446.   ' ************************************************
  447.   ' **  Name:          Collision%                 **
  448.   ' **  Type:          Function                   **
  449.   ' **  Module:        GAMES.BAS                  **
  450.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  451.   ' ************************************************
  452.   '
  453.   ' Returns TRUE if any non-zero pixels occur in the
  454.   ' same byte of video memory, as saved in the object%()
  455.   ' and backGround%() arrays.  The arrays must be the
  456.   ' same size.
  457.   '
  458.     FUNCTION Collision% (object%(), backGround%()) STATIC
  459.         lo% = LBOUND(object%)
  460.         uo% = UBOUND(object%)
  461.         lb% = LBOUND(backGround%)
  462.         ub% = UBOUND(backGround%)
  463.         IF lo% <> lb% OR uo% <> ub% THEN
  464.             PRINT "Error: Collision - The object and background"
  465.             PRINT "graphics arrays have different dimensions."
  466.             SYSTEM
  467.         END IF
  468.         FOR i% = lo% + 2 TO uo%
  469.             IF object%(i%) THEN
  470.                 IF backGround%(i%) THEN
  471.                     Collision% = TRUE
  472.                     EXIT FUNCTION
  473.                 END IF
  474.             END IF
  475.         NEXT i%
  476.         Collision% = FALSE
  477.     END FUNCTION
  478.   
  479.   ' ************************************************
  480.   ' **  Name:          Date2Day%                  **
  481.   ' **  Type:          Function                   **
  482.   ' **  Module:        CALENDAR.BAS               **
  483.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  484.   ' ************************************************
  485.   '
  486.   ' Returns the day number given a date in the
  487.   ' QuickBASIC string format "MM-DD-YYYY".
  488.   '
  489.     FUNCTION Date2Day% (dat$) STATIC
  490.         Date2Day% = VAL(MID$(dat$, 4, 2))
  491.     END FUNCTION
  492.   
  493.   ' ************************************************
  494.   ' **  Name:          Date2Julian&               **
  495.   ' **  Type:          Function                   **
  496.   ' **  Module:        CALENDAR.BAS               **
  497.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  498.   ' ************************************************
  499.   '
  500.   ' Returns the astronomical Julian day number given a
  501.   ' date in the QuickBASIC string format "MM-DD-YYYY".
  502.   '
  503.     FUNCTION Date2Julian& (dat$) STATIC
  504.         month% = Date2Month%(dat$)
  505.         day% = Date2Day%(dat$)
  506.         year% = Date2Year%(dat$)
  507.         IF year% < 1583 THEN
  508.             PRINT "Date2Julian: Year is less than 1583"
  509.             SYSTEM
  510.         END IF
  511.         IF month% > 2 THEN
  512.             month% = month% - 3
  513.         ELSE
  514.             month% = month% + 9
  515.             year% = year% - 1
  516.         END IF
  517.         ta& = 146097 * (year% \ 100) \ 4
  518.         tb& = 1461& * (year% MOD 100) \ 4
  519.         tc& = (153 * month% + 2) \ 5 + day% + 1721119
  520.         Date2Julian& = ta& + tb& + tc&
  521.     END FUNCTION
  522.   
  523.   ' ************************************************
  524.   ' **  Name:          Date2Month%                **
  525.   ' **  Type:          Function                   **
  526.   ' **  Module:        CALENDAR.BAS               **
  527.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  528.   ' ************************************************
  529.   '
  530.   ' Returns the month number given a date in the
  531.   ' QuickBASIC string format "MM-DD-YYYY".
  532.   '
  533.     FUNCTION Date2Month% (dat$) STATIC
  534.         Date2Month% = VAL(MID$(dat$, 1, 2))
  535.     END FUNCTION
  536.   
  537.   ' ************************************************
  538.   ' **  Name:          Date2Year%                 **
  539.   ' **  Type:          Function                   **
  540.   ' **  Module:        CALENDAR.BAS               **
  541.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  542.   ' ************************************************
  543.   '
  544.   ' Returns the year number given a date in the
  545.   ' QuickBASIC string format "MM-DD-YYYY".
  546.   '
  547.     FUNCTION Date2Year% (dat$) STATIC
  548.         Date2Year% = VAL(MID$(dat$, 7))
  549.     END FUNCTION
  550.   
  551.   ' ************************************************
  552.   ' **  Name:          DayOfTheWeek$              **
  553.   ' **  Type:          Function                   **
  554.   ' **  Module:        CALENDAR.BAS               **
  555.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  556.   ' ************************************************
  557.   '
  558.   ' Returns a string stating the day of the week.
  559.   ' Input is a date expressed in the QuickBASIC string
  560.   ' format "MM-DD-YYYY".
  561.   '
  562.     FUNCTION DayOfTheWeek$ (dat$) STATIC
  563.         SELECT CASE Date2Julian&(dat$) MOD 7
  564.         CASE 0
  565.             DayOfTheWeek$ = "Monday"
  566.         CASE 1
  567.             DayOfTheWeek$ = "Tuesday"
  568.         CASE 2
  569.             DayOfTheWeek$ = "Wednesday"
  570.         CASE 3
  571.             DayOfTheWeek$ = "Thursday"
  572.         CASE 4
  573.             DayOfTheWeek$ = "Friday"
  574.         CASE 5
  575.             DayOfTheWeek$ = "Saturday"
  576.         CASE 6
  577.             DayOfTheWeek$ = "Sunday"
  578.         END SELECT
  579.     END FUNCTION
  580.   
  581.   ' ************************************************
  582.   ' **  Name:          DOSVersion!                **
  583.   ' **  Type:          Function                   **
  584.   ' **  Module:        DOSCALLS.BAS               **
  585.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  586.   ' ************************************************
  587.   '
  588.   ' Returns the version number of DOS
  589.   '
  590.     FUNCTION DOSVersion! STATIC
  591.         DIM reg AS RegType
  592.         reg.ax = &H3000
  593.         Interrupt &H21, reg, reg
  594.         major% = reg.ax MOD 256
  595.         minor% = reg.ax \ 256
  596.         DOSVersion! = major% + minor% / 100!
  597.     END FUNCTION
  598.   
  599.   ' ************************************************
  600.   ' **  Name:          DrawBox                    **
  601.   ' **  Type:          Subprogram                 **
  602.   ' **  Module:        EDIT.BAS                   **
  603.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  604.   ' ************************************************
  605.   '
  606.   ' Draw a double line box
  607.   '
  608.     SUB DrawBox (row1%, col1%, row2%, col2%) STATIC
  609.       
  610.       ' Determine inside width of box
  611.         wide% = col2% - col1% - 1
  612.       
  613.       ' Across the top
  614.         LOCATE row1%, col1%, 0
  615.         PRINT CHR$(201);
  616.         PRINT STRING$(wide%, 205);
  617.         PRINT CHR$(187);
  618.       
  619.       ' down the sides
  620.         FOR row3% = row1% + 1 TO row2% - 1
  621.             LOCATE row3%, col1%, 0
  622.             PRINT CHR$(186);
  623.             PRINT SPACE$(wide%);
  624.             PRINT CHR$(186);
  625.         NEXT row3%
  626.       
  627.       ' Across the bottom
  628.         LOCATE row2%, col1%, 0
  629.         PRINT CHR$(200);
  630.         PRINT STRING$(wide%, 205);
  631.         PRINT CHR$(188);
  632.       
  633.     END SUB
  634.   
  635.   ' ************************************************
  636.   ' **  Name:          EditLine                   **
  637.   ' **  Type:          Subprogram                 **
  638.   ' **  Module:        EDIT.BAS                   **
  639.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  640.   ' ************************************************
  641.   '
  642.   ' Allows editing of a string at the current cursor position
  643.   ' on the screen.  Keys acted upon include Right-arrow,
  644.   ' Left-arrow, Ctrl-left-arrow, Ctrl-right-arrow, Home, End,
  645.   ' Insert, Escape, Enter, Backspace, and Delete.
  646.   ' Pressing the Enter, Up-arrow, or Down-arrow terminates
  647.   ' the subprogram and returns exitCode% of 0, +1, or -1.
  648.   '
  649.     SUB EditLine (a$, exitCode%) STATIC
  650.       
  651.       ' Set up some variables
  652.         row% = CSRLIN
  653.         col% = POS(0)
  654.         length% = LEN(a$)
  655.         ptr% = 0
  656.         insert% = TRUE
  657.         quit% = FALSE
  658.         original$ = a$
  659.       
  660.       ' Main processing loop
  661.         DO
  662.           
  663.           ' Display the line
  664.             LOCATE row%, col%, 0
  665.             PRINT a$;
  666.           
  667.           ' Show appropriate cursor type
  668.             IF insert% THEN
  669.                 LOCATE row%, col% + ptr%, 1, 6, 7
  670.             ELSE
  671.                 LOCATE row%, col% + ptr%, 1, 1, 7
  672.             END IF
  673.           
  674.           ' Get next key stroke
  675.             keyNumber% = KeyCode%
  676.           
  677.           ' Process the key
  678.             SELECT CASE keyNumber%
  679.               
  680.             CASE INSERTKEY
  681.                 IF insert% THEN
  682.                     insert% = FALSE
  683.                 ELSE
  684.                     insert% = TRUE
  685.                 END IF
  686.               
  687.             CASE BACKSPACE
  688.                 IF ptr% THEN
  689.                     a$ = a$ + " "
  690.                     a$ = LEFT$(a$, ptr% - 1) + MID$(a$, ptr% + 1)
  691.                     ptr% = ptr% - 1
  692.                 END IF
  693.               
  694.             CASE DELETE
  695.                 a$ = a$ + " "
  696.                 a$ = LEFT$(a$, ptr%) + MID$(a$, ptr% + 2)
  697.               
  698.             CASE UPARROW
  699.                 exitCode% = 1
  700.                 quit% = TRUE
  701.               
  702.             CASE DOWNARROW
  703.                 exitCode% = -1
  704.                 quit% = TRUE
  705.               
  706.             CASE LEFTARROW
  707.                 IF ptr% THEN
  708.                     ptr% = ptr% - 1
  709.                 END IF
  710.               
  711.             CASE RIGHTARROW
  712.                 IF ptr% < length% - 1 THEN
  713.                     ptr% = ptr% + 1
  714.                 END IF
  715.               
  716.             CASE ENTER
  717.                 exitCode% = 0
  718.                 quit% = TRUE
  719.               
  720.             CASE HOME
  721.                 ptr% = 0
  722.               
  723.             CASE ENDKEY
  724.                 ptr% = length% - 1
  725.               
  726.             CASE CTRLRIGHTARROW
  727.                 DO UNTIL MID$(a$, ptr% + 1, 1) = " " OR ptr% = length% - 1
  728.                     ptr% = ptr% + 1
  729.                 LOOP
  730.                 DO UNTIL MID$(a$, ptr% + 1, 1) <> " " OR ptr% = length% - 1
  731.                     ptr% = ptr% + 1
  732.                 LOOP
  733.               
  734.             CASE CTRLLEFTARROW
  735.                 DO UNTIL MID$(a$, ptr% + 1, 1) = " " OR ptr% = 0
  736.                     ptr% = ptr% - 1
  737.                 LOOP
  738.                 DO UNTIL MID$(a$, ptr% + 1, 1) <> " " OR ptr% = 0
  739.                     ptr% = ptr% - 1
  740.                 LOOP
  741.                 DO UNTIL MID$(a$, ptr% + 1, 1) = " " OR ptr% = 0
  742.                     ptr% = ptr% - 1
  743.                 LOOP
  744.                 IF ptr% THEN
  745.                     ptr% = ptr% + 1
  746.                 END IF
  747.               
  748.             CASE CTRLY
  749.                 a$ = SPACE$(length%)
  750.                 ptr% = 0
  751.               
  752.             CASE CTRLQ
  753.                 ctrlQflag% = TRUE
  754.               
  755.             CASE ESCAPE
  756.                 a$ = original$
  757.                 ptr% = 0
  758.                 insert% = TRUE
  759.               
  760.             CASE IS > 255
  761.                 SOUND 999, 1
  762.               
  763.             CASE IS < 32
  764.                 SOUND 999, 1
  765.               
  766.             CASE ELSE
  767.               
  768.               ' Convert key code to character string
  769.                 kee$ = CHR$(keyNumber%)
  770.               
  771.               ' Insert or overstrike
  772.                 IF insert% THEN
  773.                     a$ = LEFT$(a$, ptr%) + kee$ + MID$(a$, ptr% + 1)
  774.                     a$ = LEFT$(a$, length%)
  775.                 ELSE
  776.                     IF ptr% < length% THEN
  777.                         MID$(a$, ptr% + 1, 1) = kee$
  778.                     END IF
  779.                 END IF
  780.               
  781.               ' Are we up against the wall?
  782.                 IF ptr% < length% THEN
  783.                     ptr% = ptr% + 1
  784.                 ELSE
  785.                     SOUND 999, 1
  786.                 END IF
  787.               
  788.               ' Special check for Ctrl-q-y (del to end of line)
  789.                 IF kee$ = "y" AND ctrlQflag% THEN
  790.                     IF ptr% <= length% THEN
  791.                         sp% = length% - ptr% + 1
  792.                         MID$(a$, ptr%, sp%) = SPACE$(sp%)
  793.                         ptr% = ptr% - 1
  794.                     END IF
  795.                 END IF
  796.               
  797.               ' Clear out the Ctrl-q signal
  798.                 ctrlQflag% = FALSE
  799.               
  800.             END SELECT
  801.           
  802.         LOOP UNTIL quit%
  803.       
  804.     END SUB
  805.   
  806.   ' ************************************************
  807.   ' **  Name:          Equipment                  **
  808.   ' **  Type:          Subprogram                 **
  809.   ' **  Module:        BIOSCALL.BAS               **
  810.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  811.   ' ************************************************
  812.   '
  813.   ' Returns equipment configuration information from BIOS
  814.   '
  815.     SUB Equipment (equip AS EquipmentType) STATIC
  816.         DIM reg AS RegType
  817.         Interrupt &H11, reg, reg
  818.         equip.printers = (reg.ax AND &HC000&) \ 16384
  819.         equip.gameAdapter = (reg.ax AND &H1000) \ 4096
  820.         equip.serial = (reg.ax AND &HE00) \ 512
  821.         equip.floppies = (reg.ax AND &HC0) \ 64 + 1
  822.         equip.initialVideo = (reg.ax AND &H30) \ 16
  823.         equip.coprocessor = (reg.ax AND 2) \ 2
  824.     END SUB
  825.   
  826.   ' ************************************************
  827.   ' **  Name:          FindFirstFile              **
  828.   ' **  Type:          Subprogram                 **
  829.   ' **  Module:        FILEINFO.BAS               **
  830.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  831.   ' ************************************************
  832.   '
  833.   ' Finds first file that matches the path$
  834.   '
  835.     SUB FindFirstFile (path$, dta$, result%) STATIC
  836.       
  837.       ' Initialization
  838.         DIM reg AS RegTypeX
  839.       
  840.       ' The path must be a null terminated string
  841.         thePath$ = path$ + CHR$(0)
  842.       
  843.       ' Get current DTA address
  844.         reg.ax = &H2F00
  845.         InterruptX &H21, reg, reg
  846.         sgmt% = reg.es
  847.         ofst% = reg.bx
  848.       
  849.       ' Set dta address
  850.         dta$ = SPACE$(43)
  851.         reg.ax = &H1A00
  852.         reg.ds = VARSEG(dta$)
  853.         reg.dx = SADD(dta$)
  854.         InterruptX &H21, reg, reg
  855.       
  856.       ' Find first file match
  857.         reg.ax = &H4E00
  858.         reg.cx = FILEATTRIBUTE
  859.         reg.ds = VARSEG(thePath$)
  860.         reg.dx = SADD(thePath$)
  861.         InterruptX &H21, reg, reg
  862.       
  863.       ' The carry flag tells if a file was found or not
  864.         result% = reg.flags AND 1
  865.       
  866.       ' Reset the original DTA
  867.         reg.ax = &H1A00
  868.         reg.ds = sgmt%
  869.         reg.dx = ofst%
  870.         InterruptX &H21, reg, reg
  871.       
  872.     END SUB
  873.   
  874.   ' ************************************************
  875.   ' **  Name:          FindNextFile               **
  876.   ' **  Type:          Subprogram                 **
  877.   ' **  Module:        FILEINFO.BAS               **
  878.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  879.   ' ************************************************
  880.   '
  881.   ' Locates next file. FindFirstFile must be called
  882.   ' before this subprogram is called.
  883.   '
  884.     SUB FindNextFile (dta$, result%) STATIC
  885.       
  886.       ' Initialization
  887.         DIM reg AS RegTypeX
  888.       
  889.       ' Make sure dta$ was built (FindFirstFile should have been called)
  890.         IF LEN(dta$) <> 43 THEN
  891.             result% = 2
  892.             EXIT SUB
  893.         END IF
  894.       
  895.       ' Get current DTA address
  896.         reg.ax = &H2F00
  897.         InterruptX &H21, reg, reg
  898.         sgmt% = reg.es
  899.         ofst% = reg.bx
  900.       
  901.       ' Set dta address
  902.         reg.ax = &H1A00
  903.         reg.ds = VARSEG(dta$)
  904.         reg.dx = SADD(dta$)
  905.         InterruptX &H21, reg, reg
  906.       
  907.       ' Find next file match
  908.         reg.ax = &H4F00
  909.         reg.cx = FILEATTRIBUTE
  910.         reg.ds = VARSEG(thePath$)
  911.         reg.dx = SADD(thePath$)
  912.         InterruptX &H21, reg, reg
  913.       
  914.       ' The carry flag tells if a file was found or not
  915.         result% = reg.flags AND 1
  916.       
  917.       ' Reset the original DTA
  918.         reg.ax = &H1A00
  919.         reg.ds = sgmt%
  920.         reg.dx = ofst%
  921.         InterruptX &H21, reg, reg
  922.       
  923.     END SUB
  924.   
  925.   ' ************************************************
  926.   ' **  Name:          GetDirectory$              **
  927.   ' **  Type:          Function                   **
  928.   ' **  Module:        DOSCALLS.BAS               **
  929.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  930.   ' ************************************************
  931.   '
  932.   ' Returns the name of the current directory for any drive.
  933.   '
  934.     FUNCTION GetDirectory$ (drive$) STATIC
  935.         DIM regX AS RegTypeX
  936.         IF drive$ = "" THEN
  937.             d$ = GetDrive$
  938.         ELSE
  939.             d$ = UCASE$(drive$)
  940.         END IF
  941.         drive% = ASC(d$) - 64
  942.         regX.dx = drive%
  943.         regX.ax = &H4700
  944.         p$ = SPACE$(64)
  945.         regX.ds = VARSEG(p$)
  946.         regX.si = SADD(p$)
  947.         InterruptX &H21, regX, regX
  948.         p$ = LEFT$(p$, INSTR(p$, CHR$(0)) - 1)
  949.         GetDirectory$ = LEFT$(d$, 1) + ":\" + p$
  950.         IF regX.flags AND 1 THEN
  951.             GetDirectory$ = ""
  952.         END IF
  953.     END FUNCTION
  954.   
  955.   ' ************************************************
  956.   ' **  Name:          GetDiskFreeSpace           **
  957.   ' **  Type:          Subprogram                 **
  958.   ' **  Module:        DOSCALLS.BAS               **
  959.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  960.   ' ************************************************
  961.   '
  962.   ' Get information about a drive's organization, including
  963.   ' total number of bytes available.
  964.   '
  965.     SUB GetDiskFreeSpace (drive$, disk AS DiskFreeSpaceType)
  966.         DIM reg AS RegType
  967.         IF drive$ <> "" THEN
  968.             drive% = ASC(UCASE$(drive$)) - 64
  969.         ELSE
  970.             drive% = 0
  971.         END IF
  972.         IF drive% >= 0 THEN
  973.             reg.dx = drive%
  974.         ELSE
  975.             reg.dx = 0
  976.         END IF
  977.         reg.ax = &H3600
  978.         Interrupt &H21, reg, reg
  979.         disk.sectorsPerCluster = reg.ax
  980.         disk.bytesPerSector = reg.cx
  981.         IF reg.dx >= 0 THEN
  982.             disk.clustersPerDrive = reg.dx
  983.         ELSE
  984.             disk.clustersPerDrive = reg.dx + 65536
  985.         END IF
  986.         IF reg.bx >= 0 THEN
  987.             disk.availableClusters = reg.bx
  988.         ELSE
  989.             disk.availableClusters = reg.bx + 65536
  990.         END IF
  991.         disk.availableBytes = disk.availableClusters * reg.ax * reg.cx
  992.     END SUB
  993.   
  994.   ' ************************************************
  995.   ' **  Name:          GetDrive$                  **
  996.   ' **  Type:          Function                   **
  997.   ' **  Module:        DOSCALLS.BAS               **
  998.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  999.   ' ************************************************
  1000.   '
  1001.   ' Returns the current disk drive name, such as "A:"
  1002.   '
  1003.     FUNCTION GetDrive$ STATIC
  1004.         DIM reg AS RegType
  1005.         reg.ax = &H1900
  1006.         Interrupt &H21, reg, reg
  1007.         GetDrive$ = CHR$((reg.ax AND &HFF) + 65) + ":"
  1008.     END FUNCTION
  1009.   
  1010.   ' ************************************************
  1011.   ' **  Name:          GetFileData                **
  1012.   ' **  Type:          Subprogram                 **
  1013.   ' **  Module:        FILEINFO.BAS               **
  1014.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1015.   ' ************************************************
  1016.   '
  1017.   ' Extracts the file directory information from a Disk
  1018.   ' Transfer Area (dta$) that has been filled in by a
  1019.   ' call to either FindFirstFile or FindNextFile.
  1020.   '
  1021.     SUB GetFileData (dta$, file AS FileDataType) STATIC
  1022.       
  1023.         file.attribute = ASC(MID$(dta$, 22, 1))
  1024.         tim& = CVI(MID$(dta$, 23, 2))
  1025.         IF tim& < 0 THEN
  1026.             tim& = tim& + 65536
  1027.         END IF
  1028.         file.second = tim& AND &H1F
  1029.         file.minute = (tim& \ 32) AND &H3F
  1030.         file.hour = (tim& \ 2048) AND &H1F
  1031.         dat& = CVI(MID$(dta$, 25, 2))
  1032.         file.day = dat& AND &H1F
  1033.         file.month = (dat& \ 32) AND &HF
  1034.         file.year = ((dat& \ 512) AND &H1F) + 1980
  1035.         file.size = CVL(MID$(dta$, 27, 4))
  1036.         f$ = MID$(dta$, 31) + CHR$(0)
  1037.         file.finame = LEFT$(f$, INSTR(f$, CHR$(0)) - 1)
  1038.       
  1039.     END SUB
  1040.   
  1041.   ' ************************************************
  1042.   ' **  Name:          GetShiftStates             **
  1043.   ' **  Type:          Subprogram                 **
  1044.   ' **  Module:        BIOSCALL.BAS               **
  1045.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1046.   ' ************************************************
  1047.   '
  1048.   ' Returns state of the various shift keys and states.
  1049.   '
  1050.     SUB GetShiftStates (shift AS ShiftType) STATIC
  1051.         DIM reg AS RegType
  1052.         reg.ax = &H200
  1053.         Interrupt &H16, reg, reg
  1054.         shift.right = reg.ax AND 1
  1055.         shift.left = (reg.ax AND 2) \ 2
  1056.         shift.ctrl = (reg.ax AND 4) \ 4
  1057.         shift.alt = (reg.ax AND 8) \ 8
  1058.         shift.scrollLockState = (reg.ax AND 16) \ 16
  1059.         shift.numLockState = (reg.ax AND 32) \ 32
  1060.         shift.capsLockState = (reg.ax AND 64) \ 64
  1061.         shift.insertState = (reg.ax AND 128) \ 128
  1062.     END SUB
  1063.   
  1064.   ' ************************************************
  1065.   ' **  Name:          GetVerifyState%            **
  1066.   ' **  Type:          Function                   **
  1067.   ' **  Module:        DOSCALLS.BAS               **
  1068.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1069.   ' ************************************************
  1070.   '
  1071.     ' Returns the current state of the DOS "Verify After
  1072.     ' Write" flag.
  1073.   '
  1074.     FUNCTION GetVerifyState% STATIC
  1075.         DIM reg AS RegType
  1076.         reg.ax = &H5400
  1077.         Interrupt &H21, reg, reg
  1078.         GetVerifyState% = reg.ax AND &HFF
  1079.     END FUNCTION
  1080.   
  1081.   ' ************************************************
  1082.   ' **  Name:          InKeyCode%                 **
  1083.   ' **  Type:          Function                   **
  1084.   ' **  Module:        EDIT.BAS                   **
  1085.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1086.   ' ************************************************
  1087.   '
  1088.   ' Returns a unique integer for any key pressed, or
  1089.   ' a zero if no key has been pressed.
  1090.   '
  1091.     FUNCTION InKeyCode% STATIC
  1092.         InKeyCode% = CVI(INKEY$ + STRING$(2, 0))
  1093.     END FUNCTION
  1094.   
  1095.   ' ************************************************
  1096.   ' **  Name:          Julian2Date$               **
  1097.   ' **  Type:          Function                   **
  1098.   ' **  Module:        CALENDAR.BAS               **
  1099.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1100.   ' ************************************************
  1101.   '
  1102.   ' Returns a date in the QuickBASIC string format
  1103.   ' "MM-DD-YYYY" as calculated from a Julian day number.
  1104.   '
  1105.     FUNCTION Julian2Date$ (julian&) STATIC
  1106.       
  1107.         x& = 4 * julian& - 6884477
  1108.         y& = (x& \ 146097) * 100
  1109.         d& = (x& MOD 146097) \ 4
  1110.       
  1111.         x& = 4 * d& + 3
  1112.         y& = (x& \ 1461) + y&
  1113.         d& = (x& MOD 1461) \ 4 + 1
  1114.       
  1115.         x& = 5 * d& - 3
  1116.         m& = x& \ 153 + 1
  1117.         d& = (x& MOD 153) \ 5 + 1
  1118.       
  1119.         IF m& < 11 THEN
  1120.             month% = m& + 2
  1121.         ELSE
  1122.             month% = m& - 10
  1123.         END IF
  1124.         day% = d&
  1125.         year% = y& + m& \ 11
  1126.       
  1127.         dat$ = MDY2Date$(month%, day%, year%)
  1128.         Julian2Date$ = dat$
  1129.     END FUNCTION
  1130.   
  1131.   ' ************************************************
  1132.   ' **  Name:          KeyCode%                   **
  1133.   ' **  Type:          Function                   **
  1134.   ' **  Module:        EDIT.BAS                   **
  1135.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1136.   ' ************************************************
  1137.   '
  1138.   ' Returns a unique integer for any key pressed.
  1139.   '
  1140.     FUNCTION KeyCode% STATIC
  1141.         DO
  1142.             k$ = INKEY$
  1143.         LOOP UNTIL k$ <> ""
  1144.         KeyCode% = CVI(k$ + CHR$(0))
  1145.     END FUNCTION
  1146.   
  1147.   ' ************************************************
  1148.   ' **  Name:          MDY2Date$                  **
  1149.   ' **  Type:          Function                   **
  1150.   ' **  Module:        CALENDAR.BAS               **
  1151.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1152.   ' ************************************************
  1153.   '
  1154.   ' Converts month%, day%, and year% to a date string
  1155.   ' in the QuickBASIC string format "MM-DD-YYYY".
  1156.   '
  1157.     FUNCTION MDY2Date$ (month%, day%, year%) STATIC
  1158.         y$ = RIGHT$("000" + MID$(STR$(year%), 2), 4)
  1159.         m$ = RIGHT$("0" + MID$(STR$(month%), 2), 2)
  1160.         d$ = RIGHT$("0" + MID$(STR$(day%), 2), 2)
  1161.         MDY2Date$ = m$ + "-" + d$ + "-" + y$
  1162.     END FUNCTION
  1163.   
  1164.   ' ************************************************
  1165.   ' **  Name:          MonthName$                 **
  1166.   ' **  Type:          Function                   **
  1167.   ' **  Module:        CALENDAR.BAS               **
  1168.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1169.   ' ************************************************
  1170.   '
  1171.   ' Returns a string stating the month as indicated
  1172.   ' in dat$ (QuickBASIC string format "MM-DD-YYYY").
  1173.   '
  1174.     FUNCTION MonthName$ (dat$) STATIC
  1175.       
  1176.         IF LEN(dat$) <> 10 THEN
  1177.             dat$ = "MM-DD-YYYY"
  1178.         END IF
  1179.       
  1180.         SELECT CASE LEFT$(dat$, 2)
  1181.         CASE "01"
  1182.             MonthName$ = "January"
  1183.         CASE "02"
  1184.             MonthName$ = "February"
  1185.         CASE "03"
  1186.             MonthName$ = "March"
  1187.         CASE "04"
  1188.             MonthName$ = "April"
  1189.         CASE "05"
  1190.             MonthName$ = "May"
  1191.         CASE "06"
  1192.             MonthName$ = "June"
  1193.         CASE "07"
  1194.             MonthName$ = "July"
  1195.         CASE "08"
  1196.             MonthName$ = "August"
  1197.         CASE "09"
  1198.             MonthName$ = "September"
  1199.         CASE "10"
  1200.             MonthName$ = "October"
  1201.         CASE "11"
  1202.             MonthName$ = "November"
  1203.         CASE "12"
  1204.             MonthName$ = "December"
  1205.         CASE ELSE
  1206.             MonthName$ = "?MonthName?"
  1207.         END SELECT
  1208.       
  1209.     END FUNCTION
  1210.   
  1211.   ' ************************************************
  1212.   ' **  Name:          MouseMickey                **
  1213.   ' **  Type:          Subprogram                 **
  1214.   ' **  Module:        MOUSSUBS.BAS               **
  1215.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1216.   ' ************************************************
  1217.   '
  1218.   ' Read mouse mickey counts
  1219.   '
  1220.     SUB MouseMickey (horizontal%, vertical%) STATIC
  1221.         Mouse 11, 0, horizontal%, vertical%
  1222.     END SUB
  1223.   
  1224.   ' ************************************************
  1225.   ' **  Name:          OneMonthCalendar           **
  1226.   ' **  Type:          Subprogram                 **
  1227.   ' **  Module:        CALENDAR.BAS               **
  1228.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1229.   ' ************************************************
  1230.   '
  1231.   ' Prints a small, one month calendar at the row%
  1232.   ' and col% indicated.
  1233.   '
  1234.     SUB OneMonthCalendar (dat$, row%, col%) STATIC
  1235.         mname$ = MonthName$(dat$)
  1236.         LOCATE row%, col% + 12 - LEN(mname$) \ 2
  1237.         PRINT mname$; ","; Date2Year%(dat$)
  1238.         month% = Date2Month%(dat$)
  1239.         day% = 1
  1240.         year% = Date2Year%(dat$)
  1241.         dat1$ = MDY2Date$(month%, day%, year%)
  1242.         j& = Date2Julian&(dat1$)
  1243.         heading$ = " Sun Mon Tue Wed Thu Fri Sat"
  1244.         wa% = INSTR(heading$, LEFT$(DayOfTheWeek$(dat1$), 3)) \ 4
  1245.         LOCATE row% + 1, col%
  1246.         PRINT heading$
  1247.         rowloc% = row% + 2
  1248.         LOCATE rowloc%, col% + 4 * wa%
  1249.         DO
  1250.             PRINT USING "####"; day%;
  1251.             IF wa% = 6 THEN
  1252.                 rowloc% = rowloc% + 1
  1253.                 LOCATE rowloc%, col%
  1254.             END IF
  1255.             wa% = (wa% + 1) MOD 7
  1256.             j& = j& + 1
  1257.             day% = Date2Day%(Julian2Date$(j&))
  1258.         LOOP UNTIL day% = 1
  1259.         PRINT
  1260.     END SUB
  1261.   
  1262.   ' ************************************************
  1263.   ' **  Name:          SpaceWorms                 **
  1264.   ' **  Type:          Subprogram                 **
  1265.   ' **  Module:        INTRODEM.BAS               **
  1266.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1267.   ' ************************************************
  1268.   '
  1269.   ' A simple game that demonstrates object creation and
  1270.   ' collision detection routines.
  1271.   '
  1272.     SUB SpaceWorms STATIC
  1273.       
  1274.       ' Constants
  1275.         CONST WORMS = 2
  1276.         CONST WSEGS = 999
  1277.         CONST PI = 3.141593
  1278.         CONST RADIUS = 2
  1279.         CONST MICKFACTOR = 5
  1280.         CONST DOWNARROW = 20480
  1281.         CONST LEFTARROW = 19200
  1282.         CONST RIGHTARROW = 19712
  1283.         CONST UPARROW = 18432
  1284.         CONST HOMEKEY = 18176
  1285.         CONST ENDKEY = 20224
  1286.         CONST PGUP = 18688
  1287.         CONST PGDN = 20736
  1288.       
  1289.       ' The worm ship arrays were built in module level code
  1290.         SHARED wormship1%(), wormship2%()
  1291.       
  1292.       ' Initialization
  1293.         SCREEN 1
  1294.         COLOR BLUE, 1
  1295.         CLS
  1296.         RANDOMIZE TIMER
  1297.       
  1298.       ' Dimension arrays for worm data
  1299.         REDIM row%(WORMS, WSEGS - 1)
  1300.         REDIM col%(WORMS, WSEGS - 1)
  1301.         REDIM h(WORMS)
  1302.         REDIM p%(WORMS)
  1303.         REDIM bgd%(0 TO 54)
  1304.       
  1305.       ' Starting conditions
  1306.         t% = 0
  1307.         timeStart = TIMER
  1308.         t0 = timeStart
  1309.         n% = 0
  1310.         nseg% = 20
  1311.         zapped% = 0
  1312.         wx% = 160
  1313.         wy% = 100
  1314.         newx% = wx%
  1315.         newy% = wy%
  1316.       
  1317.       ' Put starting ship image on the screen
  1318.         PUT (wx%, wy%), wormship1%
  1319.       
  1320.       ' Set starting random worm headings
  1321.         FOR i% = 0 TO WORMS
  1322.             h(i%) = RND * PI
  1323.         NEXT i%
  1324.       
  1325.       ' Main loop
  1326.         DO
  1327.           
  1328.           ' Check for mouse movement
  1329.             MouseMickey horizontal%, vertical%
  1330.             newx% = (newx% + 293 + horizontal% / MICKFACTOR) MOD 293
  1331.             newy% = (newy% + 185 + vertical% / MICKFACTOR) MOD 185
  1332.           
  1333.           ' Check for cursor keys
  1334.             SELECT CASE InKeyCode%
  1335.             CASE UPARROW
  1336.                 newy% = (newy% + 175) MOD 185
  1337.             CASE DOWNARROW
  1338.                 newy% = (newy% + 10) MOD 185
  1339.             CASE LEFTARROW
  1340.                 newx% = (newx% + 283) MOD 293
  1341.             CASE RIGHTARROW
  1342.                 newx% = (newx% + 10) MOD 293
  1343.             CASE HOMEKEY
  1344.                 newy% = (newy% + 175) MOD 185
  1345.                 newx% = (newx% + 283) MOD 293
  1346.             CASE ENDKEY
  1347.                 newy% = (newy% + 10) MOD 185
  1348.                 newx% = (newx% + 283) MOD 293
  1349.             CASE PGUP
  1350.                 newx% = (newx% + 10) MOD 293
  1351.                 newy% = (newy% + 175) MOD 185
  1352.             CASE PGDN
  1353.                 newx% = (newx% + 10) MOD 293
  1354.                 newy% = (newy% + 10) MOD 185
  1355.             CASE ELSE
  1356.             END SELECT
  1357.           
  1358.           ' Draw the worm ship
  1359.             t% = NOT t%
  1360.             IF t% THEN
  1361.                 PUT (wx%, wy%), wormship1%
  1362.                 wx% = newx%
  1363.                 wy% = newy%
  1364.                 GET (wx%, wy%)-(wx% + 23, wy% + 14), bgd%
  1365.                 PUT (wx%, wy%), wormship2%
  1366.                 zapped% = Collision%(wormship2%(), bgd%())
  1367.             ELSE
  1368.                 PUT (wx%, wy%), wormship2%
  1369.                 wx% = newx%
  1370.                 wy% = newy%
  1371.                 GET (wx%, wy%)-(wx% + 23, wy% + 14), bgd%
  1372.                 PUT (wx%, wy%), wormship1%
  1373.                 zapped% = Collision%(wormship1%(), bgd%())
  1374.             END IF
  1375.           
  1376.           
  1377.           ' Sprout new worm or add segments every five seconds
  1378.             IF TIMER - t0 > 5 THEN
  1379.                 IF n% < WORMS THEN
  1380.                     n% = n% + 1
  1381.                 ELSE
  1382.                     IF nseg% < WSEGS THEN
  1383.                         nseg% = nseg% + 5
  1384.                     END IF
  1385.                 END IF
  1386.                 t0 = TIMER
  1387.             END IF
  1388.           
  1389.           ' Randomly adjust the worm headings
  1390.             FOR i% = 0 TO n%
  1391.                 h(i%) = h(i%) + RND - .5
  1392.             NEXT i%
  1393.           
  1394.           ' Move each worm
  1395.             FOR i% = 0 TO n%
  1396.                 p2% = p%(i%)
  1397.                 p1% = (p2% + nseg% - 1) MOD nseg%
  1398.                 p3% = (p2% + nseg% + 1) MOD nseg%
  1399.                 CIRCLE (row%(i%, p2%), col%(i%, p2%)), RADIUS, 0
  1400.                 row%(i%, p2%) = (row%(i%, p1%) + COS(h(i%)) * 5 + 320) MOD 320
  1401.                 col%(i%, p2%) = (col%(i%, p1%) + SIN(h(i%)) * 5 + 200) MOD 200
  1402.                 CIRCLE (row%(i%, p2%), col%(i%, p2%)), RADIUS, 3
  1403.                 p%(i%) = p3%
  1404.             NEXT i%
  1405.           
  1406.         LOOP UNTIL zapped%
  1407.       
  1408.       ' Display results
  1409.         LOCATE 1, 1
  1410.         elapsed = TIMER - timeStart
  1411.         PRINT USING "You stayed alive for####.# seconds!"; elapsed
  1412.         PRINT "Press <Enter> to continue..."
  1413.         DO
  1414.             backGround% = (backGround% + 1) MOD 8
  1415.             COLOR backGround%
  1416.             t0 = TIMER
  1417.             DO
  1418.             LOOP WHILE TIMER - t0 < .3
  1419.         LOOP UNTIL INKEY$ = CHR$(13)
  1420.       
  1421.       ' All done
  1422.         SCREEN 0
  1423.         WIDTH 80
  1424.       
  1425.     END SUB
  1426.   
  1427.   ' ************************************************
  1428.   ' **  Name:          VideoState                 **
  1429.   ' **  Type:          Subprogram                 **
  1430.   ' **  Module:        BIOSCALL.BAS               **
  1431.   ' **  Language:      Microsoft QuickBASIC 4.00  **
  1432.   ' ************************************************
  1433.   '
  1434.   ' Determines the current video mode parameters.
  1435.   '
  1436.     SUB VideoState (mode%, columns%, page%) STATIC
  1437.         DIM reg AS RegType
  1438.         reg.ax = &HF00
  1439.         Interrupt &H10, reg, reg
  1440.         mode% = reg.ax AND &HFF
  1441.         columns% = (CLNG(reg.ax) AND &HFF00) \ 256
  1442.         page% = (CLNG(reg.bx) AND &HFF00) \ 256
  1443.     END SUB
  1444.   
  1445.