home *** CD-ROM | disk | FTP | other *** search
- EXTERNAL
-
- SUB Format (line$(),arg$)
-
- ! FORMAT
- !
- ! a True BASIC(tm), Inc. product
- !
- ! ABSTRACT
- ! Indents a True BASIC program, and
- ! capitalizes the leading keywords
- ! and certain other keywords.
- !
- ! SYNTAX
- ! DO FORMAT [ , NOCAP ]
- !
- ! Copyright (c) 1985, 1987 by True BASIC, Incorporated
-
- LET mincom = 35 ! Comments start here
- LET newind = 0 ! Indentation for next line
- LET numflag = 0 ! Line number flag
- LET c$ = line$(1)[1:1] ! Are there line numbers?
- IF c$ >= "0" and c$ <= "9" then LET numflag = 1 else LET num$ = ""
- IF arg$ <> "" then
- IF pos("NOCAP",ucase$(arg$)) <> 1 then
- CAUSE ERROR 1, "Try ""DO FORMAT"" or ""DO FORMAT, NOCAP"""
- END IF
- LET nocap = 1
- END IF
-
- FOR i = 1 to ubound(line$)
- LET l$ = line$(i)
- CALL trisect ! Divide l$ into num$, text$, com$
- LET utext$ = ucase$(text$) ! For comparing and capitalizing
- LET p2 = 0 ! Character pointer for keyword
- CALL keyword ! Find command and capitalize
- LET ind = newind
- IF key$ = "IF" then CALL onelineif
- CALL indent ! Calculate amount of indentation
- CALL join ! Put line together
- LET line$(i) = l$
- NEXT i
-
- DECLARE DEF posnq
-
- SUB trisect ! Divide into num$, text$, com$
-
- IF numflag = 1 then
- CALL getlinenum (l$, num$, text$)
- LET num$ = num$ & " "
- ELSE
- LET text$ = l$
- END IF
-
- CALL sepcom (text$, com$)
-
- LET text$ = rtrim$(ltrim$(text$)) ! Stripped body
-
- END SUB
-
- SUB keyword
-
- ! Find keyword(s) starting at p
- ! key$ = keywords
- ! Capitalize them
-
- CALL nextword (utext$, p, p2, key$) ! Move pointers to next word
-
- LET c = ncpos(utext$, " ", p2+1) ! find next non-blank
- LET c = pos(" =(,",utext$[c:c]) ! see if possible assignment
- IF c > 1 then
- SELECT CASE key$ ! not possible if reserved
- CASE "IF", "ELSE", "ELSEIF", "PRINT", "REM", "DATA"
- CASE else ! insert LET if "=" follows
- IF posnq(utext$,"=",p2+1) > 0 then
- LET utext$[p:0] = "LET " ! insert in both copies
- LET text$[p:0] = "let " ! maybe leave it lowercase
- LET p2 = p + 2
- LET key$ = "LET"
- END IF
- END SELECT
- END IF
-
- SELECT CASE key$
-
- CASE "SELECT", "LINE", "BOX", "OPTION", "EXIT", "CAUSE", "DECLARE"
- CALL addkey ! Additional keyword
-
- CASE "PLOT"
- LET c2 = p2
- CALL nextword (utext$, c, c2, x$) ! Look at next word
- SELECT CASE x$
- CASE "POINTS", "LINES", "AREA"
- LET p2 = c2
- CASE "TEXT"
- LET p2 = c2
- CALL addkey ! also add "AT"
- CASE else
- END SELECT
-
- CASE "END"
- IF p2 < len(utext$) then
- CALL addkey ! Not simply END
- LET key$ = key$ & " " & x$ ! Make sure one space
- END IF
-
- CASE "MAT"
- LET c2 = p2
- CALL nextword (utext$, c, c2, x$) ! Look at next word
- SELECT CASE x$
- CASE "INPUT", "PRINT", "READ", "WRITE"
- LET p2 = c2 ! Add second word
- CASE "LINE", "PLOT"
- LET p2 = c2
- CALL addkey ! Third word
- CASE else ! Nothing to do
- END SELECT
-
- CASE "GET"
- CALL addkey
-
- CASE "SET", "ASK"
- CALL setask
-
- CASE "ELSE" ! Handle ELSE IF
- CALL addkey ! Must be IF, if anything
- LET key$ = key$ & x$ ! Treat like ELSEIF
-
- CASE else ! Nothing to do
-
- END SELECT
-
- IF nocap = 0 then
- LET text$[p:p2] = utext$[p:p2] ! Capitalize keywords(s)
- END IF
-
- END SUB
-
- SUB addkey ! Add next word to keyword
-
- CALL nextword (utext$, c, p2, x$) ! Move p2 only
-
- END SUB
-
- SUB setask ! Check for SET COLOR MIX, etc.
-
- LET c = posnq (utext$, ":", p2)
- IF c = 0 then ! Ordinary SET or ASK
- CALL addkey
- IF x$ = "COLOR" then ! Look for SET COLOR MIX
- LET c2 = p2
- CALL nextword (utext$, c, c2, x$)
- IF x$ = "MIX" then LET p2 = c2
- ELSE IF x$ = "MAX" or x$ = "BACKGROUND" or x$ = "FREE" then
- CALL addkey ! Allow for ASK MAX XXXXX
- END IF
- ELSE ! File SET / ASK
- LET c2 = c+1 ! Find word AFTER #n:
- CALL nextword (utext$, c, c2, x$)
- IF nocap = 0 then
- LET text$[c:c2] = utext$[c:c2] ! Capitalize it
- END IF
- END IF
-
- END SUB
-
- SUB onelineif ! Checks for keywords in one-line if-then
-
- ! Isoneline = 1 if a one-line if-then. Used in indent
- LET isoneline = 0
-
- LET savedkey$ = key$
-
- LET p = posnq (utext$, "THEN", p) ! Locate THEN
-
- LET p2 = p + 4
- IF p2 <= len(utext$) then ! One line if-then
- LET isoneline = 1
- CALL keyword ! Capitalize it
- LET p = posnq (utext$, "ELSE", p) ! Look for ELSE
- IF p <> 0 then
- LET p2 = p + 4 ! Start of word to capitalize
- CALL keyword ! Capitalize it
- END IF
- END IF
- LET key$ = savedkey$
-
- END SUB
-
- SUB indent ! Calculate number of spaces to indent
-
- SELECT CASE key$
-
- CASE "DO"
- LET newind = ind + 3
-
- CASE "IF"
- IF isoneline = 0 then LET newind = ind + 3
-
- CASE "FOR", "SUB", "PICTURE", "MODULE"
- LET newind = ind + 4
-
- CASE "DEF", "FUNCTION"
- IF posnq (utext$, "=", p) = 0 then LET newind = ind + 4
-
- CASE "SELECT", "WHEN"
- LET newind = ind + 5
-
- CASE "LOOP", "END IF" ! Structure closers
- LET newind, ind = ind - 3
-
- CASE "NEXT", "END SUB", "END DEF", "END PICTURE", "END FUNCTION", "END MODULE"
- LET newind, ind = ind - 4
-
- CASE "END SELECT", "END WHEN"
- LET newind, ind = ind - 5
-
- CASE "ELSE", "ELSEIF" ! In middle
- LET ind = ind - 3
-
- CASE "CASE", "USE"
- LET ind = ind - 5
-
- CASE else ! Nothing to do
-
- END SELECT
-
- END SUB
-
- SUB join ! Put line together
-
- IF ind < 0 then LET ind = 0 ! To prevent abort
-
- IF text$ <> "" then ! Normal case
-
- LET l$ = num$ & repeat$(" ", ind) & text$ ! Line number and indented text
- IF com$ <> "" then ! Must add comment
- LET l = len(l$)
- LET l2 = l + 3 ! At least 3 spaces
- LET l2 = max(mincom,l2) ! At least mincom
- LET r2 = mod(l2,5)
- IF r2 > 0 then LET l2 = l2 + 5 - r2 ! Multiple of 5
- LET l$ = l$ & repeat$(" ", l2-l-1) & com$
- END IF
- ELSE ! No text
- LET l$ = num$
- IF com$ <> "" then LET l$ = l$ & repeat$(" ", ind) & com$
- END IF
-
- END SUB
-
- END SUB ! Subroutine format
-
- SUB getlinenum (line$, num$, rest$)
-
- LET i = ncpos(line$,"0123456789")
- IF i = 0 then LET i = maxnum
-
- LET num$ = line$[1:i-1]
- LET rest$ = line$[i:maxnum]
-
- END SUB
-
- SUB sepcom (line$, comment$)
-
- ! Separate on-line comment, if any
-
- DECLARE DEF posnq
-
- LET p = posnq (line$, "!", 1)
- IF p = 0 then
- LET comment$ = ""
- ELSE
- LET comment$ = line$[p:maxnum]
- LET line$ = line$[1:p-1]
- END IF
-
- END SUB
-
- SUB nextword (line$, p1, p2, word$)
-
- ! word$ = next word AFTER character p2
- ! p1 points to begin, p2 to end of word$
-
- LET p1 = ncpos(line$ & "x", " ,:=(;+-""#", p2 + 1)
- LET p2 = cpos(line$ & " ", " ,:=(;+-""#", p1) - 1
-
- LET word$ = line$[p1:p2]
-
- END SUB
-
- DEF posnq (line$, s$, p)
-
- ! Search for first occurrence, on or after p, of s$ in line$,
- ! not in quotes. Return position, or 0 if none.
-
- DO
- LET p, posnq = pos(line$, s$, p)
- IF p = 0 then EXIT DEF
- DO
- LET q = pos(line$, """", q+1)
- IF q = 0 then EXIT DEF ! call it unquoted if no " follows
- LET unq = 1 - unq
- LOOP until q > p
- LET p = p + 1 ! prepare to find next occurence
- LOOP until unq = 1 ! unless this one was unquoted
-
- END DEF
-