ÚÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÑÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍ¿
³ Ü  Ü                             Ü                  ³ Volume 94  Issue 2  ³
³ÜÛÜ Û   Ûßß     ÜÜÜ ÛßÛ Ûßß ÛÛÜ ÜÜÛ ÜÜÜ ÜÜÜ ÜÜÜ Ûßß  ³ September 23, 1994  ³
³ Û  ÛßÛ Ûß      Û Û ÛÜÛ Ûß  Û Þ Û Û Û Û Û Û Û   ßßÛ  ³                     ³
³ ÛÛ Û Û ÛÛÛ     ÛÛÛ Û   ÛÛÛ Û Þ ÛÛÛ ÛÛÛ ÛÛÛ Û   ÛÛÛ  ³ "The greatest thing ³
³ Ü          Ü                               Ü        ³  to happen to       ³
³ÜÛÜ Ûßß ÜÜÜ Û         Û ÜÜÜ Û Û ÜÜÜ ÛÛÜ ßßÛ Û        ³  journalism since   ³
³ Û  Ûß  Û   ÛßÛ     Ý Û Û Û Û Û Û   Û Þ ÛßÛ Û        ³  Real People!"      ³
³ ÛÛ ÛÛÛ ÛÛÛ Û Û     ÛÛÛ ÛÛÛ ÛÛÛ Û   Û Þ ÛÛÛ ÛÛÛ      ³                     ³
³                                                     ³ Ed: Scott Burkett   ³
ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÁÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ

In this action-packed issue:

        - A Word from the Editor
        - OpenDoors 5.0 Released!
        - OpenDoors 5.0 Changes
        - Bits 'n Bytes 'n Data types
        - Adding InterBBS Capabilities to OpenDoors programs
        - Opendoors Tech Corner - Inside OD 5.0's Internal Async Routines
        - Opendoors Tech Corner - Adding Local Mode Functionality
        - Opendoors Tech Corner - Finding physical cursor position in OD 5.0
        - Opendoors Tech Corner - Using UP/DOWN Arrow Keys in OD 5.0
        - Opendoors Tech Corner - Drawing bar graphs with OD!
        - Opendoors Tech Corner - Obtaining Names/Passwords in ASCII Mode
        - Opendoors Tech Corner - Files Transfers under OpenDoors
        - Opendoors Tech Corner - Comparing file stamps
        - Opendoors Tech Corner - Generating Fidonet *.MSG Messages
        - Code Snippets!
        - Opendoors Release Notice: BCHECKERS v1.2
        - Opendoors Release Notice: BFE 4.00.2r
        - Opendoors Release Notice: GIF View Preview Door v1.0
        - Opendoors Release Notice: Operation: Office v.05
        - OpenDoors Tech Journal Information

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ A Word from the Editor ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßß

By: Scott Burkett

After many months of toiling, sweating, complaining, and otherwise
groveling - it is now official - OpenDoors v5.0 has arrived!  Brian has
once again displayed his ability to provide his customers with simply
the best C-based door kit in existence.

I've moved!  Actually, I'm still moving.  My wife and I just moved to
the Tampa Bay, Florida area and so far, so good!  Actually, the author
of RemoteAccess, Andrew Milner lives in Clearwater now - supposedly owns
his own bar.  I would go looking him up, but I'm afraid what might
happen if Andrew and I spent too much time together with his alcohol
supply.  Such is life.

If you haven't noticed by now, this issue is big - no, huge is more like
it.  I've been saving bits and pieces here and there.  Actually, I was
waiting on OD 5.0 to be released, and the pile just kept growing.  I
am convinced that the only reason Brian finally released 5.0, was to
diminish my growing stack of code snippets.... :-)

Just a reminder, but please be sure to send me any release notices for
your latest OpenDoors creations!  Without being notified, chances are,
it won't appear in ODTJ.  Free advertising is nice, isn't it.... :-)

Alles Klaar und spater! (Until the next issue...)

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ OpenDoors 5.0 Released! ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßß

By: Brian Pirie

This edition of the ODTJ marks the release of OpenDoors 5.00.  Version 5.00
embodies some major steps forward for OpenDoors, including built-in
asynchronous serial I/O capabilities, support for non-Borland C compilers
such as Microsoft C, and a set of new advanced screen, window and popup menu
functions. Many small enhancements have also been made, along with many
efficiency improvements. As the long "new features and enhancements" list for
version 5.00 will testify, this new release is one of the largest upgrades
ever made to the capabilities of OpenDoors. Version 5.00 has also been more
extensively tested than any previous version of OpenDoors, making it one of
the most stable and efficient foundations on which you could choose to build
your on-line software.

Yet, even with all of the improvements that have been made for OpenDoors
5.00, work is already going ahead for version 5.10. The aim of version 5.10
won't be so much to add new features, as to improve on already existing
features. More efficiency improvements and greater flexibility is the goal of
OpenDoors 5.10.

What will come after OpenDoors 5.10? More than anything else, the future
directions of OpenDoors is guided by the programmers who work with it. Your
suggestions, ideas and even bug reports will determine what enhancements will
continue to be made to OpenDoors. Please don't hesitate to pass along any
changes that you would like to see in future versions of OpenDoors! I am also
interested to find out how much interest there would be in the following:

-   An OS/2-specific version of OpenDoors
-   A Windows-specific version of OpenDoors
-   Graphics-mode RIP support
-   A programmer-configurable door installation/configuration utility
    that would work in conjunction with the OpenDoors configuration file
    system.
-   A multi-node file system that would facilitate easy implementation
    of multi-node doors with the ability to send messages between nodes
    and control concurrent access to configuration and user files.

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ OpenDoors 5.0 Released! ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßß

By: Brian Pirie

Version 5.00 represents several major steps forward for OpenDoors. In
addition to numerous bug fixes and minor improvements, a number of major
new features have been added to this version. These include an optional
multiple personality system which allow the sysop to choose the status
line and function key style they prefer. This version also adds
text-mode support for RIP (Remote Imaging Protocol) graphics, and adds a
group of advanced ANSI/AVATAR/RIP functions for scrolling areas of the
screen, saving and restoring portions of the screen and creating pop-up
windows and menus. Also new in this version is support for compilers
other than Borland/Turbo C(++), such as compilers from Microsoft.
Version 5.00 also adds built-in communications support, making the use
of a FOSSIL driver optional. Furthermore, direct support for additional
BBS systems has been added.

The list below provides more detail of the changes and new features in
version 5.00. For a summary of changes made for previous versions of
OpenDoors, refer to the OpenDoors History file that is available from my
system.

     - The nonstop key ([=]) now works correctly during FILES.BBS
       listing.

     - New door information file formats now supported include: RA 2.00
       EXITINFO.BBS.

     - If the TASK environment variable is set, OpenDoors will now use
       its value to determine the current node number.

     - The od_control.od_spawn_freeze_time variable now works correctly.
       Previously, the user's time would always be frozen during
       od_spawn...() execution, regardless of the value of this
       variable.

     - A new feature known as the "Multiple Personality System" has been
       added to this version. If you choose to include the Multiple
       Personality System in a door, the sysop will be able to specify
       which of a number of "personalities" should be used. Each
       personality defines the statusline appearance and function keys
       seen by the sysop, and has no effect on the door's operation from
       the user's standpoint. OpenDoors 5.00 includes personality
       definitions for WildCat, RemoteAccess, PC-Board, and it's own
       simplified RA style status lines. You can also define your own
       personalities by writing a personality definition function. If
       your choose not to include the Multiple Personality System in a
       door, you will still be able to define which single personality
       you wish OpenDoors to use.

     - This version of OpenDoors can be used with a larger variety of
       compilers than where supported by the previous version. OpenDoors
       5.00 is known to work with all versions of Turbo C, Turbo C++,
       Borland C++, Microsoft C, Microsoft C++, Quick C and Visual C++.
       It should also work with any other MS-DOS based ANSI C compiler
       that supports the Microsoft/DOS .OBJect and .LIBrary file
       formats.

     - A new diagnostics feature has been added to OpenDoors, which
       allows you to determine the reason for the most recent OpenDoors
       function failure. When any OpenDoors function returns a failure
       condition, it also sets the new od_control.od_error variable to
       indicate the reason for the failure.

     - Added additional definitions to OPENDOOR.H, to map names of
       OpenDoors functions and variables with the word "colour" from the
       U.S. spelling "color". In other words, both od_set_colour() and
       od_set_color() are now recognized by OpenDoors.

     - The od_list_files() now supports more intelligent path
       specifications. If the parameter to od_list_files() is NULL or
       empty, it will search for a FILES.BBS file in the current
       directory. If a directory path is specified, it will look for a
       FILES.BBS in that directory. If a full directory and filename are
       specified, the specified filename will be used in place of
       FILES.BBS.

     - To save space, the compact memory model library is no longer
       included in the normal OpenDoors package. The compact memory
       model library is now available seperately.

     - A new function, od_set_dtr(), has been added to allow the DTR
       line to the modem to be manually controlled. This can be useful
       in writing programs where you wish to force the modem to hangup,
       such as a call-back verification door.

     - Added additional support for various DOS multitasking
       environments. OpenDoors is now specifically Microsoft Windows
       aware. OpenDoors also now gives up time to other waiting tasks
       when it is idle during chat mode.

     - The od_edit_str() "M" mode now capitializes a character following
       a dash '-' character.

     - When transmitting more than one character at a time, OpenDoors
       now uses the FOSSIL trasfer block function, instead of multiple
       calls to the transfer character function. This should help to
       improve performance over high speed connections when running on
       slow PCs or under multitasking environments.

     - OpenDoors 4.10 would not correctly change the display colour from
       high-intensity back to low-intensity. This problem has been
       fixed.

     - OpenDoors now recognizes DORINFO?.DEF filenames with alphabetical
       identifiers (ie, DORINFOA.DEF thru DORINFOZ.DEF) for nodes 10
       thru 35.

     - Improvements have been made to the logfile system. An exit at
       errorlevel zero no longer causes garbage to be written to the
       logfile. The logfile functions have been made more reliable when
       operating under low stack availability conditions. In the past,
       if a large number of local variables where allocated on the
       stack, the logfile functions would fail, often writing garbage to
       the logfile. When the user pages the sysop for chat, the user's
       reason for wishing a chat is also written to the logfile.

     - Support for text-mode RIP (Remote Imaging Protocol) graphics has
       been added. Because this version of OpenDoors always operates in
       DOS text-mode, none of the graphics mode RIP features (such as
       drawing lines, circles and displaying icons) will appear on the
       local screen. Plans for a version of OpenDoors that will operate
       in graphics mode and optionally display graphics locally are
       currently under consideration. In this version, RIP support
       includes a number of new features. OpenDoors will now recognize
       the RIP setting passed in an RA 2.00 EXITINFO.BBS file and
       WildCat DOOR.SYS file, and also allows the RIP setting to be
       specified in a custom door information file. The od_send_file()
       and od_hotkey_menu() functions will now search for files with
       .RIP, .AVT, .ANS and .ASC extensions. When displaying RIP
       graphics to the remote user, a pop-up window appears on the local
       screen, indicating to the sysop which file is being displayed.

     - A set of new functions have been added to permit advanced screen
       manipulations. These functions include od_gettext() and
       od_puttext() to save and restore portions of the screen, and
       od_save_screen() and od_restore_screen() to save and restore the
       entire screen. od_scroll() can be used to scroll any portion of
       the screen upwards or downwards. od_save_screen() and
       od_restore_screen() will operate in any mode, but the other
       functions require ANSI/AVATAR/RIP mode to be available.

     - Three additional functions, od_window_create(),
       od_window_remove() and od_popup_menu(), have been added to
       facilitate the creation of popup windows and menus. When such a
       window or menu is removed from the screen, the are of the screen
       "under" the window is returned to it's original state. This
       allows you to create multiple overlapping windows within a door
       program. The od_popup_menu() function creates a popup window with
       a menu from a simple menu definition string. The user can select
       an option from this menu by pressing the key associated with an
       option, or by moving a menu selection bar using their arrow keys.
       These three functions require an ANSI/AVATAR/RIP mode to be
       available.

     - A new function, od_chat() has been added, to allow you to
       explicitly invoke the OpenDoors chat mode from within your
       program.

     - A new setting variable, od_control.od_always_clear has been
       added. When set to TRUE, od_clr_scr() will always clear the
       screen, regardless of the user's screen clearing setting. When
       set to FALE, od_clr_scr() will only clear the screen if the user
       has screen clearing enabled.

     - It is now possible to configure the errorlevels OpenDoors exits
       with under various circumstances, such as when the user runs out
       of time remaining online. See the od_control.od_errorlevel
       variable

     - A new setting variable, od_control.od_force_local, can be used to
       easily force OpenDoors to operate in local mode. Using this
       variable you can easily add a command line parameter such as
       "-local" to allow the sysop to force your door to operate in
       local mode. When OpenDoors is forced into local mode using this
       variable, it does not look for a door information file, and uses
       default settings for the user's name, etc.

     - OPENDOOR.H now sets structure packing to single byte alignment
       for the od_control structure when Borland and Microsoft compilers
       are being used. In the past, programmers using OpenDoors have
       experienced difficulties the od_control structure when the
       compiler has been set to use word packing.

     - OpenDoors now closes the FOSSIL driver prior to performing a
       spawn or sysop DOS shell. This allows doors or other
       communications programs which use the FOSSIL driver to be
       executed while the door's execution is suspended.

     - When used with a FOSSIL driver, OpenDoors normally changes the
       BPS rate to that passed from the BBS (if the BBS passes a valid
       FOSSIL BPS rate). This BPS rate setting may now be disabled by
       setting the DIS_BPS_SETTING bit of the od_control.od_disable
       variable.

     - A function hook has been added to allow you to install a function
       to be called whenever od_kernel() executes
       (od_control.od_ker_exec). Another function hook,
       od_control.od_time_msg_func, can be installed to override
       OpenDoor's time limit warning messages.

     - A new array, od_control.od_hot_function, allows the you to define
       functions to be called when any of the programmer-defined sysop
       hotkeys have been pressed.

     - A function hook, od_control.od_no_file_func, has been added. This
       function will be called whenever OpenDoors is unable to find or
       read a door information file. This allows you to add your own
       door information file reader, or to provide a local login prompt
       when no door information file is present.

     - Previously, OpenDoors would stop correctly updating the user's
       remaining time at midnight when running under certain BIOSes.
       This problem has been fixed.

     - The current display colour attribute can now be accessed through
       an control structure member, od_control.od_cur_attrib.

     - od_send_file() and od_hotkey_menu() no longer pause with a
       "Continue?" prompt prematurely in files that have line lengths
       greater than 254 characters.

     - The local keyboard may now be disabled by setting the
       DIS_LOCAL_INPUT bit of od_control.od_disable. This only affects
       the sysop's input in circumstances that input is also accepted
       from the remote user; this setting has no effect on the sysop
       function keys.

       A new function hook:

          void (*od_control.od_local_input)(int);

       has been added. If set, this function will be called whenever the
       sysop presses a non-sysop-function key on the local keyboard.

     - od_control.od_clear_on_exit now controls whether the screen is
       cleared before shelling or executing od_spawn...(), in addition
       to before OpenDoors shuts down.

     - od_page() now restores the original display colour before
       returning.

     - It is now possible to display an entire string of characters with
       terminal emulation, using the new function od_disp_emu().

     - OpenDoors will now display a small popup window when
       disconnecting the current connection.

     - A new variable, od_control.od_in_buf_size, can now be set prior
       to calling any OpenDoors function to set the size of OpenDoors
       combined local/remote keyboard input buffer. By default, this
       buffer is 256 bytes in size.

     - Previously, there were a number of OpenDoors API functions that
       would not correctly initialize OpenDoors if they were the first
       function called in the program. This has been fixed.

     - To facilitate setting of bps rates up to 115,200, od_control.baud
       is now an unsigned long.

     - A new setting, od_control.od_no_ra_codes, has been added to
       disable the use of RemoteAccess/QuickBBS control codes by
       od_send_file()/od_hotkey_menu()/od_disp_emu(). The
       RemoteAccess/QuickBBS ASCII 1 "pause for key" is also now
       recognized.

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ Bits and Bytes and Data Types ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ InterBBS Capabilities are Here! ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß

*** Editor's Note: The following is the press release for Brian's
InterBBS add-on library for OpenDoors.  It can be obtained by FREQ'ing
IBBS from either his system or mine.  Enjoy!

BBS "door" programs have always allowed operators of online computer
services, such as electronic bulletin board systems, to easily expand the
services provided by their systems. Doors have greatly increased the
flexibility and usability of BBS systems. A recent trend in door development
has been to allow doors running on one BBS system to be linked to doors
running on other systems, often through existing mail networks. While these
networks have traditionally provided private and conference (echo) mail
services and file sharing capabilities, they can also be useful in providing
links between special applications that wish to share information with one
another.

By taking advantage of existing mail network technology, BBS door developers
have been able to add a new dimension to the software they create. Among the
new ideas that have proven successful have been distributed BBS listing
services, distributed file listing services and games that share top-score
lists or even allow players to compete with people playing the game hundreds
or thousands of miles away. The possibilities are really limited only by the
imagination of the door software developer.

This package is designed to assist in writing Inter-BBS door programs that
can communicate across FTSC-compatible mail networks, such as FidoNet. Using
this package, you can:

       - Send information messages through any FTSC-compatible mailer
         that supports *.MSG format netmail, including FrontDoor,
         InterMail, BinkleyTerm, D'Bridge, and others.

       - Send messages containing either textual or binary data. This
         information is stored in the message body, to allow the message
         to be sent either directly (CrashMail) or by routed netmail.

       - Easily send information to just one other system, or all
         systems running the door. This allows you to implement both
         centralized and de-centralized information distribution
         schemes.

This package includes the source code for an example skiing game door that
maintains a high score list across multiple BBSes. This game requires the
OpenDoors door programming toolkit to be compiled. However, the Inter-BBS
Toolkit itself can also be use in doors that are not written with OpenDoors.

The Inter-BBS Toolkit is available free of charge, and the package includes
the complete C source code. This source code has been written for Borland C
and C++ compilers, but may easily be modified for use with other C compilers.
With the source code, you are able to customize or expand the capabilities of
the Inter-BBS Toolkit as you wish.

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ OpenDoors Tech Corner ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßß

-----------------------------------------------------------------------------
                Inside OD 5.0's Internal Async Routines
                     By: Brian Pirie/Scott Burkett
-----------------------------------------------------------------------------

 > Just a quick thought, but it would be nice if we could have
 > access to the low level communication functions (both fossil and
 > internal).  The standard od...() functions work great for doors,
 > but for critical timing matters such as protocols, etc, it
 > doesn't fly.  I am using my own fossil system for the timing
 > essential portions of BFE, for example.

I will consider adding these functions to the documented OpenDoors interface. In
the meantime, they are as follows:

int _com_carrier(void);           /* TRUE if DCD is high */
void _com_clear_inbound(void);    /* Clears outbound buffer */
void _com_clear_outbound(void);   /* Clears inbound buffer */
void _com_close(void);            /* Closes serial port */
void _com_dtr(char high);         /* TRUE raises DTR, FALSE lowers DTR */
char _com_getchar(void);          /* Returns next char from modem */
void _com_ini(void);              /* Opens serial port */
char _com_inbound(void);          /* FALSE if inbound buffer is empty */
char _com_outbound(void);         /* FALSE if outbound buffer is empty */
void _com_sendchar(char ch);      /* Sends character to modem */
                                  /* Sends multiple chars to modem */
void _com_send_buf(char *buffer,int size);

These functions assume that the door is not operating in local mode. Do not
call com_getchar() unless the _com_inbound() returns TRUE.

-----------------------------------------------------------------------------
                    Adding Local Mode Functionality
                            By: Brian Pirie
-----------------------------------------------------------------------------

OpenDoors 5.00 now permits you to easily force your door to operate in local
mode by setting the od_force_local variable. Normally OpenDoors will use a
default user name, but you can also use this feature to easily provide a
local login prompt for your door. The following example shows how you might
do this:

(This code might look long, but keep in mind that most of it is comments!)

#include "opendoor.h"

int main(int argc, char *argv[])
   {
   int counter;
   char valid_login = TRUE;

   /* Check for /LOCAL command-line parameter */
   for(counter = 1; counter < argc; ++counter)
      {
      strupr(argv[counter]); /* Do this with care! */
      if(strcmp(argv[counter], "/LOCAL")==0)
         {
         /* We are operating in local mode */
         od_control.od_force_local = TRUE;

         /* Let user login */
         do {
            /* Prompt for user name */
            printf("User Name : ");

            /* Get name from user. Note that it would be better  */
            /* to use a console input routine that limits the    */
            /* number of characters that can be inputted by the  */
            /* user, to prevent od_user_name from being overrun. */
            /* I use gets() here for simplicity.                 */
            gets(od_control.od_user_name);

            /* At this point, if you have a user file for your door */
            /* you might want to check whether the user's name is   */
            /* valid. If it is not valid, you can set valid_login   */
            /* to false, to cause door to prompt for a new name.    */
            /* (Keep in mind that you should provide some way for a */
            /* new user to log in locally!)                         */
         } while(!valid_login);

         /* Perform main door operations */
         doormain();

         /* Exit program */
         return(0);
         }
      }

   /* If we get to this point, then there is no /LOCAL parameter. So, */
   /* we should operate normally */

   /* Perform main door operations */
   doormain();

   /* Exit program */
   return(0);
   }


void doormain(void)
   {
   /* Initialize OpenDoors */
   od_init();

   /* Door's main processing, common to remote and local modes, */
   /* here */
   }

-----------------------------------------------------------------------------
               Finding physical cursor position in OD 5.0
-----------------------------------------------------------------------------
                             (Brian Pirie)

 > Quick question:  Is there a method to determine the current
 > location of the cursor, without keeping track of it manually?  I
 > vaguely recall a thread about this subject, but can't find it
 > for the life of me...

In 5.00, the following two variables:

   extern unsigned char phys_curx;
   extern unsigned char phys_cury;

store the current location of the cursor on the local screen.

-----------------------------------------------------------------------------
                 Using UP/DOWN Arrow Keys in OpenDoors 5.0
-----------------------------------------------------------------------------
                   (Taken from the OPENDOORS echo)

 > Any news on when the next beta will be out, and also do you plan
 > on allowing support for use of the UP/DOWN arrow keys?

This has been fixed in the 5.00/beta-8 package. The following program
demonstrates a door which displays a simple message when the up or down arrow
key is pressed. Since OpenDoors defaults to using the up and down arrow keys
for adjusting the user's time limit, such a program has to use different keys
for this purpose. This program reassigns the [Page Up] and [Page Down] keys
to be used for adjusting the user's remaining time.

/* Simple program to demonstrate using arrow keys within an OpenDoors door */
/* program. Since the arrow keys are normally used locally to adjust the   */
/* user's time up or down, you must choose different keys to adjust the    */
/* time limit. This program simply loops, displaying the word "up" if the  */
/* up arrow is pressed, "down" if the down arrow is pressed, and exiting   */
/* if the enter key is pressed.                                            */

#include "opendoor.h"

main()
   {
   int choice;
   char pressed;
   long timer;

   /* Notice call to od_init() here! */
   od_init();

   /* Reassign PageUp and PageDown to increase/decrease time */
   /* Remember to do this after od_init() or your first call to any OpenDoors */
   /* function. */
   od_control.key_lesstime = 0x5100;   /* Scan code for Page Down */
   od_control.key_moretime = 0x4900;   /* Scan code for Page Up */

   od_disp_str("Press up and down arrows keys. Press Enter when done\r\n");

   for(;;)
      {
      pressed = od_get_key(TRUE);

      /* Check for ANSI arrow key sequences */
      if(pressed==27)
         {
         timer=(*(long far *)0x46cL)+2L;
         while(timer>*(long far *)0x46cL && (pressed=od_get_key(FALSE))==0)
            {
            od_kernel();
            }

         if(pressed != '[')
            {
            continue;
            }
         else
            {
            timer=(*(long far *)0x46cL)+9L;
            while(timer>*(long far *)0x46cL && (pressed=od_get_key(FALSE))==0)
               {
               od_kernel();
               }
            if(pressed == 0) continue;
            switch(pressed)
               {
               case 'A':
                  goto up_arrow;

               case 'B':
                  goto down_arrow;
               }
            }
         }

      /* Check for doorway / local arrow key sequences */
      else if(pressed==0)
         {
         /* Get the next key from the keyboard */
         timer=(*(long far *)0x46cL)+9L;
         while(timer>*(long far *)0x46cL && (pressed=od_get_key(FALSE))==0)
            {
            od_kernel();
            }
         if(pressed == 0) continue;

         /* Respond appropriately */
         switch(pressed)
            {
            case 0x48:
up_arrow:
               od_disp_str("Up Arrow.\r\n");
               break;

            case 0x50:
down_arrow:
               od_disp_str("Down Arrow.\r\n");
               break;
            }
         }
      /* exit program if user presses CR or LF */
      else if (pressed == '\n' || pressed == '\r')
         {
         break;
         }
      }


   /* Note that od_exit() is now optional in OpenDoors 5.00/beta-8        */
   /* If you allow the program to return from the main() function without */
   /* first calling od_exit(), the od_exit() code will automatically be   */
   /* performed. Note, however, that this does not allow OpenDoors to     */
   /* determine the errorlevel being used, in order to report the         */
   /* errorlevel in the logfile.                                          */
   return(0);
   }


ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ Drawing Bar Graphs with OD! ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß
By: Brian Pirie

Many different types of programs can be enhanced by the use of graphical
information. Often, this graphical information can take the form of
horizontal bar graphs.

An easy way to draw horizontal bars in door programs written with
OpenDoors, is to use the od_repeat() function. Not only does od_repeat()
allow you to easily form a bar by repeating a particular character the
specified number of times, but it is also a very efficient way to do so.
od_repeat() will take advantage of terminal emulation optimizations,
when available. For instance, a character can be repeated any number of
times with AVATAR by sending a short 3-byte sequence that specifies the
character and number of times to repeat.

How do you calculate the number of character to use to form a bar in
your graph? The DrawHorizontalBar() function, which is provided below,
will do this calculation for you. Simply provide the value to be
represented by this bar, the minimum and maximum possible values, and
the maximum number of character to use to draw the bar. For example, if
you are graphing percentages (which could range from 0% to 100%), and
wanted the graph to fit in a space of 40 columns, you would use:

    DrawHorizontalBar(nPercent, 0, 100, 40);

Below the listing for the DrawHorizontalBar() function is a complete program
which demonstrates the DrawHorizontalBar() function as called from another
function that will create complete horizontal bar graphs. This second function,
DrawGraphOfPercentages(), takes an array of titles, and array of values
corresponding to each title, and draws a complete bar graph from this
information.


/* Function to draw a horizontal bar, given a value, the minimum and maximum */
/* possible values, and the number of characters the horizontal bar should   */
/* extended for the maximum value.                                           */
void DrawHorizontalBar(int nValue, int nMinValue, int nMaxValue,
   int nMaxChars)
{
   /* Determine lenght of bar */
   int nBarChars = ((nValue - nMinValue) * nMaxChars) / nMaxValue;

   if(od_control.user_ansi || od_control.user_avatar)
   {
      /* If ANSI or AVATAR graphics are available, assume that IBM extended */
      /* ASCII is also available. This function uses character 220 to form  */
      /* bars that are 1/2 the height of the line. You might also want to   */
      /* try character 119, which will form bars that are the entire height */
      /* of the line.                                                       */
      od_repeat(220, nBarChars);
   }
   else
   {
      /* In ASCII mode, the bar is formed by the '=' character. */
      od_repeat('=', nBarChars);
   }
}

----- ex_graph.c example program follows ------------------------------

/* Includes */
#include "opendoor.h"

/* Function prototypes. */
void DrawHorizontalBar(int nValue, int nMinValue, int nMaxValue,
   int nMaxChars);
void DrawGraphOfPercentages(int nItems, int *panPercentages,
   char **papszTitles, char bTitleColor, char bGraphColor,
   int nTitleWidth, int nGraphWidth);


/* Main function - program execution begins here. */
main()
{
   char *apszTitles[7] = {"Sunday", "Monday", "Tuesday", "Wednesday",
                          "Thursday", "Friday", "Saturday"};
   int anValues[7] = {50, 75, 100, 25, 83, 0, 43};

   od_printf("`bright green`Test graph:\n\r");

   DrawGraphOfPercentages(7, anValues, apszTitles, 0x02, 0x0f, 20, 50);

   od_get_key(TRUE);

   return(0);
}

/* Function to draw horizontal graph of percentages with titles, to */
/* demonstrate the use of the DrawHorizontalBar() function.         */
/* No titles should have more than nTitleWidth characters.          */
void DrawGraphOfPercentages(int nItems, int *panPercentages,
   char **papszTitles, char bTitleColor, char bGraphColor,
   int nTitleWidth, int nGraphWidth)
{
   int nCurrentItem;

   /* Loop for each item (line) in the graph. */
   for(nCurrentItem = 0; nCurrentItem < nItems; ++nCurrentItem)
   {
      /* Set display color for title text. */
      od_set_attrib(bTitleColor);

      /* Add spaces to right-align all titles. */
      od_repeat(' ', nTitleWidth - strlen(papszTitles[nCurrentItem]));

      /* Display the title. */
      od_disp_str(papszTitles[nCurrentItem]);

      /* Add space between title and graph. */
      od_printf(" ");

      /* Set display color for graph. */
      od_set_attrib(bGraphColor);

      /* Draw bar graph for this line. */
      DrawHorizontalBar(panPercentages[nCurrentItem], 0, 100, nGraphWidth);

      /* Move to the next line. */
      od_printf("\n\r");
   }
}


/* Function to draw a horizontal bar, given a value, the minimum and maximum */
/* possible values, and the number of characters the horizontal bar should   */
/* extended for the maximum value.                                           */
void DrawHorizontalBar(int nValue, int nMinValue, int nMaxValue,
   int nMaxChars)
{
   /* Determine lenght of bar */
   int nBarChars = ((nValue - nMinValue) * nMaxChars) / nMaxValue;

   if(od_control.user_ansi || od_control.user_avatar)
   {
      /* If ANSI or AVATAR graphics are available, assume that IBM extended */
      /* ASCII is also available. This function uses character 220 to form  */
      /* bars that are 1/2 the height of the line. You might also want to   */
      /* try character 119, which will form bars that are the entire height */
      /* of the line.                                                       */
      od_repeat(220, nBarChars);
   }
   else
   {
      /* In ASCII mode, the bar is formed by the '=' character. */
      od_repeat('=', nBarChars);
   }
}

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ Obtaining Names/Passwords in ASCII Mode ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß

By: Brett Barnes, THE TOWER BBS 071 636 3957, England

#include <stdio.h>
#include <string.h>
#include "opendoor.h"

void Get_Password(char *string, int length);
void Get_Full_Name(char *string, int length);

void Get_Password(char *string, int length)
{
int key;
int count = 0;

    od_clear_keybuffer();
    while((key = od_get_key(TRUE)) != 13)
    {
        if(key == 0)
            od_get_key(TRUE);

        if(key >= 32 && key <= 127 && count < length)
        {
            string[count++] = key;
            od_putch('*');
        }
        if(key == '\b' && count > 0)
        {
            od_putch('\b');
            od_putch(' ');
            od_putch('\b');
            count--;
            string[count] = '\0';
        }
    }
    strupr(string);/*make string uppercase*/
}

void Get_Full_Name(char *string, int length)
{
int key;
int count = 0;

    od_clear_keybuffer();
    while((key = od_get_key(TRUE)) != 13)/* Drops out when Enter key*/
    {
        if(key == 0)
            od_get_key(TRUE);/*cancels prob with DEL key*/
        /*If key = valid ascii code */
        if(key >= 32 && key <= 127 && count < length)
        {
       /*if uppercase */    if(key >= 65 && key <= 90)
                key = key+32;/*convert to lowercase*/
            if(count == 0 && key >= 97 && key <= 122)
                key = key-32;/*convert to uppercase*/
            else
                if(string[count-1] == ' ' && key >= 97 && key <= 122)
                key = key-32;/*convert to uppercase*/
        /*Stop spaces being put in first*/
            if(key != 32)
            {
                string[count++] = key;
                od_putch(key);
            }
            else if(count > 0)
            {
                string[count++] = key;
                od_putch(key);
            }
        }
        if(key == '\b' && count > 0)/*If Backspace*/
        {
            od_putch('\b');
            od_putch(' ');
            od_putch('\b');
            count--;
            string[count] = '\0';
        }
    }
}

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ File Transfers Under OpenDoors ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß
By: Brian Pirie, taken from the OPENDOORS echomail conference:

Often, it can be useful or necessary to send and receive files between
a program using OpenDoors, and the remote user's system. To allow file
uploading and downloadeding, you can either implement the common file
transfer protocols as part of your program, or you can call an external
file transfer program, such as DSZ. This tip demonstrates how you can
call DSZ using OpenDoors.

In order to see this program in action, you should run a terminal
program on one machine, and connect to second machine that will run this
example program. (You could also do this in two different windows on one
machine.) The demonstration program will ask whether you want to upload
or download files, and will then prompt you for the file transfer protocol
to use and the filename to transfer. Once this is done, DSZ is invoked
to perform the file transfer. As such, DSZ will need to be installed on
the machine that will run the example program.

The tip2.c program provides a function that you can use to perform file
transfers. This function is defined as follows:

  int TransferFile(char *pszFilename, eProtocol Protocol, char bReceive)

The first parameter should contain the name of the file or files to be
transfered. The second parameter should specify the file transfer
protocol to use, and should be one of the following enumerated
constants:

   XModem
   XModemCRC
   XModem1K
   YModem
   YModemG
   ZModem

The third parameter specifies whether the file is being send (FALSE) or
received (TRUE). From the user's perspective, sending the file means
downloading, and receiving the file means uploading.

The TransferFile() function returns TRUE if the file transfer was
completed, and FALSE if it was not.

If the DSZ program is not present in the system's PATH or the current
directory, then the global variable szDSZFilename must be set to the
full path and filename of the DSZ program. This could be done by adding
a custom OpenDoors configuration file line with a keyword such as
"DSZPath".

The first part of the tip2.c program discussed in the previous message
follows:

/* tip2.c - Example program to demonstrate how to send or receive files */
/*          using DSZ, from within an OpenDoors program.                */

/* Include required header files. */
#include <stdio.h>
#include <assert.h>
#include "opendoor.h"

/* File transfer protocol enumeration. */
typedef enum
{
   XModem,
   XModemCRC,
   XModem1K,
   YModem,
   YModemG,
   ZModem
} eProtocol;

/* Function prototypes. */
int TransferFile(
   char *pszFilename,
   eProtocol Protocol,
   char bReceive);
eProtocol ChooseProtocol(void);
void AddParameter(
   char **papszArguments,
   int *pnCount,
   char *pszNewArgument);

/* Manifest constants. */
#define ARGS_ARRAY_SIZE 10

/* Global variable with DSZ filename. */
char szDSZFilename[80] = "DSZ";


/* Program's execution begins here. */
main()
{
   char chAnswer;
   char bReceive;
   eProtocol Protocol;
   char szFilename[73];
   int bSuccess;

   od_printf("OpenDoors file transfer demo.\n\r\n\r");

   /* Get file transfer direction from user. */
   od_printf("Do you wish to [U]pload or [D]ownload? ");
   chAnswer = od_get_answer("UD");
   if(chAnswer == 'U')
   {
      od_printf("Upload\n\r\n\r");
      bReceive = TRUE;
   }
   else
   {
      od_printf("Download\n\r\n\r");
      bReceive = FALSE;
   }

   /* Get file transfer protocol from user. */
   Protocol = ChooseProtocol();

   /* Get filename. */
   od_printf("\n\rEnter filename(s) : ");
   od_input_str(szFilename, 72, ' ', 255);
   od_printf("\n\r");

   /* Perform file transfer. */
   bSuccess = TransferFile(szFilename, Protocol, bReceive);

   /* Display result of file transfer to user. */
   od_clr_scr();
   if(bSuccess)
   {
      od_printf("File transfer successful.\n\r");
   }
   else
   {
      od_printf("File transfer not completed.\n\r");
   }

   /* Prompt user to exit program. */
   od_printf("Press [Enter] to return to BBS.\n\r");
   od_get_answer("\n\r");

   /* Return control to calling BBS software. */
   od_exit(0, FALSE);

   return(0);
}


/* Function to allow user to choose a file transfer protocol. */
eProtocol ChooseProtocol(void)
{
   eProtocol Protocol;
   char chAnswer;

   od_printf("Available file transfer protocols:\n\r");
   od_printf("    [X] XModem\n\r");
   od_printf("    [C] XModem/CRC\n\r");
   od_printf("    [1] XModem/1K\n\r");
   od_printf("    [Y] YModem\n\r");
   od_printf("    [G] YModem-G\n\r");
   od_printf("    [Z] ZModem\n\r\n\r");
   od_printf("Please select a protocol: ");

   chAnswer = od_get_answer("XC1YGZ");

   switch(chAnswer)
   {
      case 'X':
         od_printf("XModem\n\r");
         Protocol = XModem;
         break;
      case 'C':
         od_printf("XModem/CRC\n\r");
         Protocol = XModemCRC;
         break;
      case '1':
         od_printf("XModem/1K\n\r");
         Protocol = XModem1K;
         break;
      case 'Y':
         od_printf("YModem\n\r");
         Protocol = YModem;
         break;
      case 'G':
         od_printf("YModem-G\n\r");
         Protocol = YModemG;
         break;
      case 'Z':
      default:
         od_printf("ZModem\n\r");
         Protocol = ZModem;
         break;
   }

   return(Protocol);
}

/* Function to send or receive a file to/from remote system. */
int TransferFile(
   char *pszFilename,
   eProtocol Protocol,
   char bReceive)
{
   char szPort[7];
   char *apszArguments[ARGS_ARRAY_SIZE];
   int nArgCount = 0;

   /* Filename cannot be NULL. */
   assert(pszFilename != NULL);

   /* Ensure that we are not operating in local mode. */
   if(od_control.baud == 0)
   {
      od_printf("Operating in local mode;"
                " file transfer not possible.\n\r");
      return(FALSE);
   }

   /* Generate DSZ command line */

   /* Begin with executable filename. */
   AddParameter(apszArguments, &nArgCount, szDSZFilename);

   /* Add port parameter. */
   AddParameter(apszArguments, &nArgCount, "port");
   sprintf(szPort, "%d", od_control.port + 1);
   AddParameter(apszArguments, &nArgCount, szPort);

   /* Restrict inbound files to current drive and directory. */
   AddParameter(apszArguments, &nArgCount, "restrict");

   /* Generate DSZ protocol parameters from specified protocol. */
   if(bReceive)
   {
      switch(Protocol)
      {
         case XModem:
         case XModem1K:
            AddParameter(apszArguments, &nArgCount, "rx");
            break;
         case XModemCRC:
            AddParameter(apszArguments, &nArgCount, "rc");
            break;
         case YModem:
            AddParameter(apszArguments, &nArgCount, "rb");
            break;
         case YModemG:
            AddParameter(apszArguments, &nArgCount, "rb");
            AddParameter(apszArguments, &nArgCount, "-g");
            break;
         case ZModem:
            AddParameter(apszArguments, &nArgCount, "rz");
            break;
         default:
            assert(FALSE);
      }
   }
   else
   {
      switch(Protocol)
      {
         case XModem:
         case XModemCRC:
            AddParameter(apszArguments, &nArgCount, "sx");
            break;
         case XModem1K:
            AddParameter(apszArguments, &nArgCount, "sx");
            AddParameter(apszArguments, &nArgCount, "-k");
            break;
         case YModem:
         case YModemG:
            AddParameter(apszArguments, &nArgCount, "sb");
            break;
         case ZModem:
            AddParameter(apszArguments, &nArgCount, "sz");
            break;
         default:
            assert(FALSE);
      }
   }

   /* Add filename parameter to command line. */
   AddParameter(apszArguments, &nArgCount, pszFilename);

   /* Display prompt to user providing */
   od_printf("Begin your transfer now,"
             " or press [Ctrl]-[X] several times to abort.\n\r");

   /* Execute command using the OpenDoors od_spawn() function. */
   return(od_spawnvpe(P_WAIT, apszArguments[0], apszArguments,
                      NULL) == 0);
}


/* Function to add next parameter to array of parameters to pass to */
/* od_spawnvpe().                                                   */
void AddParameter(
   char **papszArguments,
   int *pnCount,
   char *pszNewArgument)
{
   assert(*pnCount < ARGS_ARRAY_SIZE - 1);
   assert(papszArguments != NULL);
   assert(pnCount != NULL);
   assert(pszNewArgument != NULL);

   /* Add next argument to array. */
   papszArguments[(*pnCount)++] = pszNewArgument;

   /* Ensure that the array is always NULL-terminated. */
   papszArguments[*pnCount] = NULL;
}

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ Comparing File Stamps ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßß

By: Brian Pirie, taken from the OPENDOORS echomail conference:

 > I am trying to find a way to basically look at a files date and
 > then determine if it is older than say 10 days old?
 > I looked thru all of my manuals, and sample programs and could
 > find no examples, so I thought I would turn to the experts here.
 > If anyone can give me help I would greatly appreciate it!

For others who were wondering about this problem, the program below will
display the names of all files in the current directory that are more than
10 days old.

If you ran the program at 10:35:20 on Friday, August 5th, a file dated
10:35:20 on Tuesday, July 26th (or later) would not be displayed, but a file
dated 10:35:18 on Tuesday, July 26th (or earlier) would be displayed.
(DOS's file time is only accurate to the nearest 2 seconds.)

**** Editor's Note: To further explain Brian's information, here is a
brief explanation of why DOS file time stamps are only accurate to the
nearest two seconds.  The file's time field is set when you create the file,
and updated thereafter whenever you close the file, but *only* if
information has been written to the file.  This field is not updated if
the file is merely read, copied, or renamed.  When the time field gets
updated, though, the new time is retrieved from the system clock.  The
following is a breakdown of the meaning of each bit in the time field:

        FEDCAB98 76543210

        xxxxx              = Hours
             xxx xxx       = Minutes
                    xxxxx  = Two second increments

The hour is contained in five bits (on a 24 hour clock), and the minutes
in six bits.  Because this only leaves 5 bits in which to store the
seconds, DOS divides the actual value by two (2).  A little known fact
about the file time field, is that if all bits in both bytes are set to
zero (0), the DIR command will not show any time.  Ok, back to Brian's
snippet (sorry!).

Included in this file is a function named DIRToCTime(), which converts the
DOS file time format to the C time_t format.
                       
/* This example program will display the names of all files that are */
/* more than ten days old.                                           */

#include <dir.h>
#include <dos.h>
#include <time.h>
#include <stdlib.h>

time_t DIRToCTime(unsigned uDate, unsigned uTime);

#define SECONDS_PER_DAY (60 * 60 * 24)

main()
{
   struct ffblk DirEntry;
   time_t CurrentTime = time(NULL);
   time_t FileTime;
   double dElapsedSeconds;
   double dElapsedDays;

   if(!findfirst("*.*", &DirEntry, FA_ARCH))
   {
      do
      {
         /* This loop repeats for every found file ... */

         /* Convert the file's time as returned by findfirst() to a time_t. */
         FileTime = DIRToCTime(DirEntry.ff_fdate, DirEntry.ff_ftime);

         /* Determine the number of seconds that have elapsed between the */
         /* two times.                                                    */
         dElapsedSeconds = difftime(CurrentTime, FileTime);

         /* Determine the number of days that have elapsed. */
         dElapsedDays = dElapsedSeconds / SECONDS_PER_DAY;

         /* If more than ten 24-hour periods have elapsed, display filename. */
         if(dElapsedDays > 10)
         {
            printf("%s is more than ten days old.\n", DirEntry.ff_name);
         }
      } while(!findnext(&DirEntry));
   }

   return(0);
}


time_t DIRToCTime(unsigned uDate, unsigned uTime)
{
   struct tm TimeStruct;

   TimeStruct.tm_sec = (uTime & 0x001f) * 2;
   TimeStruct.tm_min = (uTime & 0x07e0) >> 5;
   TimeStruct.tm_hour = (uTime & 0xf800) >> 11;
   TimeStruct.tm_mday = uDate & 0x001f;
   TimeStruct.tm_mon = ((uDate & 0x01e0) >> 5) - 1;
   TimeStruct.tm_year = 80 + ((uDate & 0xfe00) >> 9);

   return(mktime(&TimeStruct));
}

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ Generating Fidonet *.MSG Messages ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß
By: Mark Williamson, taken from the OPENDOORS echomail conference:

/* NETMAIL.C : This little jewel will write send a netmail message to
               the names and addresses listed in NAMES.LST. This file
               has the format of:

               First Lastname
               Zone Node Net Point  (point is optional)

               example:

               Mark Williamson
               1 202 750 <0>

               When the program loads, it will load the editor EDIT.EXE.
               Just change the line system("EDIT.EXE") to load your editor.
               This program expects to find a file NET.MSG.

               This program is public domain <G>.
               The attribute used in this program is CRASH KILL SENT.

               Look in the Fidonet Technical Specifications for the full
               structures and meaning of each field.
*/

#include <dos.h>
#include <stdio.h>
#include <time.h>
#include <stdlib.h>
#include <fcntl.h>
#include <alloc.h>
#include <bp.h>
#include <string.h>

static char msghdr[50],year[5],msgfile[80],firstname[30];
char *msgtext,select;
void writefido(void);

char  *strrep(char *str, char *oldstr, char *newstr)
{
          int OldLen, NewLen;
          char *p, *q;

          if(NULL == (p = strstr(str, oldstr)))
                return p;
          OldLen = strlen(oldstr);
          NewLen = strlen(newstr);
          memmove(q = p+NewLen, p+OldLen, strlen(p+OldLen)+1);
          memcpy(p, newstr, NewLen);
          return q;
}

/* *.MSG format: */
struct {
         char from[36],
                  to[36],
                  subject[72],
                  datetime[20];
             int  timesread,
                  destnode,
                  orignode,
                  cost,
                  orignet,
                  destnet,
                  destzone,
                  origzone,
                  destpoint,
                  origpoint,
                  replyto,
                  attribute,
                  nextreply;
        } Msg;


char buffer[89];
void main(void)
{

   FILE *fp,*fp2;
   struct tm *time_now;
   time_t secs_now;
   char zone[10],node[10],net[10],point[10];

   /* load your editor to create/change the message. */

   system("EDIT C:\\NET.MSG");
   if(access("C:\\NET.MSG",0)!=0) return;

   /* open the name list */

   fp=fopen("C:\\NAMES.LST","rt");
   if(fp==NULL) return;
   time(&secs_now);
   Msg.timesread=0;
   Msg.cost=0;


   strcpy(Msg.from,"Mark Williamson");
   strcpy(Msg.subject,"Labtest Beta Message"); /* remember, the subject */
                                               /* for a file attach message */
                                               /* is the full path and file */
                                               /* name of the file(s) to send
*/
                                               /* be sure to set the correct
*/
                                               /* message attribute to send */
                                               /* or request files. */


   Msg.attribute=0x0183;

   /* change this to your fido address */
   Msg.orignode=750;
   Msg.orignet=202;
   Msg.origzone=1;
   Msg.origpoint=0;
   Msg.replyto=0;
   Msg.nextreply=0;
   time_now=localtime(&secs_now);
   strftime(Msg.datetime,20,"%a %d %b %y %X",time_now);
   while(!feof(fp))
   {
           Msg.destnode=Msg.destpoint=Msg.destnet=0;
           Msg.destzone=1;

           if(fgets(buffer,88,fp)==NULL) break;
           buffer[strlen(buffer)-1]=0;
           strcpy(Msg.to,buffer);
           if(fgets(buffer,88,fp)==NULL) break;
           buffer[strlen(buffer)-1]=0;
           zone[0]=point[0]=node[0]=net[0]=0;
           sscanf(buffer,"%s %s %s %s",zone,net,node,point);
           Msg.destnode=atoi(node);
           Msg.destpoint=atoi(point);
           Msg.destzone=atoi(zone);
           Msg.destnet=atoi(net);

           ultoa(bp(Msg.to,199309),registerkey,10);
           fp2=fopen("C:\\NET.MSG","rb");
           msgtext=(char *)malloc(filelength(fp)+1000);
           memset(msgtext,0,filelength(fp)+1000);
           fread(msgtext,filelength(fp),1,fp2);
           fclose(fp2);
           sscanf(Msg.to,"%s",firstname);


           /* here are some macros to make auto replacement easier. */
           /* this is great for form letters, kinda personnalizes the */
           /* message.*/

           strrep(msgtext,"@FIRST@",firstname);
           printf("\nWriting message to: %s, %s:%s/%s%s%s",
                     Msg.to,
                     zone,net,
                     node,Msg.
                     destpoint!=0?".":"",
                     Msg.destpoint!=0?point:"");
           writefido();
           free(msgtext);
   }
   fclose(fp);
}

void writefido(void)
{
     int count=250;  /* this controls the starting point.  */
                     /* this function will count down until */
                     /* finds a *.MSG.  It will then increment */
                     /* by one and write the message.  There */
                     /* is probably a better way, but you know. */
                     /* set this number higher if you have lots */
                     /* of netmail messages. */

         char firstname[20];
         FILE *fp;
         tzset();

          /* Fido *.msgs use a MSGID line which is preceded by */
          /* a CONTROL-A.  Then your fido address, a carriage  */
         /* return, CONTROL-A and the name of the program that */
         /* created the message and version number, then another */
         /* carriage return. */
         strcpy(msghdr,"\x01MSGID: 1:202/750\r\x01PID: MM 1.0\r");

         /* this begins our loop to find the last written */
         /* netmail message in your netmail directory */
         sprintf(msgfile,"C:\\RA\\MAIL\\%d.MSG",count);
         while(access(msgfile,0))
            sprintf(msgfile,"C:\\RA\\MAIL\\%d.MSG",--count);

         /* got this far, we have a number! */
         /* increment it by one to get the next available slot */
         sprintf(msgfile,"C:\\RA\\MAIL\\%d.MSG",++count);
         fp=fopen(msgfile,"wb");
         if(fp==NULL) puts("\nError opening file...");
         else
         {
           fwrite(&Msg,sizeof(Msg),1,fp);
           fwrite(&msghdr,strlen(msghdr),1,fp);
           fwrite(msgtext,strlen(msgtext),1,fp);
           fwrite("\0x00",1,1,fp);
           fclose(fp);
         }
}

That's it!  This is a simple program I wrote to send form letters to numerous
people who needed to have the same information.  This could be done in
Frontdoor with the ALT-L forward command, but then it was much more fun
creating something and knowing watching it work!

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ Code Snippets! ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßß

Sample code to validate credit card numbers
Posted in the OPENDOORS echo by Robert La Ferte,
Original by Andrue Carr

/*
**  Testcard.c 1.0 Copyright (C) 1993 by DruId Software.
*/

#include <conio.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int formula(char card[23],int len);

// actual formula for creditcard!
// this can be used to build the last CRC digit by changing the len by 1
// it is mainly used to make sure the card is a correct card by verifying
// the last CRC digit against the rest of the card.
// the formula will run all the numbers in the card but the last one
// which is the CRC digit.  if the number the formula kicks out is the
// same as the last digit of the card then it is a valid credit card
// if not then its not correct. this is what a ZON machine uses to
// determine if it is a good number or not. If you need more help
// let me know and i can explain this better:
// Andrue Carr: fido: 1:331/201, SL_net: 250:502/1294, ISG 91:5/201
//              Acmenet: 400:508/201
// box 473, West Tisbury, mass 02575 (Over Board BBS (508)693-5344)

int formula(char card[23], int len)
{
    int     x,
            xcard = 0,
            y;

    for(x = 0; x <= len; x++)
    {
        if(len % 2)
        {
            if(x % 2)
                y = (card[x] - 48) * 2;
            else
                y = (card[x] - 48);
        }
        else
        {
            if(x % 2)
                y = (card[x] - 48);
            else
                y = (card[x] - 48) * 2;
        }
       if(y >= 10)
           y = (y - 10) + 1;

       xcard += y;
    }

    x = (10 - (xcard % 10));

    if(x == 10)
        x = 0;

    return(x);              // send back the computed check digit!
}

void main(int argc, char *argv[])
{
    int i = 0,
        size = 0;

    if(argc < 2)
    {
            // testcard.exe [cardnumber]  if no card# then error
        cputs("\007No credit card entered");
        exit(1);
    }

    size = strlen(argv[1]);

    i = formula(argv[1], (size - 2));

    // check for all but crc digit
    // if the number returned from formula() isnt the last digit
    // of the actual card number then it isnt valid!

    if(i != (argv[1][size - 1] - 48))
        cputs("\007     -> Invalid Card!");
    else
        cputs("     -> Valid Card");

    exit(0);
}

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ OpenDoors Release Notice: BCHECKERS 1.2! ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß

BCHECKERS -- Version 1.2 -- is released!
========================================

** NOW SUPPORTS RA 2.00+! **

As a sysop of a FidoNet BBS, I was disappointed in the lack of a good,
non-interactive checkers door. Sure, there were some that allowed inter-
node play and the like, but these were expensive and few offered any
decent ANSI graphics and the simple ability to have callers make moves
on alternate logons. I also wanted to try my hand at programming in C,
having learned a number of other programming languages. BCheckers is the
result of this effort; and at only $10 is a bargain in shareware.

BCheckers offers the following sysop features (and more I've probably
overlooked in these docs):

-  As you would expect, BCheckers monitors carrier detect functions, to
   automatically recover when a user drops carrier.

-  Includes a fully-adjustable inactivity timeout monitor. A warning is
   sent 5 seconds before the caller is ejected.

-  Share-aware file i/o for use in multi-node BBS systems. You must have
   DOS's SHARE.EXE loaded for multi-node use.

-  Supports most popular BBS door information files, such as DORINFO1.DEF,
   EXITINFO.BBS, CHAIN.TXT, DOOR.SYS, etc.

-  Displays and updates a QuickBBS-style status line, with information
   available to the sysop such as user name, location, baud rate, time left,
   function keys, ANSI and AVATAR settings, and so on.

-  Keeps track of a user "wants-chat" indicator, just like the one in
   RemoteAccess, QuickBBS and other BBS systems. Allows for sysop page from
   the door, and integrated chat mode.

-  Provides the sysop with all the standard function keys for adjusting user
   time, hanging up on or even locking out the user -- and sysop shell to DOS.

-  Full support for locked baud-rates of up to 38,400 baud, using the FOSSIL
   driver for maximum compatibility with any system. If a FOSSIL is not
   available, BCheckers will use its own communications routines. Auto-detect
   of local operation.

-  BCheckers is also DesqView and Windows aware.

New features:

-  Minor bug fix to the Hall of Fame generation and key recognition code.

-  Minor cosmetic improvements.

-  Altered prompts slightly to alleviate problems with some modems when
   using XON/XOFF handshaking.

BCHECKERS is coming soon to a DOORNET or DDSDOORS distribution site in your
area, or you can freq the latest version directly from the author at FidoNet
node 1:231/710 using the "magic name" of BCHECK. Alternatively, the door can
be downloaded from the H.O.M.E. BBS at (317)539-6579. The next release will
have a "play against the computer" mode! Upgrades will be free, but a new
registration will increase by $5. Register now, while it's still cheap!

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ OpenDoors Release Notice: BFE v4.00.2r ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß

BFE v4.00.2r, the flexible telecommunications front end system from
Southeastern DataLINK, is now available!

BFE is a BBS front-end system that was designed to provide sysops with a
fast, efficient method of "connecting" things at the front end of their
site.  

BFE features include:

 * Custom multi-level menus, with RIP graphics support
 * Item and/or menu level password protection with daily scheduling
 * Hotkey items or SlashCommand(tm) mode
 * Attractive internal menuing scheme to ease your work load
 * Support for custom user menus (for aspiring sysops and artists)
 * Support for popup lightbar menus
 * Design and employ interactive full-screen data entry forms
 * Full Multinode/Multiuser Compatibility
 * Internal DOS-to-UNIX gateway!  Give your users the power of UNIX.
 * The gateway services can also be used to gate calls to a packet radio
 * Client/server approach to the UNIX gateway.  Provide interactive access
 * Provide *REAL* UUCP sessions from your UNIX server to your DOS callers
 * Intuitive menu-driven setup and customization facility (BFE/Setup)
 * Feature packed "C" like Script system
 * Run automated scripts, even offline in a maintenance role.
 * WFC (Wait-For-Caller) module to handle non-mailer sites (reg. only)
 * DESQview *and* MS-Windows aware
 * Configurable for security concerns
 * Complete carrier monitoring and timeout checking
 * Use any of 11 standard dropfiles, or define custom ones
 * Run as a normal door or as a frontend (Dropfile not required!)
 * Complete support for ANSI/ASCII/AVATAR/RIP users
 * Configurable automatic ANSI and RIP detection schemes
 * File transfer system with support for external protocols
 * Run remote jobs, such as batch files, programs, other doors, etc.
 * Internal remote shells to the operating system
 * Built in chat/paging system with hourly/daily time restrictions
 * Split screen chat mode, along with line chat mode for TTY users
 * *Total* configurability

What's changed since the last release?

        o New mailing address and contact information for Southeastern
          DataLINK:
                      Address: 13500 Feather Sound Circle West
                               Suite 1313
                               Clearwater, FL 34622
                         Data: (813) 572-6817
                        Voice: (813) 573-5078
                     Internet: scottb@sedl.com  - Scott Burkett
                               sales@sedl.com   - Sales Q/A
                               support@sedl.com - Technical support
                      Fidonet: 1:3603/500

        o BFE now supports "Before" and "After" scripts for each menu option
          in your control files.  The BEFORE scripts are called
          immedicately after the user presses the key corresponding to
          the menu option, but before the menu option's actions are
          carried out.  The AFTER scripts are executed upon completion
          of the menu item selected, and before returning to the BFE
          menu system.  These scripts should be located in the SCRIPTS
          directory (configured in BFE/Setup), and should be named as
          per the following naming convention:

                BEFORE scripts:  <CTLFILENAME>.B<HOTKEY>
                 AFTER scripts:  <CTLFILENAME>.A<HOTKEY>

          For example, if in your MAIN control file, you have a menu
          option which uses the "1" key as the hotkey, the BEFORE and
          AFTER scripts would be named "MAIN.B1" and "MAIN.A1".

        o NEW SCRIPT COMMANDS:

                CompareStr() - Compares a string with the contents of the
                               internal script buffer.

        o NEW SAMPLE SCRIPTS:

                GETPASS.SCR - Demonstrates the use of the CompareStr()
                              function call.

        o BFE's menu handler now sports a rudimentary line noise "filter".

        o All of the text used during user system logins is now
          configurable in the language file.

        * BFE's internal comm routines now supports all IRQ lines from
          1 to 15.  In addition, full support for 16550A FIFO buffers is
          in place as well.

        * The RA "personality" now accurately mimics the latest version
          of the RA status line.

        * Quite a bit of code cleanup in this release.

        * Several additional checks are now in place for the BFE setup
          sanity check, which verifies the BFE path setup.

        * When opening a .CTL file under BFE/Setup, it now defaults to open
          an "existing" .CTL file, instead of creating a new one.  (Thanks
          to George Bynum for this suggestion).

        * Quite a few changes and additions in the BFE user manual (including
          WFC, BFE/Gateway, and BFE/Setup).

        * The "p" identifier has been removed from the end of the normal
          archive naming convention, as it was confusing OS/2 users.  It
          has been replaced by an "r", which means "public (r)elease".

        * The fossil buffers are now flushed before every gateway session
          is initiated.  This was inserted in order to make BFE's internal
          serial gateway more "friendly" towards packet radio systems.
          (Thanks to Mark Stanchin).

        * A popup "dialog" box will appear whenever BFE is dropping
          carrier, informing you of such.

        * The WFC module for registered users will now place both the
          COM port number *and* the fossil port number into the
          DOBBSx.BAT files.

        ! The displayfile() script function should no longer give erroneous
          password prompts if a password was used to get into the script
          to begin with. (Thanks to Mark Stanchin).

        o A new dot command has been added to BFE/Formgen.  The ".type"
          keyword will direct Formgen to output information in either
          "delimited" format (default), or "linear".  Thanks to Dana
          Meli-Wischman.

        ! In certain situations, BFE would force the caller to hit a key
          twice before the keystroke was registered.  Squashed.

        ! BFE/Gateway sessions should now properly monitor the caller's
          carrier signal.

        ! ANSI detection would fail under certain situations.  Squashed.

        o The BFE registration system has been revamped completely.  If you
          are a previously registered user of BFE, please contact us for your
          new key file.

        o Marketing strategy has been updated, and now includes three very
          distinct levels of BFE registration.

        o New Module:   BFE/Node Monitor for DESQview systems.  BFEMON is a
          compact monitoring system designed for sites running BFE in a
          multinode environment.  This module is made available free of charge
          to registered users of BFE. (+)

        o New Module: BFE/WFC - The Wait For Caller module is designed to
          provide a flexible method of handling inbound calls on non-mailer
          nodes.  It's features include:

                - DESQview/Win31/OS2 Aware
                - Intuitive menu-driven setup and windowed environment
                - Internal screen blanker
                - Full support for 50 line VGA mode
                - Designed with multinode systems in mind
                - Custom connect strings for FAX, UUCP, etc.
                - FOSSIL driven, for maximum compatibility
                - Full internal event system

          The WFC module is made available free of charge to registered users
          of BFE. (+)

        o New subsystem: BFE/FormGen - a powerful full screen data entry form
          designer is now included as part of the base BFE package.

        o BFE/Setup now includes an internal user file editor. :-)

        o BFE now supports non-fossil sites by providing an internal serial
          communications system.  This adds three new command line switches:

                -i = No fossil!  Use internal routines
                -u = COMx Base Address (i.e. -u03F8)
                -v = COMx IRQ Line (i.e. -v4)

        o The time adjustment sysop keys (formerly cursor up/down) have been
          remapped to pageup and pagedown.

        o Multiline descriptions with embedded colors are now available.  In
          BFE/Setup, the description field is now two lines long.  Also, BFE
          now supports embedded color tokens in the description field, thus
          allowing you to highlight certain portions of the description of
          each menu item.  ** NOTE:  BFE will not automatically align the
          two lines of text if the first one wraps around!  This will be
          taken care of in a future release, but for now, you will have to
          space them out accordingly.

        o When using BFE's internal files system for providing downloads,
          the file "FILES.BBS" is no longer assumed. :-)  Now, you must
          provide the full path *AND* filename of the file to be used as the
          list file.  This allows you to keep your list files in one shared
          directory.  In addition, the full path is no longer needed in the
          list files.  If the path is not given, BFE will look for each file
          in the directory the list file resides in.

        o New Language File Keywords (Thanks to Michael Stumpp)

                PASSWORD   - "Enter password" prompt - Default is "Password:"
                Y_CONTINUE - Key used to continue in "more" prompting
                N_CONTINUE - Key used to stop in "more" prompting
                S_CONTINUE - Key used to go nonstop in "more" prompting

        o BFE's internal user system now has an additional mode of operation.
          This "Exclusive" mode is identical to the full user system, but
          will not allow new users at all.  This may come in handy if you
          wish to allow only "registered" users on certain nodes, or if you
          run a "member's only" BBS.  (Thanks to Dana Meli-Wischman).

        o NEW SCRIPT COMMANDS:

                FormGen() - Processes a BFE/FormGen Entry Form

                SaveScreen() - Saves contents of current screen

                RestoreScreen() - Restores contents of saved screens

                PopupMenu() - Creates a configurable popup menu

                RemoveFile() - Removes the passed filename (deletes it)

                MakeCustomSem() - Creates a CUSTOM.nnn semaphore in the
                                  IPC directory.

                RemCustomSem() - Removes CUSTOM.nnn semaphore from IPC dir.
                
                NoConsoleLogging() - Temporarily disables console logging
                                     while in a BFE gateway session. (+)

                ConsoleLogging() - Re-enables console logging in BFE gateway
                                   sessions. (+)

                DisableServerStr() - Temporarily disables BFE's remote server
                                     strings. (+)

                EnableServerStr() - Re-enables remote server strings. (+)

                SetAccess() - Adjusts the caller's security level

                UpdateUserRecord() - Update's user's record in user file

        o RIP support updated - when BFE is displaying a RIP file to user,
          the appropriate ANSI or ASCII version will be displayed locally
          on the sysop's console.  Note that this doesn't include the
          default BFE internal menus, although it will in the future.

        o A new utility called COLORUPD.EXE has been provided.  This program
          will read the default color configuration stored in the GLOBALS.CFG
          file, and will update the color maps of all .CTL files in the
          current directory.  This program is distributed in the \UTILS
          subdirectory in the distribution archive.  The source (COLORUDP.C)
          is located in the \DEVKIT subdirectory.  Thanks to Keith Ross for
          this suggestion.

        o If using the BFE User System, BFE will now export the user's name
          and password to two environment variables in the DOS master
          environment upon exiting on an errorlevel (USERNAME and PASSWORD
          are the new environment variables) This should make automatic
          logins through BFE a bit easier, as you can now pass the name and
          password on the command line to your BBS packages.

        o Now, a special "log text" field is attached to each menu item.
          This field serves two purposes:

                - It is used in the log file for the task to describe each
                  event.

                - It is used in the menu item selector screen in BFE/Setup,
                  in place of the description field.

          If no log text is supplied, the normal description will be used.

        o The macro system has been completely revamped, and the following
          new macros are available:

                %U  = User's name
                %W  = User's password
                %H  = User's handle

        o BFE will now periodically perform an Integrity check, to ensure
          that pertinent system files have not been altered by viruses or
          intruders.

        o Some sample .CTL files are now included under (\SAMPLES\CTL).
          These will be updated further as more features are added to the
          product.

        o New networking semaphore files implemented (nnn = node number):

                 SHELL.nnn  - User/sysop in DOS shell
                  XFER.nnn  - User transferring a file
                  CHAT.nnn  - User either paging or in a chat session
               FORMGEN.nnn  - User in a BFE/FormGen Entry Form
                CUSTOM.nnn  - Custom user-defined semaphores

        o New fields in BFE/Setup:

                Default .CTL file Path (for BFE control files)
                Default .FRM file Path (for BFE/Formgen files)
                Default .SCR file Path (for BFE/Script files)
                Default .ANS file Path (for ANS/ASC/AVT/RIP files)

        o New script hook:  This script will be executed automatically
          if it exists when a new user has logged into the system.

        * The graphics detection routines have been replaced, and should
          perform much smoother now.

        * Three new fields have been added to the user file.  They are
          handle, home phone, and work phone.

        * The documentation has been completely reorganized.

        * BFE will no longer automatically put a divider line in between
          normal and global commands when using the internal menus.  If you
          want a divider line, you can just add it to the top of your
          global menu. :-)

        * When displaying standard ASCII files, BFE will now default to light
          gray on black.  In previous releases, the color of the text would
          appear in the color of the user input.
                                                            
        * When running external processes (other doors, in particular), BFE
          will now de-initialize the fossil driver, and re-initialize it upon
          re-entry.  This was due to the fact that some doors left the fossil
          driver in quite a shabby state after exiting, so bad, in fact, that
          BFE would function normally, but not be seen on the remote end!

        * Gateway sessions will now flow much smoother.  So smooth, in fact,
          that we actually ran a SCO/UNIX UUCP session through it. :-) (+)

        * BFE now tries to avoid the use of high bit ascii characters when
          running with ASCII callers.  If any of you have seen ANSI emulation
          under UNIX terminal packages, you know why... :-)

        * Password protection now covers an entire file list when downloading
          from protected areas.  In previous releases, the user had to enter
          the password for each file.  Sounds stupid, doesn't it?  It was.

        * Rewrote the import message file routines when using external
          editors.  You should no longer lose characters under certain
          word wrapping conditions.

FREQ:  BFE from:  Southeastern DataLINK Zoom 28.8  1:3603/500 (813) 572-6817
       345K in size

 FTP:  ftp.csn.org:~/crlhq/bfe4002r.zip

Previously registered users of BFE should contact us through the normal
support channels (email, netmail, dialup) to request your new serialization
keys.

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ OpenDoors Release Notice: GIF View Preview Door v1.0 ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß

Greetings All, from Lone Wolf Software!

****************************************************************
*                    GIF View Preview Door                     *
*         FileName: VUGIF10.ZIP   FileSize: 590k               *
****************************************************************

FEATURES............

o Tag up to 50 GIFs for download.
o Users can convert GIF files to JPG files for fast file transfers.
o Internal file base engine for super quick file handling.
o RIPterm users can view the JPG files while ON LINE!
o Converts large GIF files to small (often less than 10k) JPG files
  that transfer fast. RIPterm users can tag, convert and view GIFs
  while on line in under 30 seconds at 14,400 baud.
o Non RIPterm users (ANSI callers) can use many popular graphic
  file viewers like CShow or The Graphics Work Shop to view JPG
  preview files to see if they want to download the actual GIF.
o State of the art sysop management tools, file base editor, and
  built in GIF graphic file viewer.
o Others may say "On Line GIF Viewing", but they require you to log
  off to view the graphic file, that's not on line viewing. This
  door actually lets RIPterm users view the JPG / GIF while still
  on line.
o Extensive use of RIP through out, both to the user and the sysop.
o Option to disable RIP on either local or remote end.

FREQ the file VUGIF10.ZIP from 1:3813/309 or call the board at
1-918-687-1612 at speeds up to 19,200.

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ OpenDoors Release Notice: Operation: Office v0.5 ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß

                        Operation: Office v0.5
                        ----------------------
                          An OmniWare Product
                          -------------------
                           by Mike Aleksiuk

Get your latest copy from...

Name:        C.R.I.S.I.S. HQ
Number:      686-0449
Node Number: 1:134/31
Magic Name:  OPOFF
File Name:   OPOFFV05.ZIP
Place:       Calgary, AB, Canada

        In this exciting new game, we have you set in the unexciting
role of an "office worker".  But last night you overheard your boss
talking to someone... and for no apparent reason, you decide to kill
him!  The challenge is about to begin...

        Supports all major dropfiles (DOOR.SYS, DORINFO?.DEF, etc).
Configure specific game settings.  Usable player editor for registered
users.  Unregistered users may only delete/look at the player information.

        Features
        --------

        - chat with other players in a "one-liner" fashion
        - use the phone
        - listen to the radio
        - many different enemies to fight, many different things to buy
        - work to make money

        Beta Gamma Wanna-Be
        -------------------

        Version 0.5 is the "Beta Gamma Wanna-Be" version of Operation:
Office.  In other words, it is a gamma/wide beta, for all to test.  It
is fully functioning, but still needs to be refined.

        A Few Future Features
        ---------------------

        - personal player phone numbers!
        - much more detailed fight method!
        - more enemies!

        Please send all comments/bug reports via FidoNet netmail to
1:134/21.10, or conventional mail at the address stated within the
SysOp documentation.  Unfortunately, I don't have the time/money to
return all of your messages... Sorry!

Mike Aleksiuk, Author of "O:O"

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
 ÚÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ¿
ܳ OpenDoors Tech Journal Information ³
ÛÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÙ
ßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßßß

Editors: Scott Burkett
         Southeastern DataLINK BBS
         1:3603/500@fidonet.org
         scottb@sedl.com (Internet EMail)
         root@sedl.com (Internet EMail)
         (813) 572-6817 28.8 Zoom!
         13500 Feather Sound Circle West
         Suite #1313
         Clearwater, FL 34622

         Brian Pirie
         BP ECOMM Systems
         1:243/8@fidonet.org
         brian@bpecomm.ocunix.on.ca (Internet EMail)
         75122,2303 (Compuserve)
         (613) 526-4466 14.4
         1416 - 2201 Riverside Drive
         Ottawa, Ontario
         Canada
         K1H 8K9

Published by and for programmers and users of the OpenDoors Door
Programming Toolkit.  It is a compilation of tips, reviews, and tidbits
pertaining to BBS programming and general usage. The opinions expressed
in this publication do not necessarily represent those of its editors,
the OpenDoors author, or other contributors.

OBTAINING COPIES:  The latest copy of the OpenDoors Tech Journal will always
be available under the magic name of ODTJ.

SUBMISSIONS: You are encouraged to submit articles for publication in the
journal.  Please send all items to one of the afore-mentioned systems via BBS
upload or mailer file/attach.

ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ
   ->< End of The OpenDoors Tech Journal - Volume 94 Issue Number 2 ><-
ÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ