/* ** st_start.cpp ** Handles the startup screen. ** **--------------------------------------------------------------------------- ** Copyright 2006-2007 Randy Heit ** All rights reserved. ** ** Redistribution and use in source and binary forms, with or without ** modification, are permitted provided that the following conditions ** are met: ** ** 1. Redistributions of source code must retain the above copyright ** notice, this list of conditions and the following disclaimer. ** 2. Redistributions in binary form must reproduce the above copyright ** notice, this list of conditions and the following disclaimer in the ** documentation and/or other materials provided with the distribution. ** 3. The name of the author may not be used to endorse or promote products ** derived from this software without specific prior written permission. ** ** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR ** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES ** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. ** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, ** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT ** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, ** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY ** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT ** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF ** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. **--------------------------------------------------------------------------- ** */ // HEADER FILES ------------------------------------------------------------ #include #include #include #include #include "st_start.h" #include "doomdef.h" #include "i_system.h" // MACROS ------------------------------------------------------------------ // TYPES ------------------------------------------------------------------- // EXTERNAL FUNCTION PROTOTYPES -------------------------------------------- // PUBLIC FUNCTION PROTOTYPES ---------------------------------------------- // PRIVATE FUNCTION PROTOTYPES --------------------------------------------- static void ST_TTY_Done (); static void ST_TTY_Progress (); static void ST_TTY_NetInit (const char *message, int numplayers); static void ST_TTY_NetProgress (int count); static void ST_TTY_NetMessage (const char *format, ...); static void ST_TTY_NetDone (); static bool ST_TTY_NetLoop (bool (*timer_callback)(void *), void *userdata); // EXTERNAL DATA DECLARATIONS ---------------------------------------------- // PUBLIC DATA DEFINITIONS ------------------------------------------------- void (*ST_Done)(); void (*ST_Progress)(); void (*ST_NetInit)(const char *message, int numplayers); void (*ST_NetProgress)(int count); void (*ST_NetMessage)(const char *format, ...); void (*ST_NetDone)(); bool (*ST_NetLoop)(bool (*timer_callback)(void *), void *userdata); // PRIVATE DATA DEFINITIONS ------------------------------------------------ static termios OldTermIOS; static bool DidNetInit; static int NetProgressMax, NetProgressTicker; static const char *NetMessage; static char SpinnyProgressChars[8] = { '|', '/', '-', '\\', '|', '/', '-', '\\' }; // CODE -------------------------------------------------------------------- //=========================================================================== // // ST_Init // // Sets the size of the progress bar and displays the startup screen. // //=========================================================================== void ST_Init(int maxProgress) { ST_Done = ST_TTY_Done; ST_Progress = ST_TTY_Progress; ST_NetInit = ST_TTY_NetInit; ST_NetProgress = ST_TTY_NetProgress; ST_NetMessage = ST_TTY_NetMessage; ST_NetDone = ST_TTY_NetDone; ST_NetLoop = ST_TTY_NetLoop; } //=========================================================================== // // ST_TTY_Done // // Called just before entering graphics mode to deconstruct the startup // screen. // //=========================================================================== static void ST_TTY_Done() { } //=========================================================================== // // ST_TTY_Progress // // Bumps the progress meter one notch. // //=========================================================================== static void ST_TTY_Progress() { } //=========================================================================== // // ST_TTY_NetInit // // Sets stdin for unbuffered I/O, displays the given message, and shows // a progress meter. // //=========================================================================== static void ST_TTY_NetInit(const char *message, int numplayers) { if (!DidNetInit) { termios rawtermios; fprintf (stderr, "Press 'Q' to abort network game synchronization."); // Set stdin to raw mode so we can get keypresses in ST_CheckNetAbort() // immediately without waiting for an EOL. tcgetattr (STDIN_FILENO, &OldTermIOS); rawtermios = OldTermIOS; rawtermios.c_lflag &= ~(ICANON | ECHO); tcsetattr (STDIN_FILENO, TCSANOW, &rawtermios); DidNetInit = true; atterm (ST_NetDone); } if (numplayers == 1) { // Status message without any real progress info. fprintf (stderr, "\n%s.", message); } else { fprintf (stderr, "\n%s: ", message); } fflush (stderr); NetMessage = message; NetProgressMax = numplayers; NetProgressTicker = 0; ST_NetProgress(1); // You always know about yourself } //=========================================================================== // // ST_TTY_NetDone // // Restores the old stdin tty settings. // //=========================================================================== static void ST_TTY_NetDone() { // Restore stdin settings if (DidNetInit) { tcsetattr (STDIN_FILENO, TCSANOW, &OldTermIOS); printf ("\n"); DidNetInit = false; } } //=========================================================================== // // ST_NetMessage // // Call this between ST_NetInit() and ST_NetDone() instead of Printf() to // display messages, because the progress meter is mixed in the same output // stream as normal messages. // //=========================================================================== static void ST_TTY_NetMessage(const char *format, ...) { FString str; va_list argptr; va_start (argptr, format); str.VFormat (format, argptr); va_end (argptr); fprintf (stderr, "\r%-40s\n", str.GetChars()); } //=========================================================================== // // ST_NetProgress // // Sets the network progress meter. If count is 0, it gets bumped by 1. // Otherwise, it is set to count. // //=========================================================================== static void ST_TTY_NetProgress(int count) { int i; if (count == 0) { NetProgressTicker++; } else if (count > 0) { NetProgressTicker = count; } if (NetProgressMax == 0) { // Spinny-type progress meter, because we're a guest waiting for the host. fprintf (stderr, "\r%s: %c", NetMessage, SpinnyProgressChars[NetProgressTicker & 7]); fflush (stderr); } else if (NetProgressMax > 1) { // Dotty-type progress meter. fprintf (stderr, "\r%s: ", NetMessage); for (i = 0; i < NetProgressTicker; ++i) { fputc ('.', stderr); } fprintf (stderr, "%*c[%2d/%2d]", NetProgressMax + 1 - NetProgressTicker, ' ', NetProgressTicker, NetProgressMax); fflush (stderr); } } //=========================================================================== // // ST_NetLoop // // The timer_callback function is called at least two times per second // and passed the userdata value. It should return true to stop the loop and // return control to the caller or false to continue the loop. // // ST_NetLoop will return true if the loop was halted by the callback and // false if the loop was halted because the user wants to abort the // network synchronization. // //=========================================================================== static bool ST_TTY_NetLoop(bool (*timer_callback)(void *), void *userdata) { fd_set rfds; struct timeval tv; int retval; char k; for (;;) { // Don't flood the network with packets on startup. tv.tv_sec = 0; tv.tv_usec = 500000; FD_ZERO (&rfds); FD_SET (STDIN_FILENO, &rfds); retval = select (1, &rfds, NULL, NULL, &tv); if (retval == -1) { // Error } else if (retval == 0) { if (timer_callback (userdata)) { fputc ('\n', stderr); return true; } } else if (read (STDIN_FILENO, &k, 1) == 1) { // Check input on stdin if (k == 'q' || k == 'Q') { fprintf (stderr, "\nNetwork game synchronization aborted."); return false; } } } } void ST_Endoom() { exit(0); }