home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 5 / 05.iso / a / a031 / samples.exe / LIBRARY.PRG < prev    next >
Encoding:
Text File  |  1992-03-10  |  43.3 KB  |  1,254 lines

  1. ******************************************************************************
  2. * PROGRAM NAME: LIBRARY.PRG
  3. *               LIBRARY OF PROCEDURES COMMON TO ALL BUSINESS PROGRAMS
  4. *               SAMPLE BUSINESS APPLICATION PROGRAM
  5. * LAST CHANGED: 06/20/90 8:00AM
  6. * WRITTEN BY:   Borland International Inc.
  7. ******************************************************************************
  8.  
  9. PROCEDURE Add_new
  10.    * Add new record to database file
  11.    * Erase previous record number from screen
  12.    @ 0,65 SAY SPACE(15) COLOR &c_yellow.
  13.    * Display F9 lookup key message, if lookup available
  14.    IF lookup_ok
  15.       DO Sho_look WITH dbf
  16.    ENDIF
  17.    DO Init_fld
  18.    DO Get_data
  19.    READ
  20.    * Erase lookup message from screen
  21.    @ 0,0 SAY SPACE(51)
  22.    * If user didn't enter data into key fields, exit without saving
  23.    IF "" = TRIM(&key.) .OR. READKEY() < 256
  24.        RETURN
  25.    ELSE
  26.       * Each application checks for duplicates if duplicate keys not allowed
  27.       * If duplicate key (when not allowed), exit from add mode without saving
  28.       IF rec_is_dup
  29.          * Reset status flag and exit
  30.          rec_is_dup = .F.
  31.          RETURN
  32.       ELSE
  33.          * Append and save validated record
  34.          DO Sav_data
  35.          GO record_num
  36.       ENDIF
  37.    ENDIF
  38. RETURN
  39.  
  40. PROCEDURE Bar_def
  41.    * Define the main popup OPTION MENU, main_mnu
  42.    mesg = "Press first letter of Menu choice, or highlight and press <Enter>"
  43.    DEFINE POPUP main_mnu FROM 2,58 TO 22,78 MESSAGE mesg
  44.    DEFINE BAR  1 OF main_mnu PROMPT "==  OPTION MENU  ==" SKIP
  45.    DEFINE BAR  2 OF main_mnu PROMPT " Add record"
  46.    DEFINE BAR  3 OF main_mnu PROMPT " Edit record"
  47.    DEFINE BAR  4 OF main_mnu PROMPT " Delete record"
  48.    DEFINE BAR  5 OF main_mnu PROMPT "-------------------" SKIP
  49.    DEFINE BAR  6 OF main_mnu PROMPT " Next record"
  50.    DEFINE BAR  7 OF main_mnu PROMPT " Previous record"
  51.    DEFINE BAR  8 OF main_mnu PROMPT " Top record"
  52.    DEFINE BAR  9 OF main_mnu PROMPT " Bottom record"
  53.    DEFINE BAR 10 OF main_mnu PROMPT " Skip records"
  54.    DEFINE BAR 11 OF main_mnu PROMPT " Find record"
  55.    DEFINE BAR 12 OF main_mnu PROMPT "-------------------" SKIP
  56.    DEFINE BAR 13 OF main_mnu PROMPT " List records"
  57.    DEFINE BAR 14 OF main_mnu PROMPT " Output reports"
  58.    DEFINE BAR 15 OF main_mnu PROMPT " Group records" SKIP FOR dbf = "ACCT_REC"
  59.    DEFINE BAR 16 OF main_mnu PROMPT " Count records"
  60.    DEFINE BAR 17 OF main_mnu PROMPT " Index database"
  61.    DEFINE BAR 18 OF main_mnu PROMPT " Help"
  62.    DEFINE BAR 19 OF main_mnu PROMPT " Quit to MAIN MENU"
  63.    * Define the popup dest_mnu for printing reports to a destination
  64.    DEFINE POPUP dest_mnu FROM 13,10 TO 19,38 MESSAGE mesg
  65.    DEFINE BAR 1 OF dest_mnu PROMPT "======= DESTINATION =======" SKIP
  66.    DEFINE BAR 2 OF dest_mnu PROMPT " Printer"
  67.    DEFINE BAR 3 OF dest_mnu PROMPT " File"
  68.    DEFINE BAR 4 OF dest_mnu PROMPT " Screen"
  69.    DEFINE BAR 5 OF dest_mnu PROMPT " Exit to OPTION MENU"
  70.    * Define the popup rpt_mnu for printing reports to a destination
  71.    DEFINE POPUP rpt_mnu FROM 11, 5 TO 17,38 MESSAGE mesg
  72.    DEFINE BAR 1 OF rpt_mnu  PROMPT "============ REPORTS ===========" SKIP
  73.    DEFINE BAR 2 OF rpt_mnu  PROMPT " Database report: " + dbf
  74.    DEFINE BAR 3 OF rpt_mnu  PROMPT " Mailing list: "  + mlist ;
  75.       SKIP FOR mlist = "NOT AVAILABLE"
  76.    DEFINE BAR 4 OF rpt_mnu  PROMPT " Custom programmed report: " + cust_rpt ;
  77.       SKIP FOR cust_rpt = "N/A"
  78.    DEFINE BAR 5 OF rpt_mnu  PROMPT " Exit to OPTION MENU"
  79.    * Define which procedures are executed by the defined popups
  80.    ON SELECTION POPUP main_mnu DO Barpop
  81.    ON SELECTION POPUP rpt_mnu  DO Barpop_r
  82.    ON SELECTION POPUP dest_mnu DO Barpop_d
  83.    * Define windows for text, msgs, etc.
  84.    DEFINE WINDOW alert      FROM 15, 3 TO 22,46 PANEL COLOR &c_alert.
  85.    DEFINE WINDOW duplicat   FROM 15, 5 TO 21,70 PANEL COLOR &c_alert.
  86.    DEFINE WINDOW lister     FROM  5, 5 TO 22,70 PANEL COLOR &c_list.
  87.    DEFINE WINDOW look       FROM  6, 5 TO 16,65 PANEL COLOR &c_list.
  88.    DEFINE WINDOW memo_windo FROM  7, 4 TO 19,75 PANEL COLOR &c_list.
  89. RETURN
  90.  
  91. PROCEDURE Barpop
  92.    * Perform action selected by user from OPTION MENU bars
  93.    DO CASE
  94.        * BAR() = 1 is title of menu
  95.        CASE BAR() = 2                  && Add record
  96.           DO Add_new
  97.        CASE BAR() = 3                  && Edit record
  98.           DO Edit
  99.        CASE BAR() = 4                  && Delete record
  100.           DO Eraser
  101.        CASE BAR() = 6                  && Next record
  102.           DO Skip_rec WITH 1
  103.        CASE BAR() = 7                  && Previous record
  104.           DO Skip_rec WITH -1
  105.        CASE BAR() = 8                  && Top record, in active index order
  106.           GO TOP
  107.        CASE BAR() = 9                  && Bottom record, in active index order
  108.           GO BOTTOM
  109.        CASE BAR() = 10                 && Skip records
  110.           DO Skip_rec WITH 0
  111.        CASE BAR() = 11                 && Find record
  112.           DO Find_rec WITH key, key1, keyname1, key2, keyname2, key3, keyname3
  113.        CASE BAR() = 13                 && List records
  114.           DO List_rec
  115.        CASE BAR() = 14                 && Output reports
  116.           SAVE SCREEN TO Pre_rept      && Save screen image
  117.           ACTIVATE POPUP rpt_mnu
  118.           RESTORE SCREEN FROM Pre_rept
  119.           RELEASE SCREEN Pre_rept
  120.        CASE BAR() = 15              && Group records
  121.           DO Filter
  122.        CASE BAR() = 16                 && Count records
  123.           ************
  124.           IF NETWORK()
  125.              * Turn off file lock to count
  126.              SET LOCK off
  127.              DO Kount
  128.              SET LOCK on
  129.              ***********
  130.           ELSE
  131.              DO Kount
  132.           ENDIF
  133.        CASE BAR() = 17                  && Index database
  134.           ************
  135.           IF NETWORK()
  136.              old_tag = ORDER()
  137.              USE (dbf) EXCLUSIVE
  138.              IF net_choice <> 27        && check Net_err user choice (Esc=27)
  139.                 DO Indexer
  140.                 SET EXCLUSIVE off
  141.                 USE (dbf) ORDER (old_tag)
  142.              ENDIF
  143.              ***********************
  144.           ELSE
  145.              DO Indexer
  146.           ENDIF
  147.        CASE BAR() = 18                  && Help
  148.           SET COLOR TO &c_standard.
  149.           DO Helper
  150.        CASE BAR() = 19                && Quit to Main Menu
  151.           DEACTIVATE POPUP
  152.    ENDCASE
  153.    DO Dstatus                         && Display record no and filter status
  154.    DO Show_data                       && Display screen with current record
  155.    CLEAR GETS
  156.    SET COLOR TO &c_popup.
  157. RETURN
  158.  
  159. PROCEDURE Barpop_d
  160.    * Perform action selected by user from Destination menu
  161.    SET COLOR TO &c_popup.
  162.    DO CASE
  163.       * BAR() 1 is title of menu
  164.       CASE BAR() = 2                  && Output to printer
  165.          ll_esc = .F.
  166.          DO Prt_menu                  && Activate menu for print options
  167.          IF .NOT. ll_esc
  168.             SET PRINTER on
  169.             SET CONSOLE off
  170.             DO Printout               && Output selected report
  171.             SET PRINTER off
  172.             SET CONSOLE on
  173.          ENDIF
  174.       CASE BAR() = 3                  && Output to file
  175.          answer = SPACE(8)
  176.          ACTIVATE WINDOW alert
  177.             @ 0,0 SAY "----------- SEND REPORT TO FILE ----------"
  178.             @ 2,1 SAY "Enter filename for report: " GET answer ;
  179.                VALID "" <> TRIM(answer) ;
  180.                MESSAGE "Enter a filename of up to eight characters"
  181.             READ
  182.          DEACTIVATE WINDOW alert
  183.          SET ALTERNATE TO &answer.
  184.          SET ALTERNATE on
  185.          SET CONSOLE off
  186.          GO TOP
  187.          DO Printout                  && Output report or labels to file
  188.          SET ALTERNATE off
  189.          SET CONSOLE on
  190.       CASE BAR() = 4                  && Output to screen
  191.          SET COLOR TO &c_standard.
  192.          CLEAR
  193.          * Store current page settings
  194.          plength  = _plength
  195.          rmargin  = _rmargin
  196.          * Set page width & length for screen
  197.          _plength = 25
  198.          _rmargin = 80
  199.          DO Printout                  && Output chosen report/labels to screen
  200.          CLEAR
  201.          * Reset page settings
  202.          _plength = plength
  203.          _rmargin = rmargin
  204.          GO record_num                && Return to original record
  205.       CASE BAR() = 5                  && Exit to OPTION MENU
  206.          DEACTIVATE POPUP
  207.    ENDCASE
  208.    SET COLOR TO &c_standard.
  209.    DEACTIVATE POPUP
  210. RETURN
  211.  
  212. PROCEDURE Barpop_r
  213.    * Select available reports menu
  214.    SET COLOR TO &c_popup.
  215.    reportype = SPACE(6)
  216.    DO CASE
  217.       CASE BAR() = 2                  && Output standard report to destination
  218.          reportype = "LISTING"
  219.          ACTIVATE POPUP dest_mnu      && Activate printer destination menu
  220.       CASE BAR() = 3                  && Output mailing labels to destination
  221.          reportype = "LABELS"
  222.          ACTIVATE POPUP dest_mnu      && Activate printer destination menu
  223.       CASE BAR() = 4                  && Output custom report to destination
  224.          reportype = "CUSTOM"
  225.          ACTIVATE WINDOW alert
  226.             * Get custom report name from user
  227.             * First, allow READ errors and warning bell
  228.             ON READERROR
  229.             SET BELL ON
  230.             rpt_name = SPACE(8)
  231.             @ 0,0 SAY "-------- CUSTOM PROGRAMMED REPORT --------"
  232.             @ 2,1 SAY "Enter report program name:" GET rpt_name ;
  233.                VALID FILE(TRIM(rpt_name) + ".prg") ;
  234.                MESSAGE "Enter a filename of up to eight " + ;
  235.                        "characters, e.g. Emp_rept " ;
  236.                ERROR "Invalid filename, please re-enter"
  237.             READ
  238.             * Now, put the READ error redirection back.
  239.             ON READERROR
  240.             SET BELL OFF
  241.          DEACTIVATE WINDOW alert
  242.          IF LASTKEY() <> 27           && A report filename was found
  243.             SET COLOR TO &c_popup.
  244.             ACTIVATE POPUP dest_mnu
  245.          ENDIF
  246.    ENDCASE
  247.    SET COLOR TO &c_popup.
  248.    DEACTIVATE POPUP
  249. RETURN
  250.  
  251. PROCEDURE Sub_ret
  252.    IF erased
  253.       * Pack deleted records (if any) - erases completely from database
  254.       ************
  255.       IF NETWORK()
  256.          USE (dbf) EXCLUSIVE
  257.       ENDIF
  258.       IF net_choice <> 27       && Skip if user pressed Esc
  259.       *******************       && error condition
  260.          ?? CHR(7)
  261.          ACTIVATE WINDOW alert
  262.             @ 0,0 SAY "----------- PACKING  DATABASE ------------"
  263.             @ 2,1 SAY "ERASING deleted records now......"
  264.             @ 3,1 SAY "Please wait......DO NOT TURN OFF"
  265.             PACK
  266.          DEACTIVATE WINDOW alert
  267.       ENDIF
  268.    ENDIF
  269.    * Houskeeping
  270.    CLOSE DATABASES
  271.    CLEAR WINDOWS
  272.    RELEASE ALL
  273.    CLEAR
  274.    ON ERROR
  275.    ON KEY LABEL F9             && Turn off ON KEY LABEL F9/F10 commands
  276.    ON KEY LABEL F10
  277.    * Restore environment (in case user began at Control Center or dot prompt)
  278.    DO Rest_env
  279.    CLEAR
  280. RETURN TO MASTER               && Exit Subapplication
  281.  
  282. FUNCTION Duplicat
  283.    PARAMETERS key
  284.    * Used if duplicates are not allowed in a database
  285.    * Set rec_is_dup to .T. if user entered duplicate key data
  286.    rec_is_dup = .F.
  287.    IF RECCOUNT() = 0 .OR. "" = TRIM(key)
  288.       * Do not check if database or key field(s) is empty
  289.       RETURN rec_is_dup
  290.    ENDIF
  291.    record_num = RECNO()               && Save current record position
  292.    SEEK  TRIM(key)
  293.    * Determine if record is duplicate key
  294.    * PROMPT() used instead of BAR() for clarity
  295.    DO CASE
  296.       CASE PROMPT() = " Edit record"
  297.          * If seek finds a record other than the current one,
  298.          * the edited record has a duplicate key
  299.          rec_is_dup =  record_num <> RECNO() .AND. FOUND()
  300.       CASE PROMPT() = " Add record"
  301.          * New record is duplicate if seek finds any record that matches
  302.          rec_is_dup = FOUND()
  303.    ENDCASE
  304.    IF rec_is_dup                      && Show duplicate record in window
  305.       ACTIVATE WINDOW duplicat
  306.          CLEAR
  307.          DO Warnbell
  308.          ?  "------------------ DUPLICATE " + dbf + ;
  309.             " RECORD ------------------"
  310.          ?  "                    Duplicates not allowed"
  311.          DO CASE
  312.             CASE dbf = "CUST"
  313.                ?  " " + cust_id + " " + customer
  314.                ? "This is the EXISTING record in the database; " + ;
  315.                  "re-enter Cust.ID."
  316.             CASE dbf = "VENDORS"
  317.                ?  " " + vendor_id + " " + vendor
  318.                ? "This is the EXISTING record in the database; " + ;
  319.                  "re-enter Vendor ID."
  320.             CASE dbf = "GOODS"
  321.                ?  " " + part_id + " " + part_name
  322.                ? "This is the EXISTING record in the database; " + ;
  323.                  "re-enter Part ID."
  324.             CASE dbf = "ACCT_REC"
  325.                ?  " " + invoice_no + " " + cust_id + " " + DTOC(dat_of_bil)
  326.                ? "This is the EXISTING record in the database; " + ;
  327.                  "re-enter Invoice ID."
  328.          ENDCASE
  329.          WAIT "     Press spacebar to continue..."
  330.       DEACTIVATE WINDOW duplicat
  331.    ENDIF
  332.    GO record_num                     && Return to original record
  333. RETURN .NOT. rec_is_dup
  334.  
  335. PROCEDURE Dstatus
  336.    * Display filter status and current record number
  337.    * Set colors with blink on/off depending on hardware
  338.    IF filters_on
  339.       * Show blinking msg for filter status
  340.       @ 0,51 SAY "Filter is ON" COLOR &c_blink.
  341.    ELSE
  342.       SET COLOR TO &c_standard.
  343.       * Erase message - filter is off
  344.       @ 0,51
  345.    ENDIF
  346.    * Show  current record number on screen
  347.    @ 0,66 SAY "Record #" + STR(RECNO(),5,0) COLOR &c_yellow.
  348. RETURN
  349.  
  350. PROCEDURE Edit
  351.    * Edit current record
  352.    * Display lookup key message if lookup available (set in each application)
  353.    IF lookup_ok
  354.       DO Sho_look WITH dbf
  355.    ENDIF
  356.    record_num = RECNO()
  357.    * Load data from record into memory variables
  358.    DO Load_fld
  359.    IF NETWORK()                      && Edit data in a network
  360.       ready = .F.
  361.       DO WHILE .NOT. ready
  362.          IF CHANGE()
  363.             * If the record was changed by somone since user first accessed it
  364.             DO Warnbell
  365.             GO RECNO()           && Updates database record with changed data
  366.             IF DELETED()
  367.                DO Show_msg WITH "ALERT - Record has been deleted"
  368.                SKIP
  369.                DO Show_data
  370.                RETURN            && Exit to OPTION MENU - quit edit
  371.             ELSE
  372.                DO Show_msg WITH ;
  373.                   "Data has been changed-screen shows revised data"
  374.                DO Load_fld           && Updates memvars with database data
  375.             ENDIF
  376.          ENDIF
  377.          DO Get_data
  378.          READ                        && Edit data
  379.          * Test if another user changed data while editing this data
  380.          ready = .NOT. CHANGE()      && DO loop will repeat if CHANGE() is .F.
  381.       ENDDO
  382.    ELSE                              && Non-network edit
  383.       DO Get_data
  384.       READ                           && Edit data
  385.    ENDIF
  386.    *****
  387.    * Erase F9 lookup message from screen
  388.    @ 0,0 SAY SPACE(51)
  389.    IF "" = TRIM(&key.) .OR. READKEY() < 256
  390.       * Exit if user blanked key, did not change data, or deleted record
  391.       RETURN
  392.    ELSE
  393.       * Save edited data to disk
  394.       DO Sav_data
  395.    ENDIF
  396. RETURN
  397.  
  398. PROCEDURE Eraser
  399.    * Erase current record
  400.    IF NodShake( " ;   Erase this data record?   ", ;
  401.                 9, 26, 2, 29, .F. )
  402.  
  403.       DELETE
  404.       * Position to the next record
  405.       SKIP
  406.       * Check if the last record was deleted
  407.       DO CASE
  408.          CASE filters_on .AND. EOF()
  409.             * If no records left in filter subset, turn off filter
  410.             SET FILTER TO
  411.             filters_on = .F.
  412.             * If last record deleted, go to beginning of database
  413.             GO TOP
  414.          CASE .NOT. filters_on .AND. EOF()
  415.             * If last record deleted, go to beginning of database
  416.             GO TOP
  417.       ENDCASE
  418.       * Set erased status flag that record was deleted
  419.       erased = .T.
  420.    ENDIF
  421. RETURN
  422.  
  423. PROCEDURE Filt_ans
  424.    * Get answer from user about filtering data into subset
  425.    IF filters_on
  426.       *-- Filter window - to turn off filter
  427.       ll_ans = NodShake( " ;    GROUP into SUBSET (Filter)   ;" + ;
  428.                          "   Subset is currently selected.   ;" + ;
  429.                          "         Turn Filter off?", ;
  430.                          7, 22, 4, 35, .F. )
  431.    ELSE
  432.       *-- Filter window - to turn on filter
  433.       ll_ans = NodShake( " ;    GROUP into SUBSET (Filter)   ;" + ;
  434.                          "   Select temporary subset of data   ;" + ;
  435.                          "   by entering filter condition(s)   ;" + ;
  436.                          "             Proceed?", ;
  437.                          7, 21, 5, 37, .F. )
  438.    ENDIF
  439.    IF .NOT. ll_ans              && Do not change filter status
  440.       RETURN TO Barpop          && Do not finish processing Filter proceedure
  441.       choice = "N"
  442.    ELSE
  443.       choice = "Y"
  444.    ENDIF
  445. RETURN
  446.  
  447. PROCEDURE Findcode
  448.    PARAMETERS acity
  449.    * Look up area code for phone number - by city
  450.    i = INKEY()
  451.    acode = 0
  452.    ACTIVATE WINDOW alert
  453.       CLEAR
  454.       acode = LOOKUP(Codes->code,TRIM(acity),Codes->city)
  455.       ? "------------- AREA CODE LOOKUP -----------"
  456.       IF .NOT. FOUND("Codes") .OR. "" = TRIM(acity)
  457.          DO Warnbell
  458.          ? "City: " + TRIM(acity) + " was"    AT 2
  459.          ? "NOT FOUND in areacodes database." AT 2
  460.       ELSE
  461.          ?
  462.          ? "AREA CODE is: " + STR(acode,3) AT 2
  463.          ? "for " + TRIM(acity)  AT 16
  464.       ENDIF
  465.       ?
  466.       i= INKEY(3)                   && Display for 3 seconds
  467.    DEACTIVATE WINDOW alert
  468. RETURN
  469.  
  470. PROCEDURE Findcust
  471.    PARAMETERS custid
  472.    * Look up customer from customer ID
  473.    i= INKEY()
  474.    acust = ""
  475.    ACTIVATE WINDOW alert
  476.       CLEAR
  477.       acust = LOOKUP(Cust->customer,TRIM(custid),Cust->cust_id)
  478.       ? "---------- CUSTOMER ID  LOOKUP -----------"
  479.       IF .NOT. FOUND("Cust") .OR. "" = TRIM(custid)
  480.          DO Warnbell
  481.          ? "Customer ID: " + TRIM(custid) + " was" AT 2
  482.          ? "NOT FOUND in Cust database." AT 2
  483.       ELSE
  484.          ? "Customer: " + TRIM(acust)  AT 2
  485.          ? "Phone:    " + Cust->phone  AT 2
  486.          ? "for ID: "   + TRIM(custid) AT 12
  487.       ENDIF
  488.       WAIT "  Press spacebar to continue..."
  489.    DEACTIVATE WINDOW alert
  490. RETURN
  491.  
  492. PROCEDURE Find_rec
  493.    PARAMETERS key, key1, keyname1, key2, keyname2, key3, keyname3
  494.    * Get target data to find/seek and show data record after retrieving
  495.    STORE "" TO target1, target2, target3
  496.    target1 = IIF(TYPE(key1) = "C", SPACE(LEN(&key1.)), {  /  /  })
  497.    * If key2 exists (database requires two keys)
  498.    IF "NONE" <> key2
  499.       target2 = IIF(TYPE(key2) = "C", SPACE(LEN(&key2.)), {  /  /  })
  500.       * If key3 exists (database has three keys)
  501.       IF "NONE" <> key3
  502.          target3 = IIF(TYPE(key3) = "C", SPACE(LEN(&key3.)), {  /  /  })
  503.       ENDIF
  504.    ENDIF
  505.    ACTIVATE WINDOW alert
  506.       @ 0,0 SAY "-------- ENTER TARGET DATA TO FIND -------"
  507.       @ 2, 1 SAY keyname1
  508.       @ 2,15 GET target1  MESSAGE "Enter " + keyname1
  509.       IF "NONE" <> key2
  510.          @ 3, 1 SAY keyname2
  511.          @ 3,15 GET target2
  512.          IF "NONE" <> key3
  513.             @ 4, 1 SAY keyname3
  514.             @ 4,15 GET target3
  515.          ENDIF
  516.       ENDIF
  517.       @ 5,1 SAY "Enter partial or entire data"
  518.       READ
  519.    DEACTIVATE WINDOW alert
  520.    target = IIF(type(key1) = "C", target1, DTOC(target1))
  521.    IF "NONE" <> key2
  522.       target = target + IIF(type(key2) = "C", target2, DTOC(target2))
  523.       IF "NONE" <> key3
  524.          target = target + IIF(type(key3) = "C", target3, DTOC(target3))
  525.       ENDIF
  526.    ENDIF
  527.    target = TRIM(target)
  528.    IF RIGHT(target, 6) = "  /  /"
  529.       * If a date key wasn't filled in, remove the template
  530.       target = LEFT(target, LEN(target) - 6)
  531.    ENDIF
  532.    IF "" = target
  533.       * If user entered nothing (blank key) => exit
  534.       RETURN
  535.    ENDIF
  536.    * Store record no. that the user was viewing
  537.    record_num = RECNO()
  538.    * Find record with target key
  539.    IF .NOT. SEEK(target)
  540.       * If target not found, uppercase & look again
  541.       IF .NOT. SEEK(UPPER(target))
  542.          * Sound bell and alert user with message
  543.          DO Warnbell
  544.          DO Show_msg WITH "Record with target data was NOT found."
  545.          * Return to original record user was viewing
  546.          GO record_num
  547.       ENDIF
  548.    ENDIF
  549. RETURN
  550.  
  551. PROCEDURE Findpart
  552.    PARAMETERS partid
  553.    * Look up part data using part ID number in Goods database when
  554.    * function key pressed
  555.    i = INKEY()
  556.    p_name = SPACE(30)
  557.    ACTIVATE WINDOW alert
  558.       CLEAR
  559.       p_name = LOOKUP(Goods->part_name,TRIM(partid),Goods->part_id)
  560.       ? "------------ PART CODE  LOOKUP ----------"
  561.       IF .NOT. FOUND("Goods") .OR. "" = TRIM(partid)
  562.          DO Warnbell
  563.          ? "Part ID: " + TRIM(partid) AT 2
  564.          ? "was NOT FOUND in Goods database." AT 2
  565.       ELSE
  566.          ? "For ID:    " + partid       AT 2
  567.          ? "Part name: " + TRIM(p_name) AT 2
  568.          ? "Qty on hand: " + STR(Goods->qty_onhand,4) AT 2
  569.          ? "Price: $  " AT 2, Goods->price PICTURE "99,999.99"
  570.       ENDIF
  571.       WAIT " .....Press spacebar to continue....."
  572.    DEACTIVATE WINDOW alert
  573. RETURN
  574.  
  575. PROCEDURE Findvend
  576.    PARAMETERS vendr
  577.    * Look up vendor name using vendor ID number in Vendor database
  578.    * when function key pressed
  579.    i = INKEY()
  580.    v_name = SPACE(30)
  581.    ACTIVATE WINDOW alert
  582.       CLEAR
  583.       v_name = LOOKUP(Vendors->vendor,TRIM(vendr),Vendors->vendor_id)
  584.       ? "----------- VENDOR CODE LOOKUP -----------"
  585.       IF .NOT. FOUND("Vendors")
  586.          DO Warnbell
  587.          ? "Vendor ID: " + TRIM(vendr)    AT 2
  588.          ? "was NOT FOUND in Vendors database." AT 2
  589.       ELSE
  590.          ? "VENDOR is: " + TRIM(v_name)   AT 2
  591.          ? "Phone:     " + Vendors->phone AT 2
  592.          ? "for ID:  "   + vendr          AT 16
  593.       ENDIF
  594.       WAIT "   Press spacebar to continue..."
  595.    DEACTIVATE WINDOW alert
  596. RETURN
  597.  
  598. PROCEDURE Kount
  599.    * Count and display number of records in database
  600.    record_num = RECNO()
  601.    ACTIVATE WINDOW alert
  602.      @ 0,0 SAY "------------- COUNT  RECORDS -------------"
  603.      @ 2,1 SAY "Counting, please wait..."
  604.      * Use count if filter is active (subset of records)
  605.      IF filters_on
  606.         COUNT TO kount
  607.      ELSE
  608.         * Use reccount if filter is not active (all records)
  609.         kount = RECCOUNT()
  610.      ENDIF
  611.      @ 2,1 SAY "There are: " + STR (kount,6) + " records in "+ dbf
  612.      ?
  613.      WAIT " Press any key to continue..."
  614.    DEACTIVATE WINDOW alert
  615.    * Return to original record (before count)
  616.    GO record_num
  617. RETURN
  618.  
  619. PROCEDURE List_rec
  620.    * Lists records (in active index order) from current record on
  621.    * If filter is active, then subset listed
  622.    record_num = RECNO()                 && Store current record position
  623.    ACTIVATE WINDOW lister
  624.       answer = " "
  625.       CLEAR
  626.       @ 0,0 SAY "------------------------- LIST RECORDS " + ;
  627.                 "-------------------------" ;
  628.             COLOR &c_red.
  629.       SCAN WHILE .NOT. answer $ "rR"
  630.          LIST OFF NEXT 10 &list_flds.
  631.          WAIT "Press spacebar to continue or R to return to " + ;
  632.               "OPTION MENU." TO answer
  633.          CLEAR
  634.       ENDSCAN
  635.    DEACTIVATE WINDOW lister
  636.    * Return to original record (before viewing list)
  637.    GO record_num
  638. RETURN
  639.  
  640. PROCEDURE Look_msg
  641.    DO CASE                                && Show proper lookup msg in window
  642.       CASE similar = .F.                  && No similar data found
  643.          @ 1,1 SAY "Entered "+look_name+" ID does not exist in " + ;
  644.                look_dbf+" database."
  645.          ?
  646.          WAIT "No " + look_name + " ID's are similar - " + ;
  647.               "press R to return to screen." TO answer
  648.       CASE similar = .T. .AND. listcount > 0
  649.          && Similar data found and listed
  650.          WAIT "Press spacebar to continue list or " + ;
  651.               "R to return to screen." TO answer
  652.          CLEAR
  653.    ENDCASE
  654.    CLEAR
  655. RETURN
  656.  
  657. FUNCTION Lookupid
  658.    PARAMETERS l_target, look_dbf, look_name, matchchars
  659.    * During data entry or editing, validate data entered into any of the
  660.    * fields of customer ID, parts ID, vendor ID, and employee ID by checking
  661.    * for their existence in their respective databases - list any similar data
  662.    * by matching the first one or more characters (between entered data and
  663.    * database).
  664.    * Note: matchchars = number of initial matching characters for lookup lists
  665.    * Example: list will show customers whose cust_id's first two characters
  666.    * match with the entered cust_id's first two characters (matchchars = 2)
  667.    IF .NOT. SEEK(l_target,(look_dbf))     && Seek data in its respective dbf
  668.       ACTIVATE WINDOW look
  669.       DO Warnbell
  670.       answer = " "
  671.       similar = .F.
  672.       SELECT (look_dbf)                   && Use appropriate dbf for listing
  673.       GO TOP
  674.       DO WHILE .NOT. (EOF() .OR. answer $ "rR")
  675.          * Show list of records having identical initial character(s)
  676.          * in ID number
  677.          @ 0,0 SAY "-------- DATA ENTRY ERROR: " + look_name + ;
  678.                    " ID WAS INVALID -------"
  679.          @ 1,0 SAY "          This is a list of similar " + look_name + ;
  680.                    " ID's"
  681.          ?
  682.          listcount = 0
  683.          DO CASE                         && Check which database screen in use
  684.            CASE dbf = "ORDERS"
  685.               DO CASE                    && Check which field is being read
  686.                  CASE VARREAD() = "CUST_ID"
  687.                     SCAN FOR LIKE(SUBSTR(l_target,1,matchchars)+"*",cust_id) ;
  688.                        WHILE listcount <= 4
  689.                        ? cust_id, customer           && Display a record
  690.                        listcount = listcount + 1     && Increment list counter
  691.                        similar = .T.                 && Data found and listed
  692.                     ENDSCAN
  693.                  CASE VARREAD() = "PART_ID"
  694.                     SCAN FOR LIKE(SUBSTR(l_target,1,matchchars)+"*",part_id) ;
  695.                        WHILE listcount <= 4
  696.                        ? part_id, SUBSTR(part_name,1,21), ;
  697.                          SUBSTR(descript,1,24)
  698.                        listcount = listcount + 1     && Increment list counter
  699.                        similar = .T.                 && Data found and listed
  700.                     ENDSCAN
  701.                  CASE VARREAD() = "EMP_ID"
  702.                     SCAN FOR LIKE(SUBSTR(l_target,1,matchchars)+"*",emp_id) ;
  703.                        WHILE listcount <= 4
  704.                        ? emp_id, lastname, firstname && Display a record
  705.                        listcount = listcount + 1     && Increment list counter
  706.                        similar = .T.                 && Data found and listed
  707.                     ENDSCAN
  708.               ENDCASE
  709.            CASE dbf = "GOODS"
  710.               SCAN FOR LIKE(SUBSTR(l_target,1,matchchars)+"*",vendor_id) ;
  711.                  WHILE listcount <= 4
  712.                  ? vendor_id, vendor                 && Display a record
  713.                  listcount = listcount + 1           && Increment list counter
  714.                  similar = .T.                       && Data found and listed
  715.               ENDSCAN
  716.            CASE dbf = "ACCT_REC"
  717.               SCAN FOR LIKE(SUBSTR(l_target,1,matchchars)+"*",cust_id) ;
  718.                  WHILE listcount <= 4
  719.                  ? cust_id, customer                 && Display a record
  720.                  listcount = listcount + 1           && Increment list counter
  721.                  similar = .T.                       && Data found and listed
  722.               ENDSCAN
  723.          ENDCASE
  724.          DO Look_msg                                 && Show message in window
  725.       ENDDO
  726.       DEACTIVATE WINDOW look
  727.       SELECT 1                                       && Use original dbf
  728.    ENDIF
  729. RETURN not_valid = .NOT. FOUND((look_dbf))
  730.  
  731. PROCEDURE Net_err
  732.    PARAMETERS err_number
  733.    * Error procedure for networks
  734.    DO CASE
  735.       CASE err_number = 108
  736.          * File is in use by another person
  737.          IF "" <> TRIM(LKSYS(2))
  738.             message = " " + dbf + " is in use by: " + LKSYS(2)
  739.          ELSE
  740.             message = " " + dbf + " is in use by someone"
  741.          ENDIF
  742.       CASE err_number = 109
  743.          * Record is locked by another person
  744.          message = " Record is locked by: " + LKSYS(2)
  745.       CASE err_number = 110
  746.          * File must be in exclusive use for indexing/packing
  747.          message = "File should be USEd EXCLUSIVE"
  748.       CASE err_number = 372 .OR. err_number = 373
  749.          * File or record is in use by another
  750.          message = MESSAGE()
  751.       OTHERWISE
  752.          message = " Unknown error: " + MESSAGE()
  753.    ENDCASE
  754.    DO Warnbell
  755.    ACTIVATE WINDOW alert
  756.       CLEAR
  757.       ? "------------ NETWORK ERROR --------------"
  758.       ?
  759.       ? message AT 1
  760.       ? "Press spacebar to try again" AT 1
  761.       ? " - or press Esc to Quit" AT 1
  762.       net_choice = INKEY(0)          && Wait for user to press a key
  763.    DEACTIVATE WINDOW alert
  764.    IF net_choice <> 27               && User did not press Esc key
  765.       * Execute command again that caused network error
  766.       RETRY
  767.    ENDIF
  768. RETURN
  769.  
  770. PROCEDURE Printout
  771.    * Print report or label
  772.    DO CASE
  773.       CASE reportype = "LISTING"
  774.          REPORT FORM &dbf.
  775.       CASE reportype = "LABELS"
  776.          LABEL FORM &dbf.
  777.       CASE reportype = "CUSTOM"
  778.          DO &rpt_name.
  779.    ENDCASE
  780.    GO record_num
  781. RETURN
  782.  
  783. PROCEDURE Prt_menu
  784.    * Display menu of print options
  785.    msg_num   = "Enter a number"
  786.    msg_logic = "Enter a Y or N"
  787.    msg_enum  = "Press spacebar for other options"
  788.    * Set up default values to print variables for reports
  789.    loffset  = 0
  790.    lmargin  = 0
  791.    rmargin  = 80
  792.    indent   = 4
  793.    plength  = 66           && 60 - HP laserjet printer
  794.    STORE 1 TO pspacing, pbpage, pcopies
  795.    pepage   = 9999
  796.    peject   = "NONE  "
  797.    STORE .F. TO pwait, pquality
  798.    ppitch   = "PICA     "
  799.    *
  800.    ACTIVATE WINDOW lister
  801.    CLEAR
  802.    @  0, 0 SAY "------------------------- PRINT MENU " + ;
  803.               "---------------------------" COLOR &c_red.
  804.    @  2, 1 SAY "Page settings:"
  805.    @  3, 1 SAY "============="
  806.    @  4, 1 SAY "Offset from left  " GET loffset ;
  807.            PICTURE "99" MESSAGE msg_num
  808.    @  5, 1 SAY "Left margin       " GET lmargin ;
  809.            PICTURE "99" MESSAGE msg_num
  810.    @  6, 1 SAY "Right margin      " GET rmargin ;
  811.            PICTURE "99" MESSAGE msg_num
  812.    @  7, 1 SAY "Indentation       " GET indent ;
  813.            PICTURE "99" MESSAGE msg_num
  814.    @  8, 1 SAY "Page length       " GET plength ;
  815.            PICTURE "99" MESSAGE msg_num
  816.    @  9, 1 SAY "Spacing           " GET pspacing ;
  817.            PICTURE "9"  RANGE 1,3 MESSAGE msg_num
  818.    @  2,26 SAY "Print settings:"
  819.    @  3,26 SAY "=============="
  820.    @  4,26 SAY "Begin printing on page  " GET pbpage ;
  821.            PICTURE "999"  MESSAGE msg_num
  822.    @  5,26 SAY "End printing on page    " GET pepage ;
  823.            PICTURE "9999" MESSAGE msg_num
  824.    @  6,26 SAY "Number of copies        " ;
  825.            GET pcopies  PICTURE "999"  MESSAGE msg_num
  826.    @  7,26 SAY "Eject paper             " GET peject ;
  827.            PICTURE "@M BEFORE,AFTER,BOTH,NONE"  MESSAGE msg_enum
  828.    @  8,26 SAY "Wait between pages      " GET pwait ;
  829.            PICTURE "Y" MESSAGE msg_logic
  830.    @  9,26 SAY "Pitch                   " GET ppitch ;
  831.            PICTURE "@M DEFAULT,PICA,ELITE,CONDENSED" MESSAGE msg_enum
  832.    @ 10,26 SAY "Quality print           " GET pquality ;
  833.            PICTURE "Y" MESSAGE msg_logic
  834.    @ 12, 1 SAY "Please enter desired settings; press Esc to cancel"
  835.    READ
  836.    DEACTIVATE WINDOW lister
  837.    IF LASTKEY() = 27                    && If Escaped presses
  838.       ll_esc = .T.
  839.    ELSE
  840.       ll_esc = .F.
  841.  
  842.       * Assign values to system variables
  843.       _ploffset = loffset
  844.       _lmargin  = lmargin
  845.       _rmargin  = rmargin
  846.       _indent   = indent
  847.       _plength  = plength
  848.       _pspacing = pspacing
  849.       _pbpage   = pbpage
  850.       _pepage   = pepage
  851.       _pcopies  = pcopies
  852.       _peject   = peject
  853.       _pwait    = pwait
  854.       IF PRINTSTATUS()
  855.         _ppitch   = ppitch
  856.       ENDIF
  857.       _pquality = pquality
  858.    ENDIF
  859.    SET COLOR TO &c_standard.
  860. RETURN
  861.  
  862. PROCEDURE Rest_env
  863.     * Restore database environment
  864.     SET COLOR TO &c_standard.
  865.     SET SCOREBOARD &scor.
  866.     SET DELIMITERS &deli.
  867.     SET HELP &hellp.
  868.     SET CLOCK &clock.
  869.     SET ESCAPE &esca.
  870.     SET DELETED &delee.
  871.     SET HEADING &head.
  872.     SET STATUS &stat.
  873.     SET SAFETY &safe.
  874.     SET EXACT &exac.
  875.     SET BELL &bell.
  876.     SET NEAR &near.
  877.     * Reset colors to system defaults
  878.     DO Colo_rese
  879.     SET TALK &talk.
  880. RETURN
  881.  
  882. PROCEDURE Sav_data
  883.    * If data is new: append record currently in memory to database.
  884.    * If edited/modified data: replace database record with memory fields.
  885.  
  886.    IF NodShake( " ;   Save this data to disk?   ", ;
  887.                 9, 26, 2, 29, .F. )
  888.  
  889.       IF PROMPT() = " Add record"      && Add new blank record
  890.          APPEND BLANK
  891.          record_num = RECNO()
  892.       ENDIF
  893.       * Replace database file fields with contents of memory variables
  894.       DO Repl_fld
  895.    ELSE
  896.       * Do not save data to disk, return to original record
  897.       GO record_num
  898.    ENDIF
  899. RETURN
  900.  
  901. PROCEDURE Set_env
  902.    PUBLIC talk                  && First set TALK OFF
  903.    talk         = SET("TALK")
  904.    SET TALK off
  905.  
  906.    PUBLIC c_standard, c_data, c_fields, c_popup, c_alert, c_list
  907.    PUBLIC c_red, c_blue, c_yellow, c_yelowhit, c_green, c_blink
  908.  
  909.    * Set color variables for applications
  910.    IF ISCOLOR()
  911.       * Color video card/monitor
  912.       c_standard = "W/B,BG+/R,B"
  913.       c_data     = "B/W,R/BG,B"
  914.       c_fields   = "B/BG"
  915.       c_popup    = "B/W,GR+/R"
  916.       c_alert    = "GR+/R,B/W,R/G"
  917.       c_list     = "W+/G,GR+/B,GR+/GR"
  918.       c_red      = "R/W"
  919.       c_blue     = "B/W"
  920.       c_yellow   = "GR+/B"
  921.       c_yelowhit = "GR+/W"
  922.       c_green    = "G/W"
  923.       c_blink    = "GR+*/B"
  924.    ELSE
  925.       * Monochrome video card/monitor
  926.       STORE "W+/N,N/W" TO c_standard, c_data, c_popup, c_alert, c_list
  927.       STORE "W" TO  c_red, c_blue, c_yellow, c_yelowhit, c_green, c_fields
  928.       c_blink = "W+*/N,N/W"
  929.    ENDIF
  930.    SET COLOR OF MESSAGES TO &c_blue.
  931.    SET COLOR TO &c_standard.
  932.  
  933.    * Configure working environment
  934.    * Store SET environment in case started from Control Center or dot prompt
  935.    PUBLIC scor, deli, hellp, clock, esca, delee, head, stat, safe
  936.    PUBLIC exac, bell, near
  937.    scor  = SET("SCOREBOARD")
  938.    deli  = SET("DELIMITERS")
  939.    hellp = SET("HELP")
  940.    clock = SET("CLOCK")
  941.    esca  = SET("ESCAPE")
  942.    delee = SET("DELETED")
  943.    head  = SET("HEADING")
  944.    stat  = SET("STATUS")
  945.    safe  = SET("SAFETY")
  946.    exac  = SET("EXACT")
  947.    bell  = SET("BELL")
  948.    near  = SET("NEAR")
  949.  
  950.    * Set database environment for applications
  951.    SET SCOREBOARD off
  952.    SET DELIMITERS off
  953.    SET HELP    off
  954.    SET CLOCK   off
  955.    SET ESCAPE  off
  956.    SET DELETED on
  957.    SET HEADING on
  958.    SET STATUS  off
  959.    SET SAFETY  off
  960.    SET TALK    off
  961.    SET EXACT   off
  962.    SET BELL    off
  963.    SET NEAR    off
  964.    PUBLIC erased, not_valid, rec_is_dup, filters_on, lookup_ok, choice
  965.    PUBLIC record_num, net_choice
  966.    PUBLIC target, look_dbf, matchchar, scanfield
  967.    * Logical variables used for status flags
  968.    STORE .F. TO  erased, not_valid, rec_is_dup, filters_on
  969.    lookup_ok = .T.
  970.    * Other variables
  971.    STORE "" TO choice,subset
  972.    STORE 0 TO record_num, net_choice
  973.    ************************************************
  974.    * Setup error processing if running on a network
  975.    IF NETWORK()
  976.       * Network programming assumes databases have been CONVERTed
  977.       SET EXCLUSIVE off
  978.       ON ERROR DO Net_err WITH ERROR()
  979.       * Retry a reasonable amount of time (depends on computer)
  980.       SET REPROCESS TO 3
  981.    ENDIF
  982.    ************************************************
  983.    * Turns off VALID failure's (PRESS SPACE)
  984.    ON READERROR ??
  985. RETURN
  986.  
  987. PROCEDURE Sho_look
  988.    PARAMETERS db
  989.    * Show lookup function keys on screen (if available for database)
  990.    DO CASE
  991.       CASE db = "EMPLOYEE" .OR. db = "CUST" .OR. db = "VENDORS"
  992.          look_txt = "Press F9 to look up Area code"
  993.       CASE db = "GOODS"
  994.          look_txt = "Press F9 to look up Vendor name and phone"
  995.       CASE db = "ORDERS"
  996.          look_txt = "Press F9 to look up Cust data; F10 for Part ID data"
  997.       CASE db = "ACCT_REC"
  998.          look_txt = "Press F9 to look up Customer name and phone"
  999.    ENDCASE
  1000.    @ 0,0 SAY look_txt COLOR &c_blink.
  1001.    i = INKEY(1)                                  && Blink for 1 second
  1002.    @ 0,0 SAY look_txt COLOR &c_yellow.
  1003. RETURN
  1004.  
  1005. PROCEDURE Show_msg
  1006.    PARAMETERS u_message
  1007.    _wrap = .T.
  1008.    ACTIVATE WINDOW alert
  1009.       @ 1,0 SAY u_message
  1010.       ?
  1011.       WAIT " Press spacebar to continue..."
  1012.    DEACTIVATE WINDOW alert
  1013. RETURN
  1014.  
  1015. PROCEDURE Skip_rec
  1016.    PARAMETERS skipno
  1017.    * Skip forward or backward in database by one or more records
  1018.    DO CASE
  1019.    CASE skipno = 1         && Skip to next record (in active index order)
  1020.       IF .NOT. EOF()
  1021.          SKIP
  1022.       ENDIF
  1023.    CASE skipno = -1        && Skip to previous record (in active index order)
  1024.       IF .NOT. BOF()
  1025.          SKIP -1
  1026.       ENDIF
  1027.    CASE skipno = 0
  1028.       * Skip records - to goto/view records ahead of or behind current record
  1029.       numb_2skip = 0
  1030.       ACTIVATE WINDOW alert
  1031.          @ 0,0 SAY "-------- SKIP NUMBER OF RECORDS ----------"
  1032.          @ 2,1 SAY "How many records do you want to skip?"
  1033.          @ 3,0 SAY "   (Example: 15 or -5) ?   " ;
  1034.                GET numb_2skip PICTURE "9999" ;
  1035.                MESSAGE "Enter positive no. to go forward " + ;
  1036.                        "or negative no. to go backward"
  1037.          READ
  1038.       DEACTIVATE WINDOW alert
  1039.       IF .NOT. (BOF() .AND. numb_2skip < 0) .OR. (EOF() .AND. numb_2skip > 0)
  1040.          SKIP numb_2skip
  1041.       ENDIF
  1042.    ENDCASE
  1043.  
  1044.    * Check whether record pointer hits beginning or end of file
  1045.    DO CASE
  1046.       CASE EOF()
  1047.          GO BOTTOM                  && reset record pointer if EOF
  1048.          DO Show_msg WITH " Bottom record in " + dbf + " database"
  1049.       CASE BOF()
  1050.          DO Show_msg WITH " Top record in " + dbf + " database"
  1051.    ENDCASE
  1052. RETURN
  1053.  
  1054. PROCEDURE Warnbell
  1055.    PRIVATE mwrap
  1056.    mwrap = _wrap           && Save _wrap value
  1057.    _wrap = .F.
  1058.    * Sound unique warning for errors
  1059.    SET BELL TO 880,4
  1060.    ?? CHR(7)
  1061.    SET BELL TO 1400,4
  1062.    ?? CHR(7)
  1063.    SET BELL TO 880,4
  1064.    ?? CHR(7)
  1065.    SET BELL TO
  1066.    _wrap = mwrap
  1067. RETURN
  1068.  
  1069.  
  1070. FUNCTION NodShake
  1071. PARAMETERS pc_mssg, pn_up, pn_left, pn_height, pn_max, pl_dflt_no
  1072. *---------------------------------------------------------------------------
  1073. * NAME
  1074. *   NodShake
  1075. *
  1076. * DESCRIPTION
  1077. *   Accepts a YES/NO response from user
  1078. *
  1079. * SYNOPSIS
  1080. *   DO _NodShake WITH pc_mssg, pn_up, pn_left, pn_height, pn_max, pl_dflt_no
  1081. *
  1082. * PARAMETERS
  1083. *   pc_mssg:    dialog box message
  1084. *   pn_up:      upper corrdinate of dialog box
  1085. *   pn_left:    left coordinate of dialog box
  1086. *   pn_height:  height of dialog box
  1087. *   pn_max:     maximum width of a line in message
  1088. *   pl_dflt_no: flag indicating if default pad highlighted should be "NO"
  1089. *       
  1090. * EXAMPLE
  1091. *    pl_set = _NodShake( pc_vermssg, 13, 25, 2, 28, .T. )
  1092. *       
  1093. * LIMITATIONS
  1094. *   None
  1095. *
  1096. * DEPENDENCIES
  1097. *   None
  1098. *---------------------------------------------------------------------------
  1099.  
  1100.   PRIVATE ll_ans, ll_console, ll_wrapset, ln_pspset
  1101.  
  1102.   ll_console = SET( "CONSOLE" ) = "OFF"
  1103.   SET CONSOLE ON
  1104.   ll_wrapset = _wrap
  1105.   ln_pspset = _pspacing
  1106.   _wrap = .F.
  1107.   _pspacing = 1
  1108.  
  1109.   DEFINE WINDOW NodShake DOUBLE ;
  1110.      FROM pn_up, pn_left TO pn_up + pn_height + 4, pn_left + pn_max + 1
  1111.  
  1112.   DEFINE MENU NodShake
  1113.   DEFINE PAD Yes OF NodShake PROMPT "Yes" ;
  1114.      AT pn_height + 1, (pn_max - 12) / 2;
  1115.      MESSAGE "Select option and press ENTER, or press first letter" + ;
  1116.              " of desired option"
  1117.  
  1118.   ON SELECTION PAD Yes OF NodShake DEACTIVATE MENU
  1119.   DEFINE PAD No OF NodShake PROMPT "No" ;
  1120.      AT pn_height + 1, (pn_max - 12) / 2 + 10 ;
  1121.      MESSAGE "Select option and press ENTER, or press first letter" + ;
  1122.              " of desired option"
  1123.  
  1124.   ON SELECTION PAD No OF NodShake DEACTIVATE MENU
  1125.   ACTIVATE WINDOW NodShake
  1126.   CLEAR
  1127.   ?
  1128.   @ 0, 0
  1129.   ?? pc_mssg FUNCTION ";"
  1130.  
  1131.   ON KEY LABEL Y KEYBOARD "{Alt-Y}{13}"
  1132.   ON KEY LABEL N KEYBOARD "{Alt-N}{13}"
  1133.  
  1134.   IF pl_dflt_no
  1135.     KEYBOARD "{Alt-N}"
  1136.   ENDIF
  1137.  
  1138.   ON KEY LABEL RIGHTARROW
  1139.   ON KEY LABEL LEFTARROW
  1140.  
  1141.   ACTIVATE MENU NodShake
  1142.  
  1143.   ON KEY LABEL Y
  1144.   ON KEY LABEL N
  1145.  
  1146.   IF PAD() = "YES"
  1147.     ll_ans = .T.
  1148.   ELSE
  1149.     ll_ans = .F.
  1150.   ENDIF
  1151.  
  1152.   RELEASE WINDOW NodShake
  1153.   RELEASE MENU NodShake
  1154.   _wrap = ll_wrapset
  1155.   _pspacing = ln_pspset
  1156.  
  1157.   IF ll_console
  1158.     SET CONSOLE OFF
  1159.   ENDIF
  1160.  
  1161. RETURN ll_ans
  1162. *-- EOF: NodShake( pc_mssg, pn_up, pn_left, pn_height, pn_max, pl_dflt_no )
  1163.  
  1164. PROCEDURE Err_Box
  1165. PARAMETERS pc_msg
  1166. *----------------------------------------------------------------------------
  1167. * NAME
  1168. *   Err_Box - Display an error box
  1169. *
  1170. * SYNOPSIS
  1171. *   DO Err_Box WITH <pc_msg>
  1172. *
  1173. * DESCRIPTION
  1174. *   _Err_Box will display the <pc_msg> string in a box and prompt the
  1175. *   user to press any key to continue processing.  _Err_Box will display
  1176. *   the message based on the length of <pc_msg>.
  1177. *
  1178. * PARAMETERS
  1179. *   pc_msg - the error message to display in the box.  If the length is
  1180. *            greater than 76, the trailing part is chopped off.
  1181. *
  1182. * EXAMPLE
  1183. *   DO Err_Box WITH "Incorrect window size"
  1184. *   Displays the message in a window as follows at row 9 on the screen:
  1185. *                      +------------------------------+
  1186. *                      |                              |
  1187. *                      |    Incorrect window size     |
  1188. *                      |                              |
  1189. *                      | Press any key to continue... |
  1190. *                      |                              |
  1191. *                      +------------------------------+
  1192. *   Note that the width of the window will increase to accommodate a longer
  1193. *   message string.
  1194. *
  1195. * LIMITATIONS
  1196. *   Truncates the message after 76 characters.  Assumes an 80 character
  1197. *   wide screen.  Looks best with SET CURSOR OFF.
  1198. *
  1199. *----------------------------------------------------------------------------
  1200.  
  1201.   PRIVATE lc_anykey, lc_msg, lc_msglen, lc_win, ln_press, ln_width, ll_trap,;
  1202.           ll_escape
  1203.  
  1204.   lc_anykey = [Press any key to continue...]
  1205.   ln_press  = LEN( lc_anykey )
  1206.   lc_win = WINDOW()                     && Currently activated window if any
  1207.   lc_msg = LTRIM( RTRIM( pc_msg ) )     && Trimmed message
  1208.   ln_msglen = LEN( lc_msg )             && Trimmed length of message
  1209.   ln_width = 0                          && Width of display area in window.
  1210.   ll_escape = SET("ESCAPE") = "ON"
  1211.   SET ESCAPE OFF
  1212.  
  1213.   *-- Determine the width needed for the window:
  1214.   IF ln_msglen <= ln_press
  1215.     ln_width = ln_press
  1216.   ELSE
  1217.     *-- Make sure the message fits in the window:
  1218.     IF ln_msglen > 76
  1219.       lc_msg = LEFT( lc_msg, 76 )
  1220.       ln_msglen = 76
  1221.     ENDIF
  1222.     ln_width = ln_msglen
  1223.   ENDIF
  1224.   DEFINE WINDOW _err_box FROM 9, ((76 - ln_width) + .5) / 2 ;
  1225.                 TO 15, (ln_width + 83) / 2 DOUBLE
  1226.   ln_width = ( ln_width + 2 )
  1227.  
  1228.   *-- Display the message and prompt to the window and wait for a key press
  1229.   ACTIVATE WINDOW _err_box
  1230.   @ 1, ( ln_width - ln_msglen ) / 2 SAY lc_msg
  1231.   @ 3, ( ln_width - ln_press ) / 2 SAY lc_anykey
  1232.   SET CONSOLE OFF                       && For mouse click recognition
  1233.   WAIT
  1234.   SET CONSOLE ON
  1235.  
  1236.   *-- Clean up the window display and reactivate the previous window
  1237.   RELEASE WINDOW _err_box
  1238.   IF ISBLANK( lc_win )
  1239.     ACTIVATE SCREEN
  1240.   ENDIF
  1241.  
  1242.   IF ll_escape
  1243.     SET ESCAPE ON
  1244.   ELSE
  1245.     SET ESCAPE OFF
  1246.   ENDIF
  1247.  
  1248. RETURN
  1249. *-- EOP: Err_Box WITH pc_msg
  1250.  
  1251. **************************** END OF LIBRARY.PRG ******************************
  1252.  
  1253.  
  1254.