/* =========================================================================== Copyright (C) 1997-2006 Id Software, Inc. This file is part of Quake 2 Tools source code. Quake 2 Tools source code is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. Quake 2 Tools source code 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. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Quake 2 Tools source code; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA =========================================================================== */ #include "qe3.h" QEGlobals_t g_qeglobals; void QE_CheckOpenGLForErrors(void) { int i; while ( ( i = glGetError() ) != GL_NO_ERROR ) { char buffer[100]; sprintf( buffer, "OpenGL Error: %s", gluErrorString( i ) ); MessageBox( g_qeglobals.d_hwndMain, buffer , "QuakeEd Error", MB_OK | MB_ICONEXCLAMATION ); exit( 1 ); } } char *ExpandReletivePath (char *p) { static char temp[1024]; char *base; if (!p || !p[0]) return NULL; if (p[0] == '/' || p[0] == '\\') return p; base = ValueForKey(g_qeglobals.d_project_entity, "basepath"); sprintf (temp, "%s/%s", base, p); return temp; } void *qmalloc (int size) { void *b; b = malloc(size); memset (b, 0, size); return b; } char *copystring (char *s) { char *b; b = malloc(strlen(s)+1); strcpy (b,s); return b; } /* =============== QE_CheckAutoSave If five minutes have passed since making a change and the map hasn't been saved, save it out. =============== */ void QE_CheckAutoSave( void ) { static clock_t s_start; clock_t now; now = clock(); if ( modified != 1 || !s_start) { s_start = now; return; } if ( now - s_start > ( CLOCKS_PER_SEC * 60 * QE_AUTOSAVE_INTERVAL ) ) { Sys_Printf ("Autosaving...\n"); Sys_Status ("Autosaving...", 0 ); Map_SaveFile (ValueForKey(g_qeglobals.d_project_entity, "autosave"), false); Sys_Status ("Autosaving...Saved.", 0 ); modified = 2; s_start = now; } } /* =========== QE_LoadProject =========== */ qboolean QE_LoadProject (char *projectfile) { char *data; Sys_Printf ("QE_LoadProject (%s)\n", projectfile); if ( LoadFileNoCrash (projectfile, (void *)&data) == -1) return false; StartTokenParsing (data); g_qeglobals.d_project_entity = Entity_Parse (true); if (!g_qeglobals.d_project_entity) Error ("Couldn't parse %s", projectfile); free (data); Eclass_InitForSourceDirectory (ValueForKey (g_qeglobals.d_project_entity, "entitypath")); FillClassList (); // list in entity window Map_New (); FillTextureMenu (); FillBSPMenu (); return true; } /* =========== QE_KeyDown =========== */ #define SPEED_MOVE 32 #define SPEED_TURN 22.5 qboolean QE_KeyDown (int key) { switch (key) { case 'K': PostMessage( g_qeglobals.d_hwndMain, WM_COMMAND, ID_MISC_SELECTENTITYCOLOR, 0 ); break; case VK_UP: VectorMA (camera.origin, SPEED_MOVE, camera.forward, camera.origin); Sys_UpdateWindows (W_CAMERA|W_XY_OVERLAY); break; case VK_DOWN: VectorMA (camera.origin, -SPEED_MOVE, camera.forward, camera.origin); Sys_UpdateWindows (W_CAMERA|W_XY_OVERLAY); break; case VK_LEFT: camera.angles[1] += SPEED_TURN; Sys_UpdateWindows (W_CAMERA|W_XY_OVERLAY); break; case VK_RIGHT: camera.angles[1] -= SPEED_TURN; Sys_UpdateWindows (W_CAMERA|W_XY_OVERLAY); break; case 'D': camera.origin[2] += SPEED_MOVE; Sys_UpdateWindows (W_CAMERA|W_XY_OVERLAY|W_Z_OVERLAY); break; case 'C': camera.origin[2] -= SPEED_MOVE; Sys_UpdateWindows (W_CAMERA|W_XY_OVERLAY|W_Z_OVERLAY); break; case 'A': camera.angles[0] += SPEED_TURN; if (camera.angles[0] > 85) camera.angles[0] = 85; Sys_UpdateWindows (W_CAMERA|W_XY_OVERLAY); break; case 'Z': camera.angles[0] -= SPEED_TURN; if (camera.angles[0] < -85) camera.angles[0] = -85; Sys_UpdateWindows (W_CAMERA|W_XY_OVERLAY); break; case VK_COMMA: VectorMA (camera.origin, -SPEED_MOVE, camera.right, camera.origin); Sys_UpdateWindows (W_CAMERA|W_XY_OVERLAY); break; case VK_PERIOD: VectorMA (camera.origin, SPEED_MOVE, camera.right, camera.origin); Sys_UpdateWindows (W_CAMERA|W_XY_OVERLAY); break; case '0': g_qeglobals.d_showgrid = !g_qeglobals.d_showgrid; PostMessage( g_qeglobals.d_hwndXY, WM_PAINT, 0, 0 ); break; case '1': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_GRID_1, 0); break; case '2': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_GRID_2, 0); break; case '3': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_GRID_4, 0); break; case '4': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_GRID_8, 0); break; case '5': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_GRID_16, 0); break; case '6': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_GRID_32, 0); break; case '7': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_GRID_64, 0); break; case 'E': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_SELECTION_DRAGEDGES, 0); break; case 'V': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_SELECTION_DRAGVERTECIES, 0); break; case 'N': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_VIEW_ENTITY, 0); break; case 'O': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_VIEW_CONSOLE, 0); break; case 'T': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_VIEW_TEXTURE, 0); break; case 'S': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_TEXTURES_INSPECTOR, 0); break; case ' ': PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_SELECTION_CLONE, 0); break; case VK_BACK: PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_SELECTION_DELETE, 0); break; case VK_ESCAPE: PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_SELECTION_DESELECT, 0); break; case VK_END: PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_VIEW_CENTER, 0); break; case VK_DELETE: PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_VIEW_ZOOMIN, 0); break; case VK_INSERT: PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_VIEW_ZOOMOUT, 0); break; case VK_NEXT: PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_VIEW_DOWNFLOOR, 0); break; case VK_PRIOR: PostMessage (g_qeglobals.d_hwndMain, WM_COMMAND, ID_VIEW_UPFLOOR, 0); break; default: return false; } return true; } /* =============== ConnectEntities Sets target / targetname on the two entities selected from the first selected to the secon =============== */ void ConnectEntities (void) { entity_t *e1, *e2, *e; char *target, *tn; int maxtarg, targetnum; char newtarg[32]; if (g_qeglobals.d_select_count != 2) { Sys_Status ("Must have two brushes selected.", 0); Sys_Beep (); return; } e1 = g_qeglobals.d_select_order[0]->owner; e2 = g_qeglobals.d_select_order[1]->owner; if (e1 == world_entity || e2 == world_entity) { Sys_Status ("Can't connect to the world.", 0); Sys_Beep (); return; } if (e1 == e2) { Sys_Status ("Brushes are from same entity.", 0); Sys_Beep (); return; } target = ValueForKey (e1, "target"); if (target && target[0]) strcpy (newtarg, target); else { target = ValueForKey (e2, "targetname"); if (target && target[0]) strcpy (newtarg, target); else { // make a unique target value maxtarg = 0; for (e=entities.next ; e != &entities ; e=e->next) { tn = ValueForKey (e, "targetname"); if (tn && tn[0]) { targetnum = atoi(tn+1); if (targetnum > maxtarg) maxtarg = targetnum; } } sprintf (newtarg, "t%i", maxtarg+1); } } SetKeyValue (e1, "target", newtarg); SetKeyValue (e2, "targetname", newtarg); Sys_UpdateWindows (W_XY | W_CAMERA); Select_Deselect(); Select_Brush (g_qeglobals.d_select_order[1]); } qboolean QE_SingleBrush (void) { if ( (selected_brushes.next == &selected_brushes) || (selected_brushes.next->next != &selected_brushes) ) { Sys_Printf ("Error: you must have a single brush selected\n"); return false; } if (selected_brushes.next->owner->eclass->fixedsize) { Sys_Printf ("Error: you cannot manipulate fixed size entities\n"); return false; } return true; } void QE_Init (void) { /* ** initialize variables */ g_qeglobals.d_gridsize = 8; g_qeglobals.d_showgrid = true; /* ** other stuff */ Texture_Init (); Cam_Init (); XY_Init (); Z_Init (); } void QE_ConvertDOSToUnixName( char *dst, const char *src ) { while ( *src ) { if ( *src == '\\' ) *dst = '/'; else *dst = *src; dst++; src++; } *dst = 0; } int g_numbrushes, g_numentities; void QE_CountBrushesAndUpdateStatusBar( void ) { static int s_lastbrushcount, s_lastentitycount; static qboolean s_didonce; entity_t *e; brush_t *b, *next; g_numbrushes = 0; g_numentities = 0; if ( active_brushes.next != NULL ) { for ( b = active_brushes.next ; b != NULL && b != &active_brushes ; b=next) { next = b->next; if (b->brush_faces ) { if ( !b->owner->eclass->fixedsize) g_numbrushes++; else g_numentities++; } } } if ( entities.next != NULL ) { for ( e = entities.next ; e != &entities && g_numentities != MAX_MAP_ENTITIES ; e = e->next) { g_numentities++; } } if ( ( ( g_numbrushes != s_lastbrushcount ) || ( g_numentities != s_lastentitycount ) ) || ( !s_didonce ) ) { Sys_UpdateStatusBar(); s_lastbrushcount = g_numbrushes; s_lastentitycount = g_numentities; s_didonce = true; } }