mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2024-11-23 12:52:46 +00:00
dbd3d6502a
I never liked it, but with C2x coming out, it's best to handle bools properly. I haven't gone through all the uses of int as bool (I'll leave that for fixing when I encounter them), but this gets QF working with both c2x (really, gnu2x because of raw strings).
1133 lines
24 KiB
C
1133 lines
24 KiB
C
/*
|
|
gl_draw.c
|
|
|
|
Draw functions for chars, textures, etc
|
|
|
|
Copyright (C) 1996-1997 Id Software, Inc.
|
|
|
|
This program 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.
|
|
|
|
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.
|
|
|
|
See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to:
|
|
|
|
Free Software Foundation, Inc.
|
|
59 Temple Place - Suite 330
|
|
Boston, MA 02111-1307, USA
|
|
|
|
*/
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_STRING_H
|
|
# include <string.h>
|
|
#endif
|
|
#ifdef HAVE_STRINGS_H
|
|
# include <strings.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "QF/cmd.h"
|
|
#include "QF/cvar.h"
|
|
#include "QF/draw.h"
|
|
#include "QF/dstring.h"
|
|
#include "QF/image.h"
|
|
#include "QF/quakefs.h"
|
|
#include "QF/render.h"
|
|
#include "QF/screen.h"
|
|
#include "QF/sys.h"
|
|
#include "QF/va.h"
|
|
#include "QF/vid.h"
|
|
#include "QF/GL/defines.h"
|
|
#include "QF/GL/funcs.h"
|
|
#include "QF/GL/qf_draw.h"
|
|
#include "QF/GL/qf_rmain.h"
|
|
#include "QF/GL/qf_rsurf.h"
|
|
#include "QF/GL/qf_textures.h"
|
|
#include "QF/GL/qf_vid.h"
|
|
#include "QF/GL/types.h"
|
|
#include "QF/ui/font.h"
|
|
#include "QF/ui/view.h"
|
|
|
|
#include "compat.h"
|
|
#include "r_internal.h"
|
|
#include "varrays.h"
|
|
|
|
#define CELL_SIZE (1.0 / 16.0) // conchars is 16x16
|
|
#define CELL_INSET (1.0 / 4.0) // of a pixel
|
|
typedef struct {
|
|
float tlx, tly;
|
|
float trx, try;
|
|
float brx, bry;
|
|
float blx, bly;
|
|
} cc_cell_t;
|
|
|
|
static int textUseVA;
|
|
static int tVAsize;
|
|
static int *tVAindices;
|
|
static int tVAcount;
|
|
static float *textVertices, *tV;
|
|
static float *textCoords, *tC;
|
|
|
|
static qpic_t *draw_backtile;
|
|
|
|
static cc_cell_t char_cells[256];
|
|
static GLuint translate_texture;
|
|
static int char_texture;
|
|
static int cs_texture; // crosshair texturea
|
|
static int gl_conback_texnum;
|
|
static cvar_t gl_conback_texnum_cvar = {
|
|
.name = "gl_conback_texnum",
|
|
.description =
|
|
"bind conback to this texture for debugging",
|
|
.default_value = "0",
|
|
.flags = CVAR_NONE,
|
|
.value = { .type = &cexpr_int, .value = &gl_conback_texnum },
|
|
};
|
|
|
|
static byte color_0_8[4] = { 204, 204, 204, 255 };
|
|
|
|
typedef struct {
|
|
int texnum;
|
|
} glpic_t;
|
|
|
|
typedef struct cachepic_s {
|
|
char name[MAX_QPATH];
|
|
bool dirty;
|
|
union {
|
|
qpic_t pic;
|
|
byte padding[sizeof (qpic_t) + 32];// for appended glpic FIXME
|
|
};
|
|
} cachepic_t;
|
|
|
|
#define MAX_CACHED_PICS 128
|
|
static cachepic_t cachepics[MAX_CACHED_PICS];
|
|
static int numcachepics;
|
|
|
|
static byte menuplyr_pixels[4096];
|
|
|
|
static int gl_2d_scale = 1;
|
|
|
|
typedef struct glfont_s {
|
|
font_t *font;
|
|
GLuint texid;
|
|
} glfont_t;
|
|
|
|
typedef struct glfontset_s
|
|
DARRAY_TYPE (glfont_t) glfontset_t;
|
|
|
|
static glfontset_t gl_fonts = DARRAY_STATIC_INIT (16);
|
|
|
|
static void
|
|
Draw_InitText (void)
|
|
{
|
|
int i;
|
|
|
|
if (vaelements < 0) {
|
|
textUseVA = 0;
|
|
tVAsize = 2048;
|
|
Sys_MaskPrintf (SYS_dev, "Text: Vertex Array use disabled.\n");
|
|
} else {
|
|
textUseVA = 1;
|
|
if (vaelements > 3)
|
|
tVAsize = vaelements - (vaelements % 4);
|
|
else
|
|
tVAsize = 2048;
|
|
Sys_MaskPrintf (SYS_dev, "Text: %i maximum vertex elements.\n",
|
|
tVAsize);
|
|
}
|
|
|
|
if (textVertices)
|
|
free (textVertices);
|
|
textVertices = calloc (tVAsize, 2 * sizeof (float));
|
|
|
|
if (textCoords)
|
|
free (textCoords);
|
|
textCoords = calloc (tVAsize, 2 * sizeof (float));
|
|
|
|
if (textUseVA) {
|
|
qfglTexCoordPointer (2, GL_FLOAT, 0, textCoords);
|
|
qfglVertexPointer (2, GL_FLOAT, 0, textVertices);
|
|
}
|
|
if (tVAindices)
|
|
free (tVAindices);
|
|
tVAindices = (int *) calloc (tVAsize, sizeof (int));
|
|
for (i = 0; i < tVAsize; i++)
|
|
tVAindices[i] = i;
|
|
}
|
|
|
|
qpic_t *
|
|
gl_Draw_MakePic (int width, int height, const byte *data)
|
|
{
|
|
glpic_t *gl;
|
|
qpic_t *pic;
|
|
|
|
pic = malloc (field_offset (qpic_t, data[sizeof (glpic_t)]));
|
|
pic->width = width;
|
|
pic->height = height;
|
|
gl = (glpic_t *) pic->data;
|
|
gl->texnum = GL_LoadTexture ("", width, height, data, false, true, 1);
|
|
return pic;
|
|
}
|
|
|
|
void
|
|
gl_Draw_DestroyPic (qpic_t *pic)
|
|
{
|
|
//FIXME gl texture management sucks
|
|
free (pic);
|
|
}
|
|
|
|
qpic_t *
|
|
gl_Draw_PicFromWad (const char *name)
|
|
{
|
|
glpic_t *gl;
|
|
qpic_t *p = 0, *pic;
|
|
tex_t *targa;
|
|
|
|
pic = W_GetLumpName (name);
|
|
targa = LoadImage (name, 1);
|
|
if (targa) {
|
|
p = malloc (sizeof (qpic_t) + sizeof (glpic_t));
|
|
p->width = pic->width;
|
|
p->height = pic->height;
|
|
gl = (glpic_t *) p->data;
|
|
if (targa->format < 4) {
|
|
gl->texnum = GL_LoadTexture (name, targa->width, targa->height,
|
|
targa->data, false, false, 3);
|
|
} else
|
|
gl->texnum = GL_LoadTexture (name, targa->width, targa->height,
|
|
targa->data, false, true, 4);
|
|
} else if (pic) {
|
|
p = pic;
|
|
gl = (glpic_t *) p->data;
|
|
gl->texnum = GL_LoadTexture (name, p->width, p->height, p->data,
|
|
false, true, 1);
|
|
}
|
|
return p;
|
|
}
|
|
|
|
static void
|
|
Draw_ClearCache (int phase, void *data)
|
|
{
|
|
cachepic_t *pic;
|
|
int i;
|
|
|
|
if (phase)
|
|
return;
|
|
for (pic = cachepics, i = 0; i < numcachepics; pic++, i++)
|
|
pic->dirty = true;
|
|
}
|
|
|
|
qpic_t *
|
|
gl_Draw_CachePic (const char *path, bool alpha)
|
|
{
|
|
cachepic_t *pic;
|
|
int i;
|
|
glpic_t *gl;
|
|
tex_t *targa;
|
|
|
|
// First, check if its cached..
|
|
for (pic = cachepics, i = 0; i < numcachepics; pic++, i++)
|
|
if ((!strcmp (path, pic->name)) && !pic->dirty)
|
|
return &pic->pic;
|
|
|
|
// Its not cached, lets make sure we have space in the cache..
|
|
if (numcachepics == MAX_CACHED_PICS)
|
|
Sys_Error ("menu_numcachepics == MAX_CACHED_PICS");
|
|
|
|
gl = (glpic_t *) pic->pic.data;
|
|
|
|
if (!strcmp (path + strlen (path) - 4, ".lmp")) {
|
|
// Load the picture..
|
|
qpic_t *dat = (qpic_t *) QFS_LoadFile (QFS_FOpenFile (path), 0);
|
|
if (!dat)
|
|
Sys_Error ("Draw_CachePic: failed to load %s", path);
|
|
|
|
// Adjust for endian..
|
|
SwapPic (dat);
|
|
// Check for a .tga first
|
|
targa = LoadImage (path, 1);
|
|
if (targa) {
|
|
if (targa->format < 4) {
|
|
gl->texnum = GL_LoadTexture ("", targa->width, targa->height,
|
|
targa->data, false, alpha, 3);
|
|
} else
|
|
gl->texnum = GL_LoadTexture ("", targa->width, targa->height,
|
|
targa->data, false, alpha, 4);
|
|
} else {
|
|
gl->texnum = GL_LoadTexture ("", dat->width, dat->height,
|
|
dat->data, false, alpha, 1);
|
|
}
|
|
pic->pic.width = dat->width;
|
|
pic->pic.height = dat->height;
|
|
if (!strcmp (path, "gfx/menuplyr.lmp"))
|
|
memcpy (menuplyr_pixels, dat->data, dat->width * dat->height);
|
|
free (dat);
|
|
} else
|
|
Sys_Error ("Draw_CachePic: failed to load %s", path);
|
|
|
|
memset (pic->name, 0, sizeof (pic->name));
|
|
strncpy (pic->name, path, sizeof (pic->name) - 1);
|
|
|
|
// Now lets mark this cache entry as used..
|
|
pic->dirty = false;
|
|
numcachepics++;
|
|
|
|
// And now we are done, return what was asked for..
|
|
return &pic->pic;
|
|
}
|
|
|
|
void
|
|
gl_Draw_UncachePic (const char *path)
|
|
{
|
|
cachepic_t *pic;
|
|
int i;
|
|
|
|
//FIXME chachpic and uncachepic suck in GL
|
|
for (pic = cachepics, i = 0; i < numcachepics; pic++, i++) {
|
|
if ((!strcmp (path, pic->name)) && !pic->dirty) {
|
|
pic->dirty = true;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
gl_Draw_TextBox (int x, int y, int width, int lines, byte alpha)
|
|
{
|
|
int cx, cy, n;
|
|
qpic_t *p;
|
|
|
|
// draw left side
|
|
color_white[3] = alpha;
|
|
qfglColor4ubv (color_white);
|
|
cx = x;
|
|
cy = y;
|
|
p = gl_Draw_CachePic ("gfx/box_tl.lmp", true);
|
|
gl_Draw_Pic (cx, cy, p);
|
|
p = gl_Draw_CachePic ("gfx/box_ml.lmp", true);
|
|
for (n = 0; n < lines; n++) {
|
|
cy += 8;
|
|
gl_Draw_Pic (cx, cy, p);
|
|
}
|
|
p = gl_Draw_CachePic ("gfx/box_bl.lmp", true);
|
|
gl_Draw_Pic (cx, cy + 8, p);
|
|
|
|
// draw middle
|
|
cx += 8;
|
|
while (width > 0) {
|
|
cy = y;
|
|
p = gl_Draw_CachePic ("gfx/box_tm.lmp", true);
|
|
gl_Draw_Pic (cx, cy, p);
|
|
p = gl_Draw_CachePic ("gfx/box_mm.lmp", true);
|
|
for (n = 0; n < lines; n++) {
|
|
cy += 8;
|
|
if (n == 1)
|
|
p = gl_Draw_CachePic ("gfx/box_mm2.lmp", true);
|
|
gl_Draw_Pic (cx, cy, p);
|
|
}
|
|
p = gl_Draw_CachePic ("gfx/box_bm.lmp", true);
|
|
gl_Draw_Pic (cx, cy + 8, p);
|
|
width -= 2;
|
|
cx += 16;
|
|
}
|
|
|
|
// draw right side
|
|
cy = y;
|
|
p = gl_Draw_CachePic ("gfx/box_tr.lmp", true);
|
|
gl_Draw_Pic (cx, cy, p);
|
|
p = gl_Draw_CachePic ("gfx/box_mr.lmp", true);
|
|
for (n = 0; n < lines; n++) {
|
|
cy += 8;
|
|
gl_Draw_Pic (cx, cy, p);
|
|
}
|
|
p = gl_Draw_CachePic ("gfx/box_br.lmp", true);
|
|
gl_Draw_Pic (cx, cy + 8, p);
|
|
qfglColor3ubv (color_white);
|
|
}
|
|
|
|
void
|
|
gl_Draw_Init (void)
|
|
{
|
|
int i;
|
|
tex_t *image;
|
|
float width, height;
|
|
qpic_t *ch_pic;
|
|
|
|
Cmd_AddCommand ("gl_texturemode", &GL_TextureMode_f,
|
|
"Texture mipmap quality.");
|
|
|
|
QFS_GamedirCallback (Draw_ClearCache, 0);
|
|
|
|
// load the console background and the charset by hand, because we need to
|
|
// write the version string into the background before turning it into a
|
|
// texture
|
|
|
|
image = LoadImage ("gfx/conchars", 1);
|
|
if (image) {
|
|
if (image->format < 4) {
|
|
char_texture = GL_LoadTexture ("charset", image->width,
|
|
image->height, image->data, false,
|
|
false, 3);
|
|
} else
|
|
char_texture = GL_LoadTexture ("charset", image->width,
|
|
image->height, image->data, false,
|
|
true, 4);
|
|
width = image->width;
|
|
height = image->height;
|
|
} else if ((draw_chars = W_GetLumpName ("conchars"))) {
|
|
for (i = 0; i < 256 * 64; i++) {
|
|
if (draw_chars[i] == 0) {
|
|
draw_chars[i] = 255; // proper transparent color
|
|
}
|
|
}
|
|
|
|
char_texture = GL_LoadTexture ("charset", 128, 128, draw_chars, false,
|
|
true, 1);
|
|
width = 128;
|
|
height = 128;
|
|
} else {
|
|
qpic_t *charspic = Draw_Font8x8Pic ();
|
|
char_texture = GL_LoadTexture ("charset", 128, 128, charspic->data,
|
|
false, true, 1);
|
|
width = 128;
|
|
height = 128;
|
|
}
|
|
qfglTexParameterf (GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
qfglTexParameterf (GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
|
|
// initialize the character cell texture coordinates.
|
|
for (i = 0; i < 256; i++) {
|
|
float fcol, frow;
|
|
|
|
fcol = (i & 15) * CELL_SIZE;
|
|
frow = (i >> 4) * CELL_SIZE;
|
|
|
|
char_cells[i].tlx = fcol + CELL_INSET / width;
|
|
char_cells[i].tly = frow + CELL_INSET / height;
|
|
char_cells[i].trx = fcol - CELL_INSET / width + CELL_SIZE;
|
|
char_cells[i].try = frow + CELL_INSET / height;
|
|
char_cells[i].brx = fcol - CELL_INSET / width + CELL_SIZE;
|
|
char_cells[i].bry = frow - CELL_INSET / height + CELL_SIZE;
|
|
char_cells[i].blx = fcol + CELL_INSET / width;
|
|
char_cells[i].bly = frow - CELL_INSET / height + CELL_SIZE;
|
|
}
|
|
|
|
ch_pic = Draw_CrosshairPic ();
|
|
cs_texture = GL_LoadTexture ("crosshair",
|
|
CROSSHAIR_WIDTH * CROSSHAIR_TILEX,
|
|
CROSSHAIR_HEIGHT * CROSSHAIR_TILEY,
|
|
ch_pic->data, false, true, 1);
|
|
free (ch_pic);
|
|
|
|
qfglTexParameterf (GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
qfglTexParameterf (GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
|
|
// save a texture slot for translated picture
|
|
qfglGenTextures (1, &translate_texture);
|
|
|
|
// get the other pics we need
|
|
draw_backtile = gl_Draw_PicFromWad ("backtile");
|
|
|
|
Draw_InitText ();
|
|
|
|
Cvar_Register (&gl_conback_texnum_cvar, 0, 0);
|
|
}
|
|
|
|
static inline void
|
|
flush_text (void)
|
|
{
|
|
qfglBindTexture (GL_TEXTURE_2D, char_texture);
|
|
if (textUseVA) {
|
|
qfglDrawElements (GL_QUADS, tVAcount, GL_UNSIGNED_INT, tVAindices);
|
|
} else {
|
|
float *v = textVertices;
|
|
float *c = textCoords;
|
|
int i;
|
|
|
|
qfglBegin (GL_QUADS);
|
|
for (i = 0; i < tVAcount; i++) {
|
|
qfglTexCoord2fv (c);
|
|
qfglVertex2fv (v);
|
|
c += 2;
|
|
v += 2;
|
|
}
|
|
qfglEnd ();
|
|
}
|
|
tVAcount = 0;
|
|
tV = textVertices;
|
|
tC = textCoords;
|
|
}
|
|
|
|
static inline void
|
|
queue_character (float x, float y, int chr)
|
|
{
|
|
float *coord = &char_cells[chr & 255].tlx;
|
|
*tV++ = x;
|
|
*tV++ = y;
|
|
*tV++ = x + 8.0;
|
|
*tV++ = y;
|
|
*tV++ = x + 8.0;
|
|
*tV++ = y + 8.0;
|
|
*tV++ = x;
|
|
*tV++ = y + 8.0;
|
|
*tC++ = *coord++;
|
|
*tC++ = *coord++;
|
|
*tC++ = *coord++;
|
|
*tC++ = *coord++;
|
|
*tC++ = *coord++;
|
|
*tC++ = *coord++;
|
|
*tC++ = *coord++;
|
|
*tC++ = *coord++;
|
|
}
|
|
|
|
static inline void
|
|
tVA_increment (void)
|
|
{
|
|
tVAcount += 4;
|
|
if (tVAcount + 4 > tVAsize)
|
|
flush_text ();
|
|
}
|
|
|
|
void
|
|
gl_Draw_CharBuffer (int x, int y, draw_charbuffer_t *buffer)
|
|
{
|
|
const byte *line = (byte *) buffer->chars;
|
|
int width = buffer->width;
|
|
int height = buffer->height;
|
|
while (height-- > 0) {
|
|
for (int i = 0; i < width; i++) {
|
|
gl_Draw_Character (x + i * 8, y, line[i]);
|
|
}
|
|
line += width;
|
|
y += 8;
|
|
}
|
|
}
|
|
|
|
/*
|
|
Draw_Character
|
|
|
|
Draws one 8*8 graphics character with 0 being transparent.
|
|
It can be clipped to the top of the screen to allow the console to be
|
|
smoothly scrolled off.
|
|
*/
|
|
void
|
|
gl_Draw_Character (int x, int y, unsigned int chr)
|
|
{
|
|
chr &= 255;
|
|
|
|
if (chr == 32)
|
|
return; // space
|
|
if (y <= -8)
|
|
return; // totally off screen
|
|
|
|
queue_character ((float) x, (float) y, chr);
|
|
tVA_increment ();
|
|
}
|
|
|
|
void
|
|
gl_Draw_String (int x, int y, const char *str)
|
|
{
|
|
unsigned char chr;
|
|
float x1, y1;
|
|
|
|
if (!str || !str[0])
|
|
return;
|
|
if (y <= -8)
|
|
return; // totally off screen
|
|
|
|
x1 = (float) x;
|
|
y1 = (float) y;
|
|
|
|
while (*str) {
|
|
if ((chr = *str++) != 32) { // Don't render spaces
|
|
queue_character (x1, y1, chr);
|
|
tVA_increment ();
|
|
}
|
|
x1 += 8.0;
|
|
}
|
|
}
|
|
|
|
void
|
|
gl_Draw_nString (int x, int y, const char *str, int count)
|
|
{
|
|
unsigned char chr;
|
|
float x1, y1;
|
|
|
|
if (!str || !str[0])
|
|
return;
|
|
if (y <= -8)
|
|
return; // totally off screen
|
|
|
|
x1 = (float) x;
|
|
y1 = (float) y;
|
|
|
|
while (count-- && *str) {
|
|
if ((chr = *str++) != 32) { // Don't render spaces
|
|
queue_character (x1, y1, chr);
|
|
tVA_increment ();
|
|
}
|
|
x1 += 8.0;
|
|
}
|
|
}
|
|
|
|
void
|
|
gl_Draw_AltString (int x, int y, const char *str)
|
|
{
|
|
unsigned char chr;
|
|
float x1, y1;
|
|
|
|
if (!str || !str[0])
|
|
return;
|
|
if (y <= -8)
|
|
return; // totally off screen
|
|
|
|
x1 = (float) x;
|
|
y1 = (float) y;
|
|
|
|
while (*str) {
|
|
if ((chr = *str++ | 0x80) != (0x80 | 32)) { // Don't render spaces
|
|
queue_character (x1, y1, chr);
|
|
tVA_increment ();
|
|
}
|
|
x1 += 8.0;
|
|
}
|
|
}
|
|
|
|
static void
|
|
crosshair_1 (int x, int y)
|
|
{
|
|
gl_Draw_Character (x - 4, y - 4, '+');
|
|
}
|
|
|
|
static void
|
|
crosshair_2 (int x, int y)
|
|
{
|
|
unsigned char *pColor;
|
|
|
|
pColor = (unsigned char *) &d_8to24table[crosshaircolor];
|
|
qfglColor4ubv (pColor);
|
|
qfglBindTexture (GL_TEXTURE_2D, cs_texture);
|
|
|
|
qfglBegin (GL_QUADS);
|
|
|
|
qfglTexCoord2f (0, 0);
|
|
qfglVertex2f (x - 3, y - 3);
|
|
qfglTexCoord2f (0.5, 0);
|
|
qfglVertex2f (x + 5, y - 3);
|
|
qfglTexCoord2f (0.5, 0.5);
|
|
qfglVertex2f (x + 5, y + 5);
|
|
qfglTexCoord2f (0, 0.5);
|
|
qfglVertex2f (x - 3, y + 5);
|
|
|
|
qfglEnd ();
|
|
qfglColor3ubv (color_white);
|
|
}
|
|
|
|
static void
|
|
crosshair_3 (int x, int y)
|
|
{
|
|
unsigned char *pColor;
|
|
|
|
pColor = (unsigned char *) &d_8to24table[crosshaircolor];
|
|
qfglColor4ubv (pColor);
|
|
qfglBindTexture (GL_TEXTURE_2D, cs_texture);
|
|
|
|
qfglBegin (GL_QUADS);
|
|
|
|
qfglTexCoord2f (0.5, 0);
|
|
qfglVertex2f (x - 3, y - 3);
|
|
qfglTexCoord2f (1, 0);
|
|
qfglVertex2f (x + 5, y - 3);
|
|
qfglTexCoord2f (1, 0.5);
|
|
qfglVertex2f (x + 5, y + 5);
|
|
qfglTexCoord2f (0.5, 0.5);
|
|
qfglVertex2f (x - 3, y + 5);
|
|
|
|
qfglEnd ();
|
|
qfglColor3ubv (color_white);
|
|
}
|
|
|
|
static void
|
|
crosshair_4 (int x, int y)
|
|
{
|
|
unsigned char *pColor;
|
|
|
|
pColor = (unsigned char *) &d_8to24table[crosshaircolor];
|
|
qfglColor4ubv (pColor);
|
|
qfglBindTexture (GL_TEXTURE_2D, cs_texture);
|
|
|
|
qfglBegin (GL_QUADS);
|
|
|
|
qfglTexCoord2f (0, 0.5);
|
|
qfglVertex2f (x - 3, y - 3);
|
|
qfglTexCoord2f (0.5, 0.5);
|
|
qfglVertex2f (x + 5, y - 5);
|
|
qfglTexCoord2f (0.5, 1);
|
|
qfglVertex2f (x + 5, y + 5);
|
|
qfglTexCoord2f (0, 1);
|
|
qfglVertex2f (x - 3, y + 5);
|
|
|
|
qfglEnd ();
|
|
qfglColor3ubv (color_white);
|
|
}
|
|
|
|
static void
|
|
crosshair_5 (int x, int y) //FIXME don't use until the data is filled in
|
|
{
|
|
unsigned char *pColor;
|
|
|
|
pColor = (unsigned char *) &d_8to24table[crosshaircolor];
|
|
qfglColor4ubv (pColor);
|
|
qfglBindTexture (GL_TEXTURE_2D, cs_texture);
|
|
|
|
qfglBegin (GL_QUADS);
|
|
|
|
qfglTexCoord2f (0.5, 0.5);
|
|
qfglVertex2f (x - 3, y - 3);
|
|
qfglTexCoord2f (1, 0.5);
|
|
qfglVertex2f (x + 5, y - 3);
|
|
qfglTexCoord2f (1, 1);
|
|
qfglVertex2f (x + 5, y + 5);
|
|
qfglTexCoord2f (0.5, 1);
|
|
qfglVertex2f (x - 3, y + 5);
|
|
|
|
qfglEnd ();
|
|
qfglColor3ubv (color_white);
|
|
}
|
|
|
|
static void (*crosshair_func[]) (int x, int y) = {
|
|
crosshair_1,
|
|
crosshair_2,
|
|
crosshair_3,
|
|
crosshair_4,
|
|
crosshair_5,
|
|
};
|
|
|
|
void
|
|
gl_Draw_Crosshair (void)
|
|
{
|
|
int x, y;
|
|
int ch;
|
|
|
|
ch = crosshair - 1;
|
|
if ((unsigned) ch >= sizeof (crosshair_func) / sizeof (crosshair_func[0]))
|
|
return;
|
|
|
|
int s = 2 * gl_2d_scale;
|
|
x = vid.width / s + cl_crossx;
|
|
y = vid.height / s + cl_crossy;
|
|
|
|
crosshair_func[ch] (x, y);
|
|
}
|
|
|
|
void
|
|
gl_Draw_CrosshairAt (int ch, int x, int y)
|
|
{
|
|
ch -= 1;
|
|
if ((unsigned) ch >= sizeof (crosshair_func) / sizeof (crosshair_func[0]))
|
|
return;
|
|
|
|
crosshair_func[ch] (x, y);
|
|
}
|
|
|
|
void
|
|
gl_Draw_Pic (int x, int y, qpic_t *pic)
|
|
{
|
|
glpic_t *gl;
|
|
|
|
gl = (glpic_t *) pic->data;
|
|
|
|
qfglBindTexture (GL_TEXTURE_2D, gl->texnum);
|
|
qfglBegin (GL_QUADS);
|
|
qfglTexCoord2f (0, 0);
|
|
qfglVertex2f (x, y);
|
|
qfglTexCoord2f (1, 0);
|
|
qfglVertex2f (x + pic->width, y);
|
|
qfglTexCoord2f (1, 1);
|
|
qfglVertex2f (x + pic->width, y + pic->height);
|
|
qfglTexCoord2f (0, 1);
|
|
qfglVertex2f (x, y + pic->height);
|
|
qfglEnd ();
|
|
}
|
|
|
|
void
|
|
gl_Draw_FitPic (int x, int y, int width, int height, qpic_t *pic)
|
|
{
|
|
glpic_t *gl;
|
|
|
|
gl = (glpic_t *) pic->data;
|
|
|
|
qfglBindTexture (GL_TEXTURE_2D, gl->texnum);
|
|
qfglBegin (GL_QUADS);
|
|
qfglTexCoord2f (0, 0);
|
|
qfglVertex2f (x, y);
|
|
qfglTexCoord2f (1, 0);
|
|
qfglVertex2f (x + width, y);
|
|
qfglTexCoord2f (1, 1);
|
|
qfglVertex2f (x + width, y + height);
|
|
qfglTexCoord2f (0, 1);
|
|
qfglVertex2f (x, y + height);
|
|
qfglEnd ();
|
|
}
|
|
|
|
void
|
|
gl_Draw_Picf (float x, float y, qpic_t *pic)
|
|
{
|
|
glpic_t *gl;
|
|
|
|
gl = (glpic_t *) pic->data;
|
|
|
|
qfglBindTexture (GL_TEXTURE_2D, gl->texnum);
|
|
qfglBegin (GL_QUADS);
|
|
qfglTexCoord2f (0, 0);
|
|
qfglVertex2f (x, y);
|
|
qfglTexCoord2f (1, 0);
|
|
qfglVertex2f (x + pic->width, y);
|
|
qfglTexCoord2f (1, 1);
|
|
qfglVertex2f (x + pic->width, y + pic->height);
|
|
qfglTexCoord2f (0, 1);
|
|
qfglVertex2f (x, y + pic->height);
|
|
qfglEnd ();
|
|
}
|
|
|
|
void
|
|
gl_Draw_SubPic (int x, int y, qpic_t *pic, int srcx, int srcy, int width,
|
|
int height)
|
|
{
|
|
float newsl, newtl, newsh, newth;
|
|
glpic_t *gl;
|
|
|
|
gl = (glpic_t *) pic->data;
|
|
|
|
newsl = (float) srcx / (float) pic->width;
|
|
newsh = newsl + (float) width / (float) pic->width;
|
|
|
|
newtl = (float) srcy / (float) pic->height;
|
|
newth = newtl + (float) height / (float) pic->height;
|
|
|
|
qfglColor3ubv (color_0_8);
|
|
qfglBindTexture (GL_TEXTURE_2D, gl->texnum);
|
|
qfglBegin (GL_QUADS);
|
|
qfglTexCoord2f (newsl, newtl);
|
|
qfglVertex2f (x, y);
|
|
qfglTexCoord2f (newsh, newtl);
|
|
qfglVertex2f (x + width, y);
|
|
qfglTexCoord2f (newsh, newth);
|
|
qfglVertex2f (x + width, y + height);
|
|
qfglTexCoord2f (newsl, newth);
|
|
qfglVertex2f (x, y + height);
|
|
qfglEnd ();
|
|
qfglColor3ubv (color_white);
|
|
}
|
|
|
|
/*
|
|
Draw_ConsoleBackground
|
|
|
|
Draws console background (obviously!) Completely rewritten to use
|
|
several simple yet very cool GL effects. --KB
|
|
*/
|
|
void
|
|
gl_Draw_ConsoleBackground (int lines, byte alpha)
|
|
{
|
|
float ofs;
|
|
glpic_t *gl;
|
|
qpic_t *conback;
|
|
|
|
GL_FlushText (); // Flush text that should be rendered before the console
|
|
|
|
// This can be a CachePic now, just like in software
|
|
conback = gl_Draw_CachePic ("gfx/conback.lmp", false);
|
|
gl = (glpic_t *) conback->data;
|
|
|
|
// spin the console? - effect described in a QER tutorial
|
|
if (gl_conspin) {
|
|
static float xangle = 0;
|
|
static float xfactor = .3f;
|
|
static float xstep = .005f;
|
|
|
|
qfglPushMatrix ();
|
|
qfglMatrixMode (GL_TEXTURE);
|
|
qfglPushMatrix ();
|
|
qfglLoadIdentity ();
|
|
xangle += gl_conspin;
|
|
xfactor += xstep;
|
|
if (xfactor > 8 || xfactor < .3f)
|
|
xstep = -xstep;
|
|
qfglRotatef (xangle, 0, 0, 1);
|
|
qfglScalef (xfactor, xfactor, xfactor);
|
|
}
|
|
// slide console up/down or stretch it?
|
|
if (gl_constretch) {
|
|
ofs = 0;
|
|
} else
|
|
ofs = (vid.height - gl_2d_scale * lines) / (float) vid.height;
|
|
|
|
color_0_8[3] = alpha;
|
|
qfglColor4ubv (color_0_8);
|
|
|
|
// draw the console texture
|
|
if (gl_conback_texnum) {
|
|
qfglBindTexture (GL_TEXTURE_2D, gl_conback_texnum);
|
|
} else {
|
|
qfglBindTexture (GL_TEXTURE_2D, gl->texnum);
|
|
}
|
|
qfglBegin (GL_QUADS);
|
|
qfglTexCoord2f (0, 0 + ofs);
|
|
qfglVertex2f (0, 0);
|
|
qfglTexCoord2f (1, 0 + ofs);
|
|
qfglVertex2f (vid.width / gl_2d_scale, 0);
|
|
qfglTexCoord2f (1, 1);
|
|
qfglVertex2f (vid.width / gl_2d_scale, lines);
|
|
qfglTexCoord2f (0, 1);
|
|
qfglVertex2f (0, lines);
|
|
qfglEnd ();
|
|
|
|
// turn off alpha blending
|
|
if (alpha < 255) {
|
|
qfglColor3ubv (color_0_8);
|
|
}
|
|
|
|
if (gl_conspin) {
|
|
qfglPopMatrix ();
|
|
qfglMatrixMode (GL_MODELVIEW);
|
|
qfglPopMatrix ();
|
|
}
|
|
|
|
int len = strlen (cl_verstring);
|
|
gl_Draw_AltString (vid.width - len * 8 - 11, lines - 14,
|
|
cl_verstring);
|
|
qfglColor3ubv (color_white);
|
|
}
|
|
|
|
/*
|
|
Draw_TileClear
|
|
|
|
This repeats a 64*64 tile graphic to fill the screen around a sized down
|
|
refresh window.
|
|
*/
|
|
void
|
|
gl_Draw_TileClear (int x, int y, int w, int h)
|
|
{
|
|
if (!draw_backtile) {
|
|
return;
|
|
}
|
|
glpic_t *gl;
|
|
qfglColor3ubv (color_0_8);
|
|
gl = (glpic_t *) draw_backtile->data;
|
|
qfglBindTexture (GL_TEXTURE_2D, gl->texnum);
|
|
qfglBegin (GL_QUADS);
|
|
qfglTexCoord2f (x / 64.0, y / 64.0);
|
|
qfglVertex2f (x, y);
|
|
qfglTexCoord2f ((x + w) / 64.0, y / 64.0);
|
|
qfglVertex2f (x + w, y);
|
|
qfglTexCoord2f ((x + w) / 64.0, (y + h) / 64.0);
|
|
qfglVertex2f (x + w, y + h);
|
|
qfglTexCoord2f (x / 64.0, (y + h) / 64.0);
|
|
qfglVertex2f (x, y + h);
|
|
qfglEnd ();
|
|
qfglColor3ubv (color_white);
|
|
}
|
|
|
|
/*
|
|
Draw_Fill
|
|
|
|
Fills a box of pixels with a single color
|
|
*/
|
|
void
|
|
gl_Draw_Fill (int x, int y, int w, int h, int c)
|
|
{
|
|
qfglDisable (GL_TEXTURE_2D);
|
|
qfglColor3ubv (vid.palette + c * 3);
|
|
|
|
qfglBegin (GL_QUADS);
|
|
|
|
qfglVertex2f (x, y);
|
|
qfglVertex2f (x + w, y);
|
|
qfglVertex2f (x + w, y + h);
|
|
qfglVertex2f (x, y + h);
|
|
|
|
qfglEnd ();
|
|
qfglColor3ubv (color_white);
|
|
qfglEnable (GL_TEXTURE_2D);
|
|
}
|
|
|
|
void
|
|
gl_Draw_Line (int x0, int y0, int x1, int y1, int c)
|
|
{
|
|
qfglDisable (GL_TEXTURE_2D);
|
|
qfglColor3ubv (vid.palette + c * 3);
|
|
|
|
qfglBegin (GL_LINES);
|
|
|
|
qfglVertex2f (x0, y0);
|
|
qfglVertex2f (x1, y1);
|
|
|
|
qfglEnd ();
|
|
qfglColor3ubv (color_white);
|
|
qfglEnable (GL_TEXTURE_2D);
|
|
}
|
|
|
|
void
|
|
gl_Draw_FadeScreen (void)
|
|
{
|
|
GL_FlushText (); // Flush text that should be rendered before the menu
|
|
|
|
qfglDisable (GL_TEXTURE_2D);
|
|
qfglColor4ub (0, 0, 0, 179);
|
|
qfglBegin (GL_QUADS);
|
|
|
|
qfglVertex2f (0, 0);
|
|
qfglVertex2f (vid.width, 0);
|
|
qfglVertex2f (vid.width, vid.height);
|
|
qfglVertex2f (0, vid.height);
|
|
|
|
qfglEnd ();
|
|
qfglColor3ubv (color_white);
|
|
qfglEnable (GL_TEXTURE_2D);
|
|
}
|
|
|
|
static void
|
|
set_2d (int width, int height)
|
|
{
|
|
qfglViewport (0, 0, vid.width, vid.height);
|
|
|
|
qfglMatrixMode (GL_PROJECTION);
|
|
qfglLoadIdentity ();
|
|
qfglOrtho (0, width, height, 0, -99999, 99999);
|
|
|
|
qfglMatrixMode (GL_MODELVIEW);
|
|
qfglLoadIdentity ();
|
|
|
|
qfglDisable (GL_DEPTH_TEST);
|
|
qfglDisable (GL_CULL_FACE);
|
|
|
|
qfglColor3ubv (color_white);
|
|
|
|
qfglEnableClientState (GL_VERTEX_ARRAY);
|
|
qfglVertexPointer (2, GL_FLOAT, 0, textVertices);
|
|
qfglEnableClientState (GL_TEXTURE_COORD_ARRAY);
|
|
qfglTexCoordPointer (2, GL_FLOAT, 0, textCoords);
|
|
qfglDisableClientState (GL_COLOR_ARRAY);
|
|
}
|
|
|
|
void
|
|
GL_Set2D (void)
|
|
{
|
|
set_2d (vid.width, vid.height);
|
|
}
|
|
|
|
void
|
|
GL_Set2DScaled (void)
|
|
{
|
|
set_2d (vid.width / gl_2d_scale, vid.height / gl_2d_scale);
|
|
}
|
|
|
|
void
|
|
gl_Draw_SetScale (int scale)
|
|
{
|
|
gl_2d_scale = scale;
|
|
}
|
|
|
|
void
|
|
GL_DrawReset (void)
|
|
{
|
|
tVAcount = 0;
|
|
tV = textVertices;
|
|
tC = textCoords;
|
|
}
|
|
|
|
void
|
|
GL_FlushText (void)
|
|
{
|
|
if (tVAcount) {
|
|
flush_text ();
|
|
}
|
|
}
|
|
|
|
void
|
|
gl_Draw_BlendScreen (quat_t color)
|
|
{
|
|
if (!color[3])
|
|
return;
|
|
|
|
qfglDisable (GL_TEXTURE_2D);
|
|
|
|
qfglBegin (GL_QUADS);
|
|
|
|
qfglColor4fv (color);
|
|
qfglVertex2f (0, 0);
|
|
qfglVertex2f (vid.width, 0);
|
|
qfglVertex2f (vid.width, vid.height);
|
|
qfglVertex2f (0, vid.height);
|
|
|
|
qfglEnd ();
|
|
|
|
qfglColor3ubv (color_white);
|
|
qfglEnable (GL_TEXTURE_2D);
|
|
}
|
|
|
|
int
|
|
gl_Draw_AddFont (font_t *rfont)
|
|
{
|
|
int fontid = gl_fonts.size;
|
|
DARRAY_OPEN_AT (&gl_fonts, fontid, 1);
|
|
glfont_t *font = &gl_fonts.a[fontid];
|
|
|
|
font->font = rfont;
|
|
tex_t tex = {
|
|
.width = rfont->scrap.width,
|
|
.height = rfont->scrap.height,
|
|
.format = tex_a,
|
|
.loaded = 1,
|
|
.data = rfont->scrap_bitmap,
|
|
};
|
|
font->texid = GL_LoadTex ("", 0, &tex);
|
|
return fontid;
|
|
}
|
|
|
|
void
|
|
gl_Draw_Glyph (int x, int y, int fontid, int glyphid, int c)
|
|
{
|
|
if (fontid < 0 || (unsigned) fontid > gl_fonts.size) {
|
|
return;
|
|
}
|
|
|
|
glfont_t *font = &gl_fonts.a[fontid];
|
|
font_t *rfont = font->font;
|
|
vrect_t *rect = &rfont->glyph_rects[glyphid];
|
|
|
|
float w = rect->width;
|
|
float h = rect->height;
|
|
float u = rect->x;
|
|
float v = rect->y;
|
|
float s = 1.0 / rfont->scrap.width;
|
|
float t = 1.0 / rfont->scrap.height;
|
|
|
|
qfglBindTexture (GL_TEXTURE_2D, gl_fonts.a[fontid].texid);
|
|
qfglBegin (GL_QUADS);
|
|
|
|
byte color[4] = { VectorExpand (vid.palette + c * 3), 255 };
|
|
qfglColor4ubv (color);
|
|
qfglTexCoord2f (u * s, v * t);
|
|
qfglVertex2f (x, y);
|
|
qfglTexCoord2f ((u + w) * s, v * t);
|
|
qfglVertex2f (x + w, y);
|
|
qfglTexCoord2f ((u + w) * s, (v + h) * t);
|
|
qfglVertex2f (x + w, y + h);
|
|
qfglTexCoord2f (u * s, (v + h) * t);
|
|
qfglVertex2f (x, y + h);
|
|
|
|
qfglEnd ();
|
|
qfglColor4ubv (color_white);
|
|
}
|