]> de.git.xonotic.org Git - xonotic/darkplaces.git/blobdiff - menu.c
Updated Transfusion map list
[xonotic/darkplaces.git] / menu.c
diff --git a/menu.c b/menu.c
index 3d7e46dfc63d1cdffbbcacb85d71be4601107a30..38ff1e8bb1e46a5d5e328ad578c56482f193769f 100644 (file)
--- a/menu.c
+++ b/menu.c
@@ -8,7 +8,7 @@ of the License, or (at your option) any later version.
 
 This program is distributed in the hope that it will be useful,
 but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 
 See the GNU General Public License for more details.
 
@@ -18,13 +18,8 @@ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 
 */
 #include "quakedef.h"
+#include "image.h"
 
-#ifdef _WIN32
-#include "winquake.h"
-#endif
-
-void (*vid_menudrawfn)(void);
-void (*vid_menukeyfn)(int key);
 
 #define TYPE_DEMO 1
 #define TYPE_GAME 2
@@ -42,6 +37,7 @@ void M_Menu_Main_f (void);
                void M_Menu_Setup_f (void);
                void M_Menu_Net_f (void);
        void M_Menu_Options_f (void);
+       void M_Menu_Options_Effects_f (void);
                void M_Menu_Keys_f (void);
                void M_Menu_Video_f (void);
        void M_Menu_Help_f (void);
@@ -49,6 +45,7 @@ void M_Menu_Main_f (void);
 void M_Menu_LanConfig_f (void);
 void M_Menu_GameOptions_f (void);
 void M_Menu_Search_f (void);
+void M_Menu_InetSearch_f (void);
 void M_Menu_ServerList_f (void);
 
 void M_Main_Draw (void);
@@ -59,6 +56,7 @@ void M_Main_Draw (void);
                void M_Setup_Draw (void);
                void M_Net_Draw (void);
        void M_Options_Draw (void);
+       void M_Options_Effects_Draw (void);
                void M_Keys_Draw (void);
                void M_Video_Draw (void);
        void M_Help_Draw (void);
@@ -66,6 +64,7 @@ void M_Main_Draw (void);
 void M_LanConfig_Draw (void);
 void M_GameOptions_Draw (void);
 void M_Search_Draw (void);
+void M_InetSearch_Draw (void);
 void M_ServerList_Draw (void);
 
 void M_Main_Key (int key);
@@ -76,6 +75,7 @@ void M_Main_Key (int key);
                void M_Setup_Key (int key);
                void M_Net_Key (int key);
        void M_Options_Key (int key);
+       void M_Options_Effects_Key (int key);
                void M_Keys_Key (int key);
                void M_Video_Key (int key);
        void M_Help_Key (int key);
@@ -83,6 +83,7 @@ void M_Main_Key (int key);
 void M_LanConfig_Key (int key);
 void M_GameOptions_Key (int key);
 void M_Search_Key (int key);
+void M_InetSearch_Key (int key);
 void M_ServerList_Key (int key);
 
 qboolean       m_entersound;           // play after drawing a frame, so caching
@@ -153,7 +154,6 @@ void M_DrawBackground(void)
        menu_height = 200;
        menu_x = (vid.conwidth - menu_width) * 0.5;
        menu_y = (vid.conheight - menu_height) * 0.5;
-       //DrawQ_Fill(menu_x, menu_y, menu_width, menu_height, 0, 0, 0, 0.5, 0);
        DrawQ_Fill(0, 0, vid.conwidth, vid.conheight, 0, 0, 0, 0.5, 0);
 }
 
@@ -172,32 +172,18 @@ void M_DrawCharacter (float cx, float cy, int num)
        DrawQ_String(menu_x + cx, menu_y + cy, temp, 1, 8, 8, 1, 1, 1, 1, 0);
 }
 
-void M_Print (float cx, float cy, char *str)
+void M_Print (float cx, float cy, const char *str)
 {
-       /*
-       while (*str)
-       {
-               M_DrawCharacter (cx, cy, (*str++)+128);
-               cx += 8;
-       }
-       */
        DrawQ_String(menu_x + cx, menu_y + cy, str, 0, 8, 8, 1, 1, 1, 1, 0);
 }
 
-void M_PrintWhite (float cx, float cy, char *str)
+void M_PrintWhite (float cx, float cy, const char *str)
 {
        DrawQ_String(menu_x + cx, menu_y + cy, str, 0, 8, 8, 1, 1, 1, 1, 0);
 }
 
 void M_ItemPrint (float cx, float cy, char *str, int unghosted)
 {
-       /*
-       while (*str)
-       {
-               M_DrawCharacter (cx, cy, (*str++)+128);
-               cx += 8;
-       }
-       */
        if (unghosted)
                DrawQ_String(menu_x + cx, menu_y + cy, str, 0, 8, 8, 1, 1, 1, 1, 0);
        else
@@ -239,12 +225,6 @@ void M_BuildTranslationTable(int top, int bottom)
 }
 
 
-void M_DrawPicTranslate (float cx, float cy, char *picname)
-{
-       DrawQ_PicTranslate (menu_x + cx, menu_y + cy, picname, translationTable);
-}
-
-
 void M_DrawTextBox (float x, float y, float width, float height)
 {
        int n;
@@ -293,7 +273,7 @@ void M_DrawTextBox (float x, float y, float width, float height)
 
 //=============================================================================
 
-int m_save_demonum;
+//int m_save_demonum;
 
 /*
 ================
@@ -315,60 +295,12 @@ void M_ToggleMenu_f (void)
                m_state = m_none;
                return;
        }
-       if (key_dest == key_console)
-       {
-               Con_ToggleConsole_f ();
-       }
-       else
-       {
+       //if (key_dest == key_console)
+       //      Con_ToggleConsole_f ();
+       //else
                M_Menu_Main_f ();
-       }
-}
-
-// LordHavoc: FIXME: finish this menu stuff
-#if 0
-#define MAXMENUITEMS 128
-
-typedef struct menuitem_s
-{
-       char *string; // may be text, or an image to use, or a cvar name, depending on the functions used
-       char *description;
-       char *command; // used by command items mainly (when used, this command is executed)
-       cvar_t *cvar; // used for cvar items (sliders, number boxes), value is retrieved from the cvar itself
-       int selectable; // purely decorative if this is false
-       int selected; // true if this menu item is currently selected, used by funcs so they don't need to know anything but fields in the menuitem
-       float selecttime; // the time that this menu item was activated (copied from realtime), used for animating selection flashs and such
-       float color[4]; // current color for the item (may be different than base color, due to selection flash effects)
-       float basecolor[4]; // the base color
-       float x, y, width, height; // width and height are used for mouse selection
-       void(*drawfunc)(struct menuitem_s *item);
-       void(*activefunc)(struct menuitem_s *item);
-//     void(*selectfunc)(struct menuitem_s *item);
-//     void(*deselectfunc)(struct menuitem_s *item);
-       void(*usefunc)(struct menuitem_s *item);
-}
-menuitem_t;
-
-menuitem_t menuitem[MAXMENUITEMS];
-int menuitems;
-
-void menuitem_text_drawfunc(struct menuitem_s *item)
-{
-       // FIXME: handle color flashs and such when selected
-       M_Print (item->x, item->y, item->string);
 }
 
-void menuitem_image_drawfunc(struct menuitem_s *item)
-{
-       // FIXME: handle color flashs and such when selected
-       M_DrawPic (item->x, item->y, item->string);
-}
-
-void menuitem_command_usefunc(struct menuitem_s *item)
-{
-       Cbuf_AddText (item->command);
-}
-#endif
 
 int demo_cursor;
 void M_Demo_Draw (void)
@@ -427,7 +359,6 @@ void M_Demo_Key (int k)
 /* MAIN MENU */
 
 int    m_main_cursor;
-//#define      MAIN_ITEMS      5
 
 int MAIN_ITEMS = 4; // Nehahra: Menu Disable
 
@@ -445,11 +376,13 @@ void M_Menu_Main_f (void)
        else
                MAIN_ITEMS = 5;
 
+       /*
        if (key_dest != key_menu)
        {
                m_save_demonum = cls.demonum;
                cls.demonum = -1;
        }
+       */
        key_dest = key_menu;
        m_state = m_main;
        m_entersound = true;
@@ -490,9 +423,9 @@ void M_Main_Key (int key)
        case K_ESCAPE:
                key_dest = key_game;
                m_state = m_none;
-               cls.demonum = m_save_demonum;
-               if (cls.demonum != -1 && !cls.demoplayback && cls.state != ca_connected)
-                       CL_NextDemo ();
+               //cls.demonum = m_save_demonum;
+               //if (cls.demonum != -1 && !cls.demoplayback && cls.state != ca_connected)
+               //      CL_NextDemo ();
                break;
 
        case K_DOWNARROW:
@@ -642,22 +575,43 @@ void M_Menu_SinglePlayer_f (void)
 
 void M_SinglePlayer_Draw (void)
 {
-       int             f;
        cachepic_t      *p;
 
        M_DrawPic (16, 4, "gfx/qplaque.lmp");
        p = Draw_CachePic ("gfx/ttl_sgl.lmp");
-       M_DrawPic ( (320-p->width)/2, 4, "gfx/ttl_sgl.lmp");
-       M_DrawPic (72, 32, "gfx/sp_menu.lmp");
 
-       f = (int)(realtime * 10)%6;
+       // Transfusion doesn't have a single player mode
+       if (gamemode == GAME_TRANSFUSION)
+       {
+               M_DrawPic ((320 - p->width) / 2, 4, "gfx/ttl_sgl.lmp");
 
-       M_DrawPic (54, 32 + m_singleplayer_cursor * 20, va("gfx/menudot%i.lmp", f+1));
+               M_DrawTextBox (60, 8 * 8, 23, 4);
+               M_PrintWhite (95, 10 * 8, "Transfusion is for");
+               M_PrintWhite (83, 11 * 8, "multiplayer play only");
+       }
+       else
+       {
+               int             f;
+
+               M_DrawPic ( (320-p->width)/2, 4, "gfx/ttl_sgl.lmp");
+               M_DrawPic (72, 32, "gfx/sp_menu.lmp");
+
+               f = (int)(realtime * 10)%6;
+
+               M_DrawPic (54, 32 + m_singleplayer_cursor * 20, va("gfx/menudot%i.lmp", f+1));
+       }
 }
 
 
 void M_SinglePlayer_Key (int key)
 {
+       if (gamemode == GAME_TRANSFUSION)
+       {
+               if (key == K_ESCAPE || key == K_ENTER)
+                       m_state = m_main;
+               return;
+       }
+
        switch (key)
        {
        case K_ESCAPE:
@@ -901,7 +855,7 @@ void M_MultiPlayer_Draw (void)
 
        if (ipxAvailable || tcpipAvailable)
                return;
-       M_PrintWhite ((320/2) - ((27*8)/2), 148, "No Communications Available");
+       M_PrintWhite ((320/2) - ((27*8)/2), 168, "No Communications Available");
 }
 
 
@@ -972,6 +926,54 @@ void M_Menu_Setup_f (void)
        setup_bottom = setup_oldbottom = cl_color.integer & 15;
 }
 
+// LordHavoc: rewrote this code greatly
+void M_MenuPlayerTranslate (qbyte *translation, int top, int bottom)
+{
+       int i;
+       unsigned int trans[4096];
+       qbyte *data, *f;
+       static qbyte pixels[4096];
+       static int menuplyr_width, menuplyr_height, menuplyr_top, menuplyr_bottom, menuplyr_load = true, menuplyr_failed = false;
+
+       if (menuplyr_failed)
+               return;
+       if (menuplyr_top == top && menuplyr_bottom == bottom)
+               return;
+
+       menuplyr_top = top;
+       menuplyr_bottom = bottom;
+
+       if (menuplyr_load)
+       {
+               menuplyr_load = false;
+               f = COM_LoadFile("gfx/menuplyr.lmp", true);
+               if (!f)
+               {
+                       menuplyr_failed = true;
+                       return;
+               }
+               data = LoadLMPAs8Bit (f, 0, 0);
+               Mem_Free(f);
+               if (image_width * image_height > 4096)
+               {
+                       Con_Printf("M_MenuPlayerTranslate: image larger than 4096 pixel buffer\n");
+                       Mem_Free(data);
+                       menuplyr_failed = true;
+                       return;
+               }
+               menuplyr_width = image_width;
+               menuplyr_height = image_height;
+               memcpy(pixels, data, menuplyr_width * menuplyr_height);
+               Mem_Free(data);
+       }
+
+       M_BuildTranslationTable (menuplyr_top*16, menuplyr_bottom*16);
+
+       for (i = 0;i < menuplyr_width * menuplyr_height;i++)
+               trans[i] = palette_complete[translation[pixels[i]]];
+
+       Draw_NewPic("gfx/menuplyr.lmp", menuplyr_width, menuplyr_height, true, (qbyte *)trans);
+}
 
 void M_Setup_Draw (void)
 {
@@ -996,8 +998,10 @@ void M_Setup_Draw (void)
        M_Print (72, 140, "Accept Changes");
 
        M_DrawPic (160, 64, "gfx/bigbox.lmp");
-       M_BuildTranslationTable(setup_top*16, setup_bottom*16);
-       M_DrawPicTranslate (172, 72, "gfx/menuplyr.lmp");
+
+       // LordHavoc: rewrote this code greatly
+       M_MenuPlayerTranslate (translationTable, setup_top, setup_bottom);
+       M_DrawPic (172, 72, "gfx/menuplyr.lmp");
 
        M_DrawCharacter (56, setup_cursor_table [setup_cursor], 12+((int)(realtime*4)&1));
 
@@ -1108,14 +1112,14 @@ forward:
                }
        }
 
-       if (setup_top > 13)
+       if (setup_top > 15)
                setup_top = 0;
        if (setup_top < 0)
-               setup_top = 13;
-       if (setup_bottom > 13)
+               setup_top = 15;
+       if (setup_bottom > 15)
                setup_bottom = 0;
        if (setup_bottom < 0)
-               setup_bottom = 13;
+               setup_bottom = 15;
 }
 
 //=============================================================================
@@ -1241,10 +1245,34 @@ again:
 //=============================================================================
 /* OPTIONS MENU */
 
-#define        OPTIONS_ITEMS   27
-
 #define        SLIDER_RANGE    10
 
+void M_DrawSlider (int x, int y, float range)
+{
+       int     i;
+
+       if (range < 0)
+               range = 0;
+       if (range > 1)
+               range = 1;
+       M_DrawCharacter (x-8, y, 128);
+       for (i=0 ; i<SLIDER_RANGE ; i++)
+               M_DrawCharacter (x + i*8, y, 129);
+       M_DrawCharacter (x+i*8, y, 130);
+       M_DrawCharacter (x + (SLIDER_RANGE-1)*8 * range, y, 131);
+}
+
+void M_DrawCheckbox (int x, int y, int on)
+{
+       if (on)
+               M_Print (x, y, "on");
+       else
+               M_Print (x, y, "off");
+}
+
+
+#define        OPTIONS_ITEMS   28
+
 int            options_cursor;
 
 void M_Menu_Options_f (void)
@@ -1255,23 +1283,20 @@ void M_Menu_Options_f (void)
 }
 
 
-void M_AdjustSliders (int dir)
+void M_Menu_Options_AdjustSliders (int dir)
 {
        S_LocalSound ("misc/menu3.wav");
 
        switch (options_cursor)
        {
-       case 4:
-               Cvar_SetValueQuick (&scr_2dresolution, bound(0, scr_2dresolution.value + dir * 0.2, 1));
-               break;
        case 5:
-               Cvar_SetValueQuick (&scr_viewsize, bound(30, scr_viewsize.value + dir * 10, 120));
+               Cvar_SetValueQuick (&scr_2dresolution, bound(0, scr_2dresolution.value + dir * 0.2, 1));
                break;
        case 6:
-               Cvar_SetValueQuick (&r_skyquality, bound(0, r_skyquality.integer + dir, 2));
+               Cvar_SetValueQuick (&scr_viewsize, bound(30, scr_viewsize.value + dir * 10, 120));
                break;
        case 7:
-               Cvar_SetValueQuick (&r_ser, !r_ser.integer);
+               Cvar_SetValueQuick (&r_sky, !r_sky.integer);
                break;
        case 8:
                Cvar_SetValueQuick (&v_overbrightbits, bound(0, v_overbrightbits.integer + dir, 4));
@@ -1310,10 +1335,13 @@ void M_AdjustSliders (int dir)
        case 18:
                Cvar_SetValueQuick (&crosshair_size, bound(1, crosshair_size.value + dir, 5));
                break;
-       case 19: // show framerate
+       case 19: // static crosshair
+               Cvar_SetValueQuick (&crosshair_static, !crosshair_static.integer);
+               break;
+       case 20: // show framerate
                Cvar_SetValueQuick (&showfps, !showfps.integer);
                break;
-       case 20: // always run
+       case 21: // always run
                if (cl_forwardspeed.value > 200)
                {
                        Cvar_SetValueQuick (&cl_forwardspeed, 200);
@@ -1325,110 +1353,27 @@ void M_AdjustSliders (int dir)
                        Cvar_SetValueQuick (&cl_backspeed, 400);
                }
                break;
-       case 21: // lookspring
+       case 22: // lookspring
                Cvar_SetValueQuick (&lookspring, !lookspring.integer);
                break;
-       case 22: // lookstrafe
+       case 23: // lookstrafe
                Cvar_SetValueQuick (&lookstrafe, !lookstrafe.integer);
                break;
-       case 23: // mouse speed
+       case 24: // mouse speed
                Cvar_SetValueQuick (&sensitivity, bound(1, sensitivity.value + dir * 0.5, 50));
                break;
-       case 24: // mouse look
+       case 25: // mouse look
                Cvar_SetValueQuick (&freelook, !freelook.integer);
                break;
-       case 25: // invert mouse
+       case 26: // invert mouse
                Cvar_SetValueQuick (&m_pitch, -m_pitch.value);
                break;
-       case 26: // windowed mouse
+       case 27: // windowed mouse
                Cvar_SetValueQuick (&vid_mouse, !vid_mouse.integer);
                break;
        }
 }
 
-
-void M_DrawSlider (int x, int y, float range)
-{
-       int     i;
-
-       if (range < 0)
-               range = 0;
-       if (range > 1)
-               range = 1;
-       M_DrawCharacter (x-8, y, 128);
-       for (i=0 ; i<SLIDER_RANGE ; i++)
-               M_DrawCharacter (x + i*8, y, 129);
-       M_DrawCharacter (x+i*8, y, 130);
-       M_DrawCharacter (x + (SLIDER_RANGE-1)*8 * range, y, 131);
-}
-
-void M_DrawCheckbox (int x, int y, int on)
-{
-#if 0
-       if (on)
-               M_DrawCharacter (x, y, 131);
-       else
-               M_DrawCharacter (x, y, 129);
-#endif
-       if (on)
-               M_Print (x, y, "on");
-       else
-               M_Print (x, y, "off");
-}
-
-/*
-int m_2dres[] =
-{
-       320, 200,
-       320, 240,
-       400, 300,
-       512, 384,
-       640, 480,
-       800, 600,
-       1024, 768,
-       1280, 960,
-       1600, 1200,
-       2048, 1536
-};
-
-int M_Num2DResolutions(void)
-{
-       return sizeof(m_2dres) / sizeof(int[2]);
-};
-
-float M_Classify2DResolution(void)
-{
-       int i, num, *res, best, bestdist, diff[3];
-       num = M_Num2DResolutions();
-       best = -1;
-       bestdist = 1000000000;
-       for (i = 0;i < num;i++)
-       {
-               res = m_2dres + i * 2;
-               diff[0] = res[0] - vid.conwidth;
-               diff[1] = res[1] - vid.conheight;
-               diff[2] = 0;
-               dist = DotProduct(diff, diff);
-               if (bestdist > dist)
-               {
-                       bestdist = dist;
-                       best = i;
-               }
-       }
-       return i;
-}
-
-void M_Adjust2DResolution(int dir)
-{
-       int i, num;
-       i = M_Classify2DResolution() + dir;
-       num = M_Num2DResolutions() - 1;
-       i = bound(0, i, num);
-       Cvar_SetValue("v_2dwidth", m_2dres[i*2]);
-       Cvar_SetValue("v_2dheight", m_2dres[i*2+1]);
-}
-*/
-
 void M_Options_Draw (void)
 {
        float y;
@@ -1442,11 +1387,11 @@ void M_Options_Draw (void)
        M_Print(16, y, "    Customize controls");y += 8;
        M_Print(16, y, "         Go to console");y += 8;
        M_Print(16, y, "     Reset to defaults");y += 8;
-       M_ItemPrint(16, y, "         Video Options", vid_menudrawfn != NULL);y += 8;
+       M_Print(16, y, "         Video Options");y += 8;
+       M_Print(16, y, "       Effects Options");y += 8;
        M_Print(16, y, "         2D Resolution");M_DrawSlider(220, y, scr_2dresolution.value);y += 8;
        M_Print(16, y, "           Screen size");M_DrawSlider(220, y, (scr_viewsize.value - 30) /(120 - 30));y += 8;
-       M_Print(16, y, "           Sky Quality");M_DrawSlider(220, y, r_skyquality.value / 2);y += 8;
-       M_Print(16, y, "Hidden Surface Removal");M_DrawCheckbox(220, y, r_ser.integer);y += 8;
+       M_Print(16, y, "                   Sky");M_DrawCheckbox(220, y, r_sky.integer);y += 8;
        M_Print(16, y, "       Overbright Bits");M_DrawSlider(220, y, (v_overbrightbits.value) / 4);y += 8;
        M_Print(16, y, "       Texture Combine");M_DrawCheckbox(220, y, gl_combine.integer);y += 8;
        M_Print(16, y, "             Dithering");M_DrawCheckbox(220, y, gl_dither.integer);y += 8;
@@ -1458,6 +1403,7 @@ void M_Options_Draw (void)
        M_ItemPrint(16, y, "          Sound Volume", snd_initialized);M_DrawSlider(220, y, volume.value);y += 8;
        M_Print(16, y, "             Crosshair");M_DrawSlider(220, y, crosshair.value / 5);y += 8;
        M_Print(16, y, "        Crosshair Size");M_DrawSlider(220, y, (crosshair_size.value - 1) / 4);y += 8;
+       M_Print(16, y, "      Static Crosshair");M_DrawCheckbox(220, y, crosshair_static.integer);y += 8;
        M_Print(16, y, "        Show Framerate");M_DrawCheckbox(220, y, showfps.integer);y += 8;
        M_Print(16, y, "            Always Run");M_DrawCheckbox(220, y, cl_forwardspeed.value > 200);y += 8;
        M_Print(16, y, "            Lookspring");M_DrawCheckbox(220, y, lookspring.integer);y += 8;
@@ -1495,11 +1441,13 @@ void M_Options_Key (int k)
                        Cbuf_AddText ("exec default.cfg\n");
                        break;
                case 3:
-                       if (vid_menudrawfn)
-                               M_Menu_Video_f ();
+                       M_Menu_Video_f ();
+                       break;
+               case 4:
+                       M_Menu_Options_Effects_f ();
                        break;
                default:
-                       M_AdjustSliders (1);
+                       M_Menu_Options_AdjustSliders (1);
                        break;
                }
                return;
@@ -1519,11 +1467,165 @@ void M_Options_Key (int k)
                break;
 
        case K_LEFTARROW:
-               M_AdjustSliders (-1);
+               M_Menu_Options_AdjustSliders (-1);
                break;
 
        case K_RIGHTARROW:
-               M_AdjustSliders (1);
+               M_Menu_Options_AdjustSliders (1);
+               break;
+       }
+}
+
+#define        OPTIONS_EFFECTS_ITEMS   16
+
+int options_effects_cursor;
+
+void M_Menu_Options_Effects_f (void)
+{
+       key_dest = key_menu;
+       m_state = m_options_effects;
+       m_entersound = true;
+}
+
+
+extern cvar_t r_detailtextures;
+extern cvar_t cl_particles;
+extern cvar_t cl_explosions;
+extern cvar_t cl_stainmaps;
+extern cvar_t r_explosionclip;
+extern cvar_t r_dlightmap;
+extern cvar_t r_modellights;
+extern cvar_t r_coronas;
+extern cvar_t gl_flashblend;
+extern cvar_t cl_particles_bulletimpacts;
+extern cvar_t cl_particles_smoke;
+extern cvar_t cl_particles_sparks;
+extern cvar_t cl_particles_bubbles;
+extern cvar_t cl_particles_blood;
+extern cvar_t cl_particles_blood_size;
+extern cvar_t cl_particles_blood_alpha;
+
+void M_Menu_Options_Effects_AdjustSliders (int dir)
+{
+       S_LocalSound ("misc/menu3.wav");
+
+       switch (options_effects_cursor)
+       {
+       case 0:
+               Cvar_SetValueQuick (&r_modellights, bound(0, r_modellights.value + dir, 8));
+               break;
+       case 1:
+               Cvar_SetValueQuick (&r_dlightmap, !r_dlightmap.integer);
+               break;
+       case 2:
+               Cvar_SetValueQuick (&r_coronas, !r_coronas.integer);
+               break;
+       case 3:
+               Cvar_SetValueQuick (&gl_flashblend, !gl_flashblend.integer);
+               break;
+       case 4:
+               Cvar_SetValueQuick (&cl_particles, !cl_particles.integer);
+               break;
+       case 5:
+               Cvar_SetValueQuick (&cl_explosions, !cl_explosions.integer);
+               break;
+       case 6:
+               Cvar_SetValueQuick (&r_explosionclip, !r_explosionclip.integer);
+               break;
+       case 7:
+               Cvar_SetValueQuick (&cl_stainmaps, !cl_stainmaps.integer);
+               break;
+       case 8:
+               Cvar_SetValueQuick (&r_detailtextures, !r_detailtextures.integer);
+               break;
+       case 9:
+               Cvar_SetValueQuick (&cl_particles_bulletimpacts, !cl_particles_bulletimpacts.integer);
+               break;
+       case 10:
+               Cvar_SetValueQuick (&cl_particles_smoke, !cl_particles_smoke.integer);
+               break;
+       case 11:
+               Cvar_SetValueQuick (&cl_particles_sparks, !cl_particles_sparks.integer);
+               break;
+       case 12:
+               Cvar_SetValueQuick (&cl_particles_bubbles, !cl_particles_bubbles.integer);
+               break;
+       case 13:
+               Cvar_SetValueQuick (&cl_particles_blood, !cl_particles_blood.integer);
+               break;
+       case 14:
+               Cvar_SetValueQuick (&cl_particles_blood_size, bound(2, cl_particles_blood_size.value + dir * 1, 20));
+               break;
+       case 15:
+               Cvar_SetValueQuick (&cl_particles_blood_alpha, bound(0.2, cl_particles_blood_alpha.value + dir * 0.1, 1));
+               break;
+       }
+}
+
+void M_Options_Effects_Draw (void)
+{
+       float y;
+       cachepic_t      *p;
+
+       M_DrawPic(16, 4, "gfx/qplaque.lmp");
+       p = Draw_CachePic("gfx/p_option.lmp");
+       M_DrawPic((320-p->width)/2, 4, "gfx/p_option.lmp");
+
+       y = 32;
+       M_Print(16, y, "      Lights Per Model");M_DrawSlider(220, y, r_modellights.value / 8);y += 8;
+       M_Print(16, y, " Fast Dynamic Lighting");M_DrawCheckbox(220, y, !r_dlightmap.integer);y += 8;
+       M_Print(16, y, "               Coronas");M_DrawCheckbox(220, y, r_coronas.integer);y += 8;
+       M_Print(16, y, "      Use Only Coronas");M_DrawCheckbox(220, y, gl_flashblend.integer);y += 8;
+       M_Print(16, y, "             Particles");M_DrawCheckbox(220, y, cl_particles.integer);y += 8;
+       M_Print(16, y, "            Explosions");M_DrawCheckbox(220, y, cl_explosions.integer);y += 8;
+       M_Print(16, y, "    Explosion Clipping");M_DrawCheckbox(220, y, r_explosionclip.integer);y += 8;
+       M_Print(16, y, "             Stainmaps");M_DrawCheckbox(220, y, cl_stainmaps.integer);y += 8;
+       M_Print(16, y, "      Detail Texturing");M_DrawCheckbox(220, y, r_detailtextures.integer);y += 8;
+       M_Print(16, y, "        Bullet Impacts");M_DrawCheckbox(220, y, cl_particles_bulletimpacts.integer);y += 8;
+       M_Print(16, y, "                 Smoke");M_DrawCheckbox(220, y, cl_particles_smoke.integer);y += 8;
+       M_Print(16, y, "                Sparks");M_DrawCheckbox(220, y, cl_particles_sparks.integer);y += 8;
+       M_Print(16, y, "               Bubbles");M_DrawCheckbox(220, y, cl_particles_bubbles.integer);y += 8;
+       M_Print(16, y, "                 Blood");M_DrawCheckbox(220, y, cl_particles_blood.integer);y += 8;
+       M_Print(16, y, "            Blood Size");M_DrawSlider(220, y, (cl_particles_blood_size.value - 2) / 18);y += 8;
+       M_Print(16, y, "         Blood Opacity");M_DrawSlider(220, y, (cl_particles_blood_alpha.value - 0.2) / 0.8);y += 8;
+
+       // cursor
+       M_DrawCharacter(200, 32 + options_effects_cursor*8, 12+((int)(realtime*4)&1));
+}
+
+
+void M_Options_Effects_Key (int k)
+{
+       switch (k)
+       {
+       case K_ESCAPE:
+               M_Menu_Options_f ();
+               break;
+
+       case K_ENTER:
+               M_Menu_Options_Effects_AdjustSliders (1);
+               break;
+
+       case K_UPARROW:
+               S_LocalSound ("misc/menu1.wav");
+               options_effects_cursor--;
+               if (options_effects_cursor < 0)
+                       options_effects_cursor = OPTIONS_EFFECTS_ITEMS-1;
+               break;
+
+       case K_DOWNARROW:
+               S_LocalSound ("misc/menu1.wav");
+               options_effects_cursor++;
+               if (options_effects_cursor >= OPTIONS_EFFECTS_ITEMS)
+                       options_effects_cursor = 0;
+               break;
+
+       case K_LEFTARROW:
+               M_Menu_Options_Effects_AdjustSliders (-1);
+               break;
+
+       case K_RIGHTARROW:
+               M_Menu_Options_Effects_AdjustSliders (1);
                break;
        }
 }
@@ -1531,10 +1633,11 @@ void M_Options_Key (int k)
 //=============================================================================
 /* KEYS MENU */
 
-char *bindnames[][2] =
+char *quakebindnames[][2] =
 {
 {"+attack",            "attack"},
-{"impulse 10",                 "change weapon"},
+{"impulse 10",                 "next weapon"},
+{"impulse 12",                 "previous weapon"},
 {"+jump",                      "jump / swim up"},
 {"+forward",           "walk forward"},
 {"+back",                      "backpedal"},
@@ -1553,7 +1656,135 @@ char *bindnames[][2] =
 {"+movedown",          "swim down"}
 };
 
-#define        NUMCOMMANDS     (sizeof(bindnames)/sizeof(bindnames[0]))
+char *transfusionbindnames[][2] =
+{
+{"+forward",           "walk forward"},
+{"+back",                      "backpedal"},
+{"+left",                      "turn left"},
+{"+right",                     "turn right"},
+{"+moveleft",          "step left"},
+{"+moveright",                 "step right"},
+{"+jump",                      "jump / swim up"},
+{"+movedown",          "swim down"},
+{"+attack",            "attack"},
+{"+button3",           "altfire"},
+{"impulse 1",          "Pitch Fork"},
+{"impulse 2",          "Flare Gun"},
+{"impulse 3",          "Shotgun"},
+{"impulse 4",          "Machine Gun"},
+{"impulse 5",          "Incinerator"},
+{"impulse 6",          "Bombs"},
+{"impulse 7",          "Aerosol Can"},
+{"impulse 8",          "Tesla Cannon"},
+{"impulse 9",          "Life Leech"},
+{"impulse 17",         "Voodoo Doll"},
+{"impulse 11",         "previous weapon"},
+{"impulse 10",         "next weapon"},
+{"impulse 14",         "previous item"},
+{"impulse 15",         "next item"},
+{"impulse 13",         "use item"},
+{"impulse 100",                "add bot (red)"},
+{"impulse 101",                "add bot (blue)"},
+{"impulse 102",                "kick a bot"},
+{"impulse 50",         "voting menu"},
+{"impulse 141",                "identify player"},
+{"impulse 16",         "next armor type"},
+{"impulse 20",         "observer mode"}
+};
+
+int numcommands;
+char *(*bindnames)[2];
+
+/*
+typedef struct binditem_s
+{
+       char *command, *description;
+       struct binditem_s *next;
+}
+binditem_t;
+
+typedef struct bindcategory_s
+{
+       char *name;
+       binditem_t *binds;
+       struct bindcategory_s *next;
+}
+bindcategory_t;
+
+bindcategory_t *bindcategories = NULL;
+
+void M_ClearBinds (void)
+{
+       for (c = bindcategories;c;c = cnext)
+       {
+               cnext = c->next;
+               for (b = c->binds;b;b = bnext)
+               {
+                       bnext = b->next;
+                       Z_Free(b);
+               }
+               Z_Free(c);
+       }
+       bindcategories = NULL;
+}
+
+void M_AddBindToCategory(bindcategory_t *c, char *command, char *description)
+{
+       for (b = &c->binds;*b;*b = &(*b)->next);
+       *b = Z_Alloc(sizeof(binditem_t) + strlen(command) + 1 + strlen(description) + 1);
+       *b->command = (char *)((*b) + 1);
+       *b->description = *b->command + strlen(command) + 1;
+       strcpy(*b->command, command);
+       strcpy(*b->description, description);
+}
+
+void M_AddBind (char *category, char *command, char *description)
+{
+       for (c = &bindcategories;*c;c = &(*c)->next)
+       {
+               if (!strcmp(category, (*c)->name))
+               {
+                       M_AddBindToCategory(*c, command, description);
+                       return;
+               }
+       }
+       *c = Z_Alloc(sizeof(bindcategory_t));
+       M_AddBindToCategory(*c, command, description);
+}
+
+void M_DefaultBinds (void)
+{
+       M_ClearBinds();
+       M_AddBind("movement", "+jump", "jump / swim up");
+       M_AddBind("movement", "+forward", "walk forward");
+       M_AddBind("movement", "+back", "backpedal");
+       M_AddBind("movement", "+left", "turn left");
+       M_AddBind("movement", "+right", "turn right");
+       M_AddBind("movement", "+speed", "run");
+       M_AddBind("movement", "+moveleft", "step left");
+       M_AddBind("movement", "+moveright", "step right");
+       M_AddBind("movement", "+strafe", "sidestep");
+       M_AddBind("movement", "+lookup", "look up");
+       M_AddBind("movement", "+lookdown", "look down");
+       M_AddBind("movement", "centerview", "center view");
+       M_AddBind("movement", "+mlook", "mouse look");
+       M_AddBind("movement", "+klook", "keyboard look");
+       M_AddBind("movement", "+moveup", "swim up");
+       M_AddBind("movement", "+movedown", "swim down");
+       M_AddBind("weapons", "+attack", "attack");
+       M_AddBind("weapons", "impulse 10", "next weapon");
+       M_AddBind("weapons", "impulse 12", "previous weapon");
+       M_AddBind("weapons", "impulse 1", "select weapon 1 (axe)");
+       M_AddBind("weapons", "impulse 2", "select weapon 2 (shotgun)");
+       M_AddBind("weapons", "impulse 3", "select weapon 3 (super )");
+       M_AddBind("weapons", "impulse 4", "select weapon 4 (nailgun)");
+       M_AddBind("weapons", "impulse 5", "select weapon 5 (super nailgun)");
+       M_AddBind("weapons", "impulse 6", "select weapon 6 (grenade launcher)");
+       M_AddBind("weapons", "impulse 7", "select weapon 7 (rocket launcher)");
+       M_AddBind("weapons", "impulse 8", "select weapon 8 (lightning gun)");
+}
+*/
+
 
 int            keys_cursor;
 int            bind_grab;
@@ -1565,14 +1796,17 @@ void M_Menu_Keys_f (void)
        m_entersound = true;
 }
 
+#define NUMKEYS 5
 
-void M_FindKeysForCommand (char *command, int *twokeys)
+void M_FindKeysForCommand (char *command, int *keys)
 {
        int             count;
        int             j;
        char    *b;
 
-       twokeys[0] = twokeys[1] = -1;
+       for (j = 0;j < NUMKEYS;j++)
+               keys[j] = -1;
+
        count = 0;
 
        for (j=0 ; j<256 ; j++)
@@ -1582,9 +1816,8 @@ void M_FindKeysForCommand (char *command, int *twokeys)
                        continue;
                if (!strcmp (b, command) )
                {
-                       twokeys[count] = j;
-                       count++;
-                       if (count == 2)
+                       keys[count++] = j;
+                       if (count == NUMKEYS)
                                break;
                }
        }
@@ -1608,11 +1841,11 @@ void M_UnbindCommand (char *command)
 
 void M_Keys_Draw (void)
 {
-       int             i, l;
-       int             keys[2];
-       char    *name;
-       int             x, y;
+       int             i, j;
+       int             keys[NUMKEYS];
+       int             y;
        cachepic_t      *p;
+       char    keystring[1024];
 
        p = Draw_CachePic ("gfx/ttl_cstm.lmp");
        M_DrawPic ( (320-p->width)/2, 4, "gfx/ttl_cstm.lmp");
@@ -1623,44 +1856,44 @@ void M_Keys_Draw (void)
                M_Print (18, 32, "Enter to change, backspace to clear");
 
 // search for known bindings
-       for (i=0 ; i<NUMCOMMANDS ; i++)
+       for (i=0 ; i<numcommands ; i++)
        {
                y = 48 + 8*i;
 
                M_Print (16, y, bindnames[i][1]);
 
-               l = strlen (bindnames[i][0]);
-
                M_FindKeysForCommand (bindnames[i][0], keys);
 
+               // LordHavoc: redesigned to print more than 2 keys, inspired by Tomaz's MiniRacer
                if (keys[0] == -1)
-               {
-                       M_Print (140, y, "???");
-               }
+                       strcpy(keystring, "???");
                else
                {
-                       name = Key_KeynumToString (keys[0]);
-                       M_Print (140, y, name);
-                       x = strlen(name) * 8;
-                       if (keys[1] != -1)
+                       keystring[0] = 0;
+                       for (j = 0;j < NUMKEYS;j++)
                        {
-                               M_Print (140 + x + 8, y, "or");
-                               M_Print (140 + x + 32, y, Key_KeynumToString (keys[1]));
+                               if (keys[j] != -1)
+                               {
+                                       if (j > 0)
+                                               strcat(keystring, " or ");
+                                       strcat(keystring, Key_KeynumToString (keys[j]));
+                               }
                        }
                }
+               M_Print (150, y, keystring);
        }
 
        if (bind_grab)
-               M_DrawCharacter (130, 48 + keys_cursor*8, '=');
+               M_DrawCharacter (140, 48 + keys_cursor*8, '=');
        else
-               M_DrawCharacter (130, 48 + keys_cursor*8, 12+((int)(realtime*4)&1));
+               M_DrawCharacter (140, 48 + keys_cursor*8, 12+((int)(realtime*4)&1));
 }
 
 
 void M_Keys_Key (int k)
 {
        char    cmd[80];
-       int             keys[2];
+       int             keys[NUMKEYS];
 
        if (bind_grab)
        {       // defining a key
@@ -1669,7 +1902,7 @@ void M_Keys_Key (int k)
                {
                        bind_grab = false;
                }
-               else if (k != '`')
+               else //if (k != '`')
                {
                        sprintf (cmd, "bind \"%s\" \"%s\"\n", Key_KeynumToString (k), bindnames[keys_cursor][0]);
                        Cbuf_InsertText (cmd);
@@ -1690,21 +1923,21 @@ void M_Keys_Key (int k)
                S_LocalSound ("misc/menu1.wav");
                keys_cursor--;
                if (keys_cursor < 0)
-                       keys_cursor = NUMCOMMANDS-1;
+                       keys_cursor = numcommands-1;
                break;
 
        case K_DOWNARROW:
        case K_RIGHTARROW:
                S_LocalSound ("misc/menu1.wav");
                keys_cursor++;
-               if (keys_cursor >= NUMCOMMANDS)
+               if (keys_cursor >= numcommands)
                        keys_cursor = 0;
                break;
 
        case K_ENTER:           // go into bind mode
                M_FindKeysForCommand (bindnames[keys_cursor][0], keys);
                S_LocalSound ("misc/menu2.wav");
-               if (keys[1] != -1)
+               if (keys[NUMKEYS - 1] != -1)
                        M_UnbindCommand (bindnames[keys_cursor][0]);
                bind_grab = true;
                break;
@@ -1720,23 +1953,165 @@ void M_Keys_Key (int k)
 //=============================================================================
 /* VIDEO MENU */
 
+#define VIDEO_ITEMS 5
+
+int video_cursor = 0;
+int video_cursor_table[] = {56, 68, 80, 92, 116};
+unsigned short video_resolutions[][2] = {{512,384}, {640,480}, {800,600}, {1024,768}, {1280,960}};
+int video_resolution;
+
+extern int current_vid_fullscreen;
+extern int current_vid_width;
+extern int current_vid_height;
+extern int current_vid_bitsperpixel;
+extern int current_vid_stencil;
+
+
 void M_Menu_Video_f (void)
 {
        key_dest = key_menu;
        m_state = m_video;
        m_entersound = true;
+
+       // Look for the current resolution
+       for (video_resolution = 0; video_resolution < (int) (sizeof (video_resolutions) / sizeof (video_resolutions[0])); video_resolution++)
+       {
+               if (video_resolutions[video_resolution][0] == current_vid_width &&
+                       video_resolutions[video_resolution][1] == current_vid_height)
+                       break;
+       }
+
+       // Default to 800x600 if we didn't find it
+       if (video_resolution == sizeof (video_resolutions) / sizeof (video_resolutions[0]))
+       {
+               video_resolution = 2;
+               Cvar_SetValueQuick (&vid_width, video_resolutions[video_resolution][0]);
+               Cvar_SetValueQuick (&vid_height, video_resolutions[video_resolution][1]);
+       }
 }
 
 
 void M_Video_Draw (void)
 {
-       (*vid_menudrawfn) ();
+       cachepic_t      *p;
+       const char* string;
+
+       M_DrawPic(16, 4, "gfx/qplaque.lmp");
+       p = Draw_CachePic("gfx/vidmodes.lmp");
+       M_DrawPic((320-p->width)/2, 4, "gfx/vidmodes.lmp");
+
+       // Resolution
+       M_Print(16, video_cursor_table[0], "            Resolution");
+       string = va("%dx%d", video_resolutions[video_resolution][0], video_resolutions[video_resolution][1]);
+       M_Print (220, video_cursor_table[0], string);
+
+       // Bits per pixel
+       M_Print(16, video_cursor_table[1], "        Bits per pixel");
+       M_Print (220, video_cursor_table[1], (vid_bitsperpixel.integer == 32) ? "32" : "16");
+
+       // Fullscreen
+       M_Print(16, video_cursor_table[2], "            Fullscreen");
+       M_DrawCheckbox(220, video_cursor_table[2], vid_fullscreen.integer);
+
+       // Stencil
+       M_Print(16, video_cursor_table[3], "               Stencil");
+       M_DrawCheckbox(220, video_cursor_table[3], vid_stencil.integer);
+
+       // "Apply" button
+       M_Print(220, video_cursor_table[4], "Apply");
+
+       // Cursor
+       M_DrawCharacter(200, video_cursor_table[video_cursor], 12+((int)(realtime*4)&1));
+}
+
+
+void M_Menu_Video_AdjustSliders (int dir)
+{
+       S_LocalSound ("misc/menu3.wav");
+
+       switch (video_cursor)
+       {
+               // Resolution
+               case 0:
+               {
+                       int new_resolution = video_resolution + dir;
+                       if (new_resolution < 0)
+                               video_resolution = sizeof (video_resolutions) / sizeof (video_resolutions[0]) - 1;
+                       else if (new_resolution > (int) (sizeof (video_resolutions) / sizeof (video_resolutions[0]) - 1))
+                               video_resolution = 0;
+                       else
+                               video_resolution = new_resolution;
+
+                       Cvar_SetValueQuick (&vid_width, video_resolutions[video_resolution][0]);
+                       Cvar_SetValueQuick (&vid_height, video_resolutions[video_resolution][1]);
+                       break;
+               }
+
+               // Bits per pixel
+               case 1:
+                       Cvar_SetValueQuick (&vid_bitsperpixel, (vid_bitsperpixel.integer == 32) ? 16 : 32);
+                       break;
+               case 2:
+                       Cvar_SetValueQuick (&vid_fullscreen, !vid_fullscreen.integer);
+                       break;
+               case 3:
+                       Cvar_SetValueQuick (&vid_stencil, !vid_stencil.integer);
+                       break;
+       }
 }
 
 
 void M_Video_Key (int key)
 {
-       (*vid_menukeyfn) (key);
+       switch (key)
+       {
+               case K_ESCAPE:
+                       // vid_shared.c has a copy of the current video config. We restore it
+                       Cvar_SetValueQuick(&vid_fullscreen, current_vid_fullscreen);
+                       Cvar_SetValueQuick(&vid_width, current_vid_width);
+                       Cvar_SetValueQuick(&vid_height, current_vid_height);
+                       Cvar_SetValueQuick(&vid_bitsperpixel, current_vid_bitsperpixel);
+                       Cvar_SetValueQuick(&vid_stencil, current_vid_stencil);
+
+                       S_LocalSound ("misc/menu1.wav");
+                       M_Menu_Options_f ();
+                       break;
+
+               case K_ENTER:
+                       m_entersound = true;
+                       switch (video_cursor)
+                       {
+                               case 4:
+                                       Cbuf_AddText ("vid_restart\n");
+                                       M_Menu_Options_f ();
+                                       break;
+                               default:
+                                       M_Menu_Video_AdjustSliders (1);
+                       }
+                       break;
+
+               case K_UPARROW:
+                       S_LocalSound ("misc/menu1.wav");
+                       video_cursor--;
+                       if (video_cursor < 0)
+                               video_cursor = VIDEO_ITEMS-1;
+                       break;
+
+               case K_DOWNARROW:
+                       S_LocalSound ("misc/menu1.wav");
+                       video_cursor++;
+                       if (video_cursor >= VIDEO_ITEMS)
+                               video_cursor = 0;
+                       break;
+
+               case K_LEFTARROW:
+                       M_Menu_Video_AdjustSliders (-1);
+                       break;
+
+               case K_RIGHTARROW:
+                       M_Menu_Video_AdjustSliders (1);
+                       break;
+       }
 }
 
 //=============================================================================
@@ -1794,8 +2169,7 @@ int               msgNumber;
 int            m_quit_prevstate;
 qboolean       wasInMenus;
 
-//#ifndef      _WIN32
-char *quitMessage [] = 
+char *quitMessage [] =
 {
 /* .........1.........2.... */
 /*
@@ -1818,17 +2192,17 @@ char *quitMessage [] =
   "   for trying to quit!  ",
   "     Press Y to get     ",
   "      smacked out.      ",
+
   " Press Y to quit like a ",
   "   big loser in life.   ",
   "  Press N to stay proud ",
   "    and successful!     ",
+
   "   If you press Y to    ",
   "  quit, I will summon   ",
   "  Satan all over your   ",
   "      hard drive!       ",
+
   "  Um, Asmodeus dislikes ",
   " his children trying to ",
   " quit. Press Y to return",
@@ -1861,7 +2235,6 @@ char *quitMessage [] =
   "      constructive?     ",
   "                        ",
 };
-//#endif
 
 void M_Menu_Quit_f (void)
 {
@@ -1897,7 +2270,6 @@ void M_Quit_Key (int key)
 
        case 'Y':
        case 'y':
-               key_dest = key_console;
                Host_Quit_f ();
                break;
 
@@ -1910,46 +2282,19 @@ void M_Quit_Key (int key)
 
 void M_Quit_Draw (void)
 {
-/*
-#ifdef _WIN32
-       M_DrawTextBox (0, 0, 38, 23);
-       M_PrintWhite (16, 12,  "  Quake version 1.09 by id Software\n\n");
-       M_PrintWhite (16, 28,  "Programming        Art \n");
-       M_Print (16, 36,  " John Carmack       Adrian Carmack\n");
-       M_Print (16, 44,  " Michael Abrash     Kevin Cloud\n");
-       M_Print (16, 52,  " John Cash          Paul Steed\n");
-       M_Print (16, 60,  " Dave 'Zoid' Kirsch\n");
-       M_PrintWhite (16, 68,  "Design             Biz\n");
-       M_Print (16, 76,  " John Romero        Jay Wilbur\n");
-       M_Print (16, 84,  " Sandy Petersen     Mike Wilson\n");
-       M_Print (16, 92,  " American McGee     Donna Jackson\n");
-       M_Print (16, 100,  " Tim Willits        Todd Hollenshead\n");
-       M_PrintWhite (16, 108, "Support            Projects\n");
-       M_Print (16, 116, " Barrett Alexander  Shawn Green\n");
-       M_PrintWhite (16, 124, "Sound Effects\n");
-       M_Print (16, 132, " Trent Reznor and Nine Inch Nails\n\n");
-       M_PrintWhite (16, 140, "Quake is a trademark of Id Software,\n");
-       M_PrintWhite (16, 148, "inc., (c)1996 Id Software, inc. All\n");
-       M_PrintWhite (16, 156, "rights reserved. NIN logo is a\n");
-       M_PrintWhite (16, 164, "registered trademark licensed to\n");
-       M_PrintWhite (16, 172, "Nothing Interactive, Inc. All rights\n");
-       M_PrintWhite (16, 180, "reserved. Press y to exit\n");
-#else
-*/
        M_DrawTextBox (56, 76, 24, 4);
        M_Print (64, 84,  quitMessage[msgNumber*4+0]);
        M_Print (64, 92,  quitMessage[msgNumber*4+1]);
        M_Print (64, 100, quitMessage[msgNumber*4+2]);
        M_Print (64, 108, quitMessage[msgNumber*4+3]);
-//#endif
 }
 
 //=============================================================================
 /* LAN CONFIG MENU */
 
 int            lanConfig_cursor = -1;
-int            lanConfig_cursor_table [] = {72, 92, 124};
-#define NUM_LANCONFIG_CMDS     3
+int            lanConfig_cursor_table [] = {72, 92, 112, 144};
+#define NUM_LANCONFIG_CMDS     4
 
 int    lanConfig_port;
 char   lanConfig_portname[6];
@@ -2013,9 +2358,10 @@ void M_LanConfig_Draw (void)
        if (JoiningGame)
        {
                M_Print (basex, lanConfig_cursor_table[1], "Search for local games...");
-               M_Print (basex, 108, "Join game at:");
-               M_DrawTextBox (basex+8, lanConfig_cursor_table[2]-8, 22, 1);
-               M_Print (basex+16, lanConfig_cursor_table[2], lanConfig_joinname);
+               M_Print (basex, lanConfig_cursor_table[2], "Search for internet games...");
+               M_Print (basex, 128, "Join game at:");
+               M_DrawTextBox (basex+8, lanConfig_cursor_table[3]-8, 22, 1);
+               M_Print (basex+16, lanConfig_cursor_table[3], lanConfig_joinname);
        }
        else
        {
@@ -2028,11 +2374,11 @@ void M_LanConfig_Draw (void)
        if (lanConfig_cursor == 0)
                M_DrawCharacter (basex+9*8 + 8*strlen(lanConfig_portname), lanConfig_cursor_table [0], 10+((int)(realtime*4)&1));
 
-       if (lanConfig_cursor == 2)
-               M_DrawCharacter (basex+16 + 8*strlen(lanConfig_joinname), lanConfig_cursor_table [2], 10+((int)(realtime*4)&1));
+       if (lanConfig_cursor == 3)
+               M_DrawCharacter (basex+16 + 8*strlen(lanConfig_joinname), lanConfig_cursor_table [3], 10+((int)(realtime*4)&1));
 
        if (*m_return_reason)
-               M_PrintWhite (basex, 148, m_return_reason);
+               M_PrintWhite (basex, 168, m_return_reason);
 }
 
 
@@ -2068,18 +2414,21 @@ void M_LanConfig_Key (int key)
 
                M_ConfigureNetSubsystem ();
 
-               if (lanConfig_cursor == 1)
+               if (lanConfig_cursor == 1 || lanConfig_cursor == 2)
                {
                        if (StartingGame)
                        {
                                M_Menu_GameOptions_f ();
                                break;
                        }
-                       M_Menu_Search_f();
+                       if (lanConfig_cursor == 1)
+                               M_Menu_Search_f();
+                       else
+                               M_Menu_InetSearch_f();
                        break;
                }
 
-               if (lanConfig_cursor == 2)
+               if (lanConfig_cursor == 3)
                {
                        m_return_state = m_state;
                        m_return_onerror = true;
@@ -2098,7 +2447,7 @@ void M_LanConfig_Key (int key)
                                lanConfig_portname[strlen(lanConfig_portname)-1] = 0;
                }
 
-               if (lanConfig_cursor == 2)
+               if (lanConfig_cursor == 3)
                {
                        if (strlen(lanConfig_joinname))
                                lanConfig_joinname[strlen(lanConfig_joinname)-1] = 0;
@@ -2109,7 +2458,7 @@ void M_LanConfig_Key (int key)
                if (key < 32 || key > 127)
                        break;
 
-               if (lanConfig_cursor == 2)
+               if (lanConfig_cursor == 3)
                {
                        l = strlen(lanConfig_joinname);
                        if (l < 21)
@@ -2132,7 +2481,7 @@ void M_LanConfig_Key (int key)
                }
        }
 
-       if (StartingGame && lanConfig_cursor == 2)
+       if (StartingGame && lanConfig_cursor == 3)
        {
                if (key == K_UPARROW)
                        lanConfig_cursor = 1;
@@ -2335,17 +2684,63 @@ episode_t       nehahraepisodes[] =
        {"Dimension of the Lost", 17, 2}
 };
 
-// these are placeholders for bloodbath developers to fill in more correctly
-level_t                bloodbathlevels[] =
-{
-       {"start",       "Welcome to BloodBath"},
-       {"bb1m1",       "Level 1"}
+// Map list for Transfusion
+level_t                transfusionlevels[] =
+{
+       {"bb1",                 "The Stronghold"},
+       {"bb2",                 "Winter Wonderland"},
+       {"bb3",                 "Bodies"},
+       {"bb4",                 "The Tower"},
+       {"bb5",                 "Click!"},
+       {"bb6",                 "Twin Fortress"},
+       {"bb7",                 "Midgard"},
+       {"bb8",                 "Fun With Heads"},
+
+       {"e1m1",                "Cradle to Grave"},
+       {"e1m2",                "Wrong Side of the Tracks"},
+       {"e1m7",                "Altar of Stone"},
+       {"e2m8",                "The Lair of Shial"},
+       {"e3m7",                "The Pit of Cerberus"},
+       {"e4m8",                "The Hall of the Epiphany"},
+
+       {"dm1",                 "Monolith Building 11"},
+       {"dm2",                 "Power!"},
+       {"dm3",                 "Area 15"},
+       {"e6m1",                "Welcome to Your Life"},
+       {"e6m8",                "Beauty and the Beast"},
+
+       {"cpbb01",              "Crypt of Despair"},
+       {"cpbb03",              "Unholy Cathedral"},
+
+       {"b2a15",               "Area 15 (B2)"},
+       {"barena",              "Blood Arena"},
+       {"bkeep",               "Blood Keep"},
+       {"bstar",               "Brown Star"},
+       {"crypt",               "The Crypt"},
+
+       {"bb3_2k1",             "Bodies Infusion"},
+       {"dcamp",               "DeathCamp"},
+       {"highnoon",    "HighNoon"},
+       {"qbb1",                "The Confluence"},
+       {"qbb2",                "KathartiK"},
+       {"qbb3",                "Caleb's Woodland Retreat"},
+
+       {"dranzbb6",    "Black Coffee"},
+       {"fragm",               "Frag'M"},
+       {"maim",                "Maim"},
+       {"qe1m7",               "The House of Chthon"},
+       {"simple",              "Dead Simple"}
 };
 
-episode_t      bloodbathepisodes[] =
+episode_t      transfusionepisodes[] =
 {
-       {"Welcome to BloodBath", 0, 1},
-       {"BloodBath Episode 1", 1, 1}
+       {"Blood", 0, 8},
+       {"Blood Single Player", 8, 6},
+       {"Plasma Pack", 14, 5},
+       {"Cryptic Passage", 19, 2},
+       {"Blood 2", 21, 5},
+       {"Transfusion", 26, 6},
+       {"Conversions", 32, 5}
 };
 
 gamelevels_t sharewarequakegame = {"Shareware Quake", quakelevels, quakeepisodes, 2};
@@ -2353,7 +2748,7 @@ gamelevels_t registeredquakegame = {"Quake", quakelevels, quakeepisodes, 7};
 gamelevels_t hipnoticgame = {"Scourge of Armagon", hipnoticlevels, hipnoticepisodes, 6};
 gamelevels_t roguegame = {"Dissolution of Eternity", roguelevels, rogueepisodes, 4};
 gamelevels_t nehahragame = {"Nehahra", nehahralevels, nehahraepisodes, 4};
-gamelevels_t bloodbathgame = {"BloodBath", bloodbathlevels, bloodbathepisodes, 2};
+gamelevels_t transfusiongame = {"Transfusion", transfusionlevels, transfusionepisodes, 7};
 
 typedef struct
 {
@@ -2369,9 +2764,7 @@ gameinfo_t gamelist[] =
        {GAME_HIPNOTIC, &hipnoticgame, &hipnoticgame},
        {GAME_ROGUE, &roguegame, &roguegame},
        {GAME_NEHAHRA, &nehahragame, &nehahragame},
-       {GAME_FIENDARENA, &sharewarequakegame, &registeredquakegame},
-       {GAME_ZYMOTIC, &sharewarequakegame, &registeredquakegame},
-       {GAME_BLOODBATH, &bloodbathgame, &bloodbathgame},
+       {GAME_TRANSFUSION, &transfusiongame, &transfusiongame},
        {-1, &sharewarequakegame, &registeredquakegame} // final fallback
 };
 
@@ -2399,7 +2792,7 @@ void M_Menu_GameOptions_f (void)
        if (maxplayers == 0)
                maxplayers = svs.maxclients;
        if (maxplayers < 2)
-               maxplayers = svs.maxclientslimit;
+               maxplayers = MAX_SCOREBOARD;
 }
 
 
@@ -2424,12 +2817,24 @@ void M_GameOptions_Draw (void)
        M_Print (160, 56, va("%i", maxplayers) );
 
        M_Print (0, 64, "        Game Type");
-       if (!coop.integer && !deathmatch.integer)
-               Cvar_SetValue("deathmatch", 1);
-       if (coop.integer)
-               M_Print (160, 64, "Cooperative");
+       if (gamemode == GAME_TRANSFUSION)
+       {
+               if (!deathmatch.integer)
+                       Cvar_SetValue("deathmatch", 1);
+               if (deathmatch.integer == 2)
+                       M_Print (160, 64, "Capture the Flag");
+               else
+                       M_Print (160, 64, "Blood Bath");
+       }
        else
-               M_Print (160, 64, "Deathmatch");
+       {
+               if (!coop.integer && !deathmatch.integer)
+                       Cvar_SetValue("deathmatch", 1);
+               if (coop.integer)
+                       M_Print (160, 64, "Cooperative");
+               else
+                       M_Print (160, 64, "Deathmatch");
+       }
 
        M_Print (0, 72, "        Teamplay");
        if (gamemode == GAME_ROGUE)
@@ -2448,6 +2853,18 @@ void M_GameOptions_Draw (void)
                }
                M_Print (160, 72, msg);
        }
+       else if (gamemode == GAME_TRANSFUSION)
+       {
+               char *msg;
+
+               switch (teamplay.integer)
+               {
+                       case 0: msg = "Off"; break;
+                       case 2: msg = "Friendly Fire"; break;
+                       default: msg = "No Friendly Fire"; break;
+               }
+               M_Print (160, 72, msg);
+       }
        else
        {
                char *msg;
@@ -2505,13 +2922,7 @@ void M_GameOptions_Draw (void)
                        M_Print (x, 146, " More than 64 players?? ");
                        M_Print (x, 154, "  First, question your  ");
                        M_Print (x, 162, "   sanity, then email   ");
-                       M_Print (x, 170, " havoc@gamevisions.com  ");
-                       /*
-                       M_Print (x, 146, "  More than 4 players   ");
-                       M_Print (x, 154, " requires using command ");
-                       M_Print (x, 162, "line parameters; please ");
-                       M_Print (x, 170, "   see techinfo.txt.    ");
-                       */
+                       M_Print (x, 170, " havoc@telefragged.com  ");
                }
                else
                {
@@ -2530,9 +2941,9 @@ void M_NetStart_Change (int dir)
        {
        case 1:
                maxplayers += dir;
-               if (maxplayers > svs.maxclientslimit)
+               if (maxplayers > MAX_SCOREBOARD)
                {
-                       maxplayers = svs.maxclientslimit;
+                       maxplayers = MAX_SCOREBOARD;
                        m_serverInfoMessage = true;
                        m_serverInfoMessageTime = realtime;
                }
@@ -2541,15 +2952,25 @@ void M_NetStart_Change (int dir)
                break;
 
        case 2:
-               if (deathmatch.integer) // changing from deathmatch to coop
+               if (gamemode == GAME_TRANSFUSION)
                {
-                       Cvar_SetValueQuick (&coop, 1);
-                       Cvar_SetValueQuick (&deathmatch, 0);
+                       if (deathmatch.integer == 2) // changing from CTF to BloodBath
+                               Cvar_SetValueQuick (&deathmatch, 0);
+                       else // changing from BloodBath to CTF
+                               Cvar_SetValueQuick (&deathmatch, 2);
                }
-               else // changing from coop to deathmatch
+               else
                {
-                       Cvar_SetValueQuick (&coop, 0);
-                       Cvar_SetValueQuick (&deathmatch, 1);
+                       if (deathmatch.integer) // changing from deathmatch to coop
+                       {
+                               Cvar_SetValueQuick (&coop, 1);
+                               Cvar_SetValueQuick (&deathmatch, 0);
+                       }
+                       else // changing from coop to deathmatch
+                       {
+                               Cvar_SetValueQuick (&coop, 0);
+                               Cvar_SetValueQuick (&deathmatch, 1);
+                       }
                }
                break;
 
@@ -2694,6 +3115,7 @@ void M_Menu_Search_f (void)
 
 void M_Search_Draw (void)
 {
+       const char* string;
        cachepic_t      *p;
        int x;
 
@@ -2721,7 +3143,11 @@ void M_Search_Draw (void)
                return;
        }
 
-       M_PrintWhite ((320/2) - ((22*8)/2), 64, "No Quake servers found");
+       if (gamemode == GAME_TRANSFUSION)
+               string = "No Transfusion servers found";
+       else
+               string = "No Quake servers found";
+       M_PrintWhite ((320/2) - ((22*8)/2), 64, string);
        if ((realtime - searchCompleteTime) < 3.0)
                return;
 
@@ -2733,6 +3159,32 @@ void M_Search_Key (int key)
 {
 }
 
+//=============================================================================
+/* INTERNET SEARCH MENU */
+
+void M_Menu_InetSearch_f (void)
+{
+       key_dest = key_menu;
+       m_state = m_search;
+       m_entersound = false;
+       slistSilent = true;
+       slistLocal = false;
+       searchComplete = false;
+       NET_InetSlist_f();
+
+}
+
+
+void M_InetSearch_Draw (void)
+{
+       M_Search_Draw ();  // it's the same one, so why bother?
+}
+
+
+void M_InetSearch_Key (int key)
+{
+}
+
 //=============================================================================
 /* SLIST MENU */
 
@@ -2788,7 +3240,7 @@ void M_ServerList_Draw (void)
        M_DrawCharacter (0, 32 + slist_cursor*8, 12+((int)(realtime*4)&1));
 
        if (*m_return_reason)
-               M_PrintWhite (16, 148, m_return_reason);
+               M_PrintWhite (16, 168, m_return_reason);
 }
 
 
@@ -2851,11 +3303,23 @@ void M_Init (void)
        Cmd_AddCommand ("menu_multiplayer", M_Menu_MultiPlayer_f);
        Cmd_AddCommand ("menu_setup", M_Menu_Setup_f);
        Cmd_AddCommand ("menu_options", M_Menu_Options_f);
+       Cmd_AddCommand ("menu_options_effects", M_Menu_Options_Effects_f);
        Cmd_AddCommand ("menu_keys", M_Menu_Keys_f);
        Cmd_AddCommand ("menu_video", M_Menu_Video_f);
        Cmd_AddCommand ("help", M_Menu_Help_f);
        Cmd_AddCommand ("menu_quit", M_Menu_Quit_f);
 
+       if (gamemode == GAME_TRANSFUSION)
+       {
+               numcommands = sizeof(transfusionbindnames) / sizeof(transfusionbindnames[0]);
+               bindnames = transfusionbindnames;
+       }
+       else
+       {
+               numcommands = sizeof(quakebindnames) / sizeof(quakebindnames[0]);
+               bindnames = quakebindnames;
+       }
+
        if (gamemode == GAME_NEHAHRA)
        {
                if (COM_FileExists("maps/neh1m4.bsp"))
@@ -2935,6 +3399,10 @@ void M_Draw (void)
                M_Options_Draw ();
                break;
 
+       case m_options_effects:
+               M_Options_Effects_Draw ();
+               break;
+
        case m_keys:
                M_Keys_Draw ();
                break;
@@ -3021,6 +3489,10 @@ void M_Keydown (int key)
                M_Options_Key (key);
                return;
 
+       case m_options_effects:
+               M_Options_Effects_Key (key);
+               return;
+
        case m_keys:
                M_Keys_Key (key);
                return;
@@ -3065,3 +3537,4 @@ void M_ConfigureNetSubsystem(void)
        if (IPXConfig || TCPIPConfig)
                net_hostport = lanConfig_port;
 }
+