mirror of
https://github.com/DrBeef/JKXR.git
synced 2025-01-22 16:31:12 +00:00
2176 lines
56 KiB
C++
2176 lines
56 KiB
C++
/*
|
|
===========================================================================
|
|
Copyright (C) 1999 - 2005, Id Software, Inc.
|
|
Copyright (C) 2000 - 2013, Raven Software, Inc.
|
|
Copyright (C) 2001 - 2013, Activision, Inc.
|
|
Copyright (C) 2005 - 2015, ioquake3 contributors
|
|
Copyright (C) 2013 - 2015, OpenJK contributors
|
|
|
|
This file is part of the OpenJK source code.
|
|
|
|
OpenJK is free software; you can redistribute it and/or modify it
|
|
under the terms of the GNU General Public License version 2 as
|
|
published by the Free Software Foundation.
|
|
|
|
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, see <http://www.gnu.org/licenses/>.
|
|
===========================================================================
|
|
*/
|
|
|
|
#include "../server/exe_headers.h"
|
|
|
|
/*****************************************************************************
|
|
* name: cl_cin.c
|
|
*
|
|
* desc: video and cinematic playback
|
|
*
|
|
* $Archive: /MissionPack/code/client/cl_cin.c $
|
|
* $Author: Ttimo $
|
|
* $Revision: 82 $
|
|
* $Modtime: 4/13/01 4:48p $
|
|
* $Date: 4/13/01 4:48p $
|
|
*
|
|
* cl_glconfig.hwtype trtypes 3dfx/ragepro need 256x256
|
|
*
|
|
*****************************************************************************/
|
|
|
|
#include "client.h"
|
|
#include "client_ui.h" // CHC
|
|
#include "snd_local.h"
|
|
#include "qcommon/stringed_ingame.h"
|
|
|
|
#define MAXSIZE 8
|
|
#define MINSIZE 4
|
|
|
|
#define DEFAULT_CIN_WIDTH 512
|
|
#define DEFAULT_CIN_HEIGHT 512
|
|
|
|
#define ROQ_QUAD 0x1000
|
|
#define ROQ_QUAD_INFO 0x1001
|
|
#define ROQ_CODEBOOK 0x1002
|
|
#define ROQ_QUAD_VQ 0x1011
|
|
#define ROQ_QUAD_JPEG 0x1012
|
|
#define ROQ_QUAD_HANG 0x1013
|
|
#define ROQ_PACKET 0x1030
|
|
#define ZA_SOUND_MONO 0x1020
|
|
#define ZA_SOUND_STEREO 0x1021
|
|
|
|
#define MAX_VIDEO_HANDLES 16
|
|
|
|
extern void S_CIN_StopSound(sfxHandle_t sfxHandle);
|
|
static void RoQ_init( void );
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Class: trFMV
|
|
*
|
|
* Description: RoQ/RnR manipulation routines
|
|
* not entirely complete for first run
|
|
*
|
|
******************************************************************************/
|
|
|
|
static long ROQ_YY_tab[256];
|
|
static long ROQ_UB_tab[256];
|
|
static long ROQ_UG_tab[256];
|
|
static long ROQ_VG_tab[256];
|
|
static long ROQ_VR_tab[256];
|
|
static unsigned short vq2[256*16*4];
|
|
static unsigned short vq4[256*64*4];
|
|
static unsigned short vq8[256*256*4];
|
|
|
|
typedef struct {
|
|
byte linbuf[DEFAULT_CIN_WIDTH*DEFAULT_CIN_HEIGHT*4*2];
|
|
byte file[65536];
|
|
short sqrTable[256];
|
|
|
|
int mcomp[256];
|
|
byte *qStatus[2][32768];
|
|
|
|
long oldXOff, oldYOff, oldysize, oldxsize;
|
|
|
|
int currentHandle;
|
|
} cinematics_t;
|
|
|
|
typedef struct {
|
|
char fileName[MAX_OSPATH];
|
|
int CIN_WIDTH, CIN_HEIGHT;
|
|
int xpos, ypos, width, height;
|
|
qboolean looping, holdAtEnd, dirty, alterGameState, silent, shader;
|
|
fileHandle_t iFile; // 0 = none
|
|
e_status status;
|
|
unsigned int startTime;
|
|
unsigned int lastTime;
|
|
long tfps;
|
|
long RoQPlayed;
|
|
long ROQSize;
|
|
unsigned int RoQFrameSize;
|
|
long onQuad;
|
|
long numQuads;
|
|
long samplesPerLine;
|
|
unsigned int roq_id;
|
|
long screenDelta;
|
|
|
|
void ( *VQ0)(byte *status, void *qdata );
|
|
void ( *VQ1)(byte *status, void *qdata );
|
|
void ( *VQNormal)(byte *status, void *qdata );
|
|
void ( *VQBuffer)(byte *status, void *qdata );
|
|
|
|
long samplesPerPixel; // defaults to 2
|
|
byte* gray;
|
|
unsigned int xsize, ysize, maxsize, minsize;
|
|
|
|
qboolean half, smootheddouble, inMemory;
|
|
long normalBuffer0;
|
|
long roq_flags;
|
|
long roqF0;
|
|
long roqF1;
|
|
long t[2];
|
|
long roqFPS;
|
|
int playonwalls;
|
|
byte* buf;
|
|
long drawX, drawY;
|
|
sfxHandle_t hSFX; // 0 = none
|
|
qhandle_t hCRAWLTEXT; // 0 = none
|
|
} cin_cache;
|
|
|
|
static cinematics_t cin;
|
|
static cin_cache cinTable[MAX_VIDEO_HANDLES];
|
|
static int currentHandle = -1;
|
|
static int CL_handle = -1;
|
|
static int CL_iPlaybackStartTime; // so I can stop users quitting playback <1 second after it starts
|
|
|
|
extern int s_soundtime; // sample PAIRS
|
|
extern int s_paintedtime; // sample PAIRS
|
|
|
|
|
|
void CIN_CloseAllVideos(void) {
|
|
int i;
|
|
|
|
for ( i = 0 ; i < MAX_VIDEO_HANDLES ; i++ ) {
|
|
if (cinTable[i].fileName[0] != 0 ) {
|
|
CIN_StopCinematic(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static int CIN_HandleForVideo(void) {
|
|
int i;
|
|
|
|
for ( i = 0 ; i < MAX_VIDEO_HANDLES ; i++ ) {
|
|
if ( cinTable[i].fileName[0] == 0 ) {
|
|
return i;
|
|
}
|
|
}
|
|
Com_Error( ERR_DROP, "CIN_HandleForVideo: none free" );
|
|
return -1;
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// RllSetupTable
|
|
//
|
|
// Allocates and initializes the square table.
|
|
//
|
|
// Parameters: None
|
|
//
|
|
// Returns: Nothing
|
|
//-----------------------------------------------------------------------------
|
|
static void RllSetupTable( void )
|
|
{
|
|
int z;
|
|
|
|
for (z=0;z<128;z++) {
|
|
cin.sqrTable[z] = (short)(z*z);
|
|
cin.sqrTable[z+128] = (short)(-cin.sqrTable[z]);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// RllDecodeMonoToMono
|
|
//
|
|
// Decode mono source data into a mono buffer.
|
|
//
|
|
// Parameters: from -> buffer holding encoded data
|
|
// to -> buffer to hold decoded data
|
|
// size = number of bytes of input (= # of shorts of output)
|
|
// signedOutput = 0 for unsigned output, non-zero for signed output
|
|
// flag = flags from asset header
|
|
//
|
|
// Returns: Number of samples placed in output buffer
|
|
//-----------------------------------------------------------------------------
|
|
/*
|
|
static long RllDecodeMonoToMono(unsigned char *from,short *to,unsigned int size,char signedOutput ,unsigned short flag)
|
|
{
|
|
unsigned int z;
|
|
int prev;
|
|
|
|
if (signedOutput)
|
|
prev = flag - 0x8000;
|
|
else
|
|
prev = flag;
|
|
|
|
for (z=0;z<size;z++) {
|
|
prev = to[z] = (short)(prev + cin.sqrTable[from[z]]);
|
|
}
|
|
return size; //*sizeof(short));
|
|
}
|
|
*/
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// RllDecodeMonoToStereo
|
|
//
|
|
// Decode mono source data into a stereo buffer. Output is 4 times the number
|
|
// of bytes in the input.
|
|
//
|
|
// Parameters: from -> buffer holding encoded data
|
|
// to -> buffer to hold decoded data
|
|
// size = number of bytes of input (= 1/4 # of bytes of output)
|
|
// signedOutput = 0 for unsigned output, non-zero for signed output
|
|
// flag = flags from asset header
|
|
//
|
|
// Returns: Number of samples placed in output buffer
|
|
//-----------------------------------------------------------------------------
|
|
static long RllDecodeMonoToStereo(unsigned char *from,short *to,unsigned int size,char signedOutput,unsigned short flag)
|
|
{
|
|
unsigned int z;
|
|
int prev;
|
|
|
|
if (signedOutput)
|
|
prev = flag - 0x8000;
|
|
else
|
|
prev = flag;
|
|
|
|
for (z = 0; z < size; z++) {
|
|
prev = (short)(prev + cin.sqrTable[from[z]]);
|
|
to[z*2+0] = to[z*2+1] = (short)(prev);
|
|
}
|
|
|
|
return size; // * 2 * sizeof(short));
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// RllDecodeStereoToStereo
|
|
//
|
|
// Decode stereo source data into a stereo buffer.
|
|
//
|
|
// Parameters: from -> buffer holding encoded data
|
|
// to -> buffer to hold decoded data
|
|
// size = number of bytes of input (= 1/2 # of bytes of output)
|
|
// signedOutput = 0 for unsigned output, non-zero for signed output
|
|
// flag = flags from asset header
|
|
//
|
|
// Returns: Number of samples placed in output buffer
|
|
//-----------------------------------------------------------------------------
|
|
static long RllDecodeStereoToStereo(unsigned char *from,short *to,unsigned int size,char signedOutput, unsigned short flag)
|
|
{
|
|
unsigned int z;
|
|
unsigned char *zz = from;
|
|
int prevL, prevR;
|
|
|
|
if (signedOutput) {
|
|
prevL = (flag & 0xff00) - 0x8000;
|
|
prevR = ((flag & 0x00ff) << 8) - 0x8000;
|
|
} else {
|
|
prevL = flag & 0xff00;
|
|
prevR = (flag & 0x00ff) << 8;
|
|
}
|
|
|
|
for (z=0;z<size;z+=2) {
|
|
prevL = (short)(prevL + cin.sqrTable[*zz++]);
|
|
prevR = (short)(prevR + cin.sqrTable[*zz++]);
|
|
to[z+0] = (short)(prevL);
|
|
to[z+1] = (short)(prevR);
|
|
}
|
|
|
|
return (size>>1); //*sizeof(short));
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// RllDecodeStereoToMono
|
|
//
|
|
// Decode stereo source data into a mono buffer.
|
|
//
|
|
// Parameters: from -> buffer holding encoded data
|
|
// to -> buffer to hold decoded data
|
|
// size = number of bytes of input (= # of bytes of output)
|
|
// signedOutput = 0 for unsigned output, non-zero for signed output
|
|
// flag = flags from asset header
|
|
//
|
|
// Returns: Number of samples placed in output buffer
|
|
//-----------------------------------------------------------------------------
|
|
/*
|
|
static long RllDecodeStereoToMono(unsigned char *from,short *to,unsigned int size,char signedOutput, unsigned short flag)
|
|
{
|
|
unsigned int z;
|
|
int prevL,prevR;
|
|
|
|
if (signedOutput) {
|
|
prevL = (flag & 0xff00) - 0x8000;
|
|
prevR = ((flag & 0x00ff) << 8) -0x8000;
|
|
} else {
|
|
prevL = flag & 0xff00;
|
|
prevR = (flag & 0x00ff) << 8;
|
|
}
|
|
|
|
for (z=0;z<size;z+=1) {
|
|
prevL= prevL + cin.sqrTable[from[z*2]];
|
|
prevR = prevR + cin.sqrTable[from[z*2+1]];
|
|
to[z] = (short)((prevL + prevR)/2);
|
|
}
|
|
|
|
return size;
|
|
}
|
|
*/
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void move8_32( byte *src, byte *dst, int spl )
|
|
{
|
|
int i;
|
|
|
|
for(i = 0; i < 8; ++i)
|
|
{
|
|
memcpy(dst, src, 32);
|
|
src += spl;
|
|
dst += spl;
|
|
}
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void move4_32( byte *src, byte *dst, int spl )
|
|
{
|
|
int i;
|
|
|
|
for(i = 0; i < 4; ++i)
|
|
{
|
|
memcpy(dst, src, 16);
|
|
src += spl;
|
|
dst += spl;
|
|
}
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void blit8_32( byte *src, byte *dst, int spl )
|
|
{
|
|
int i;
|
|
|
|
for(i = 0; i < 8; ++i)
|
|
{
|
|
memcpy(dst, src, 32);
|
|
src += 32;
|
|
dst += spl;
|
|
}
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
static void blit4_32( byte *src, byte *dst, int spl )
|
|
{
|
|
int i;
|
|
|
|
for(i = 0; i < 4; ++i)
|
|
{
|
|
memmove(dst, src, 16);
|
|
src += 16;
|
|
dst += spl;
|
|
}
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void blit2_32( byte *src, byte *dst, int spl )
|
|
{
|
|
memcpy(dst, src, 8);
|
|
memcpy(dst+spl, src+8, 8);
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void blitVQQuad32fs( byte **status, unsigned char *data )
|
|
{
|
|
unsigned short newd, celdata, code;
|
|
unsigned int index, i;
|
|
int spl;
|
|
|
|
newd = 0;
|
|
celdata = 0;
|
|
index = 0;
|
|
|
|
spl = cinTable[currentHandle].samplesPerLine;
|
|
|
|
do {
|
|
if (!newd) {
|
|
newd = 7;
|
|
celdata = data[0] + data[1]*256;
|
|
data += 2;
|
|
} else {
|
|
newd--;
|
|
}
|
|
|
|
code = (unsigned short)(celdata&0xc000);
|
|
celdata <<= 2;
|
|
|
|
switch (code) {
|
|
case 0x8000: // vq code
|
|
blit8_32( (byte *)&vq8[(*data)*128], status[index], spl );
|
|
data++;
|
|
index += 5;
|
|
break;
|
|
case 0xc000: // drop
|
|
index++; // skip 8x8
|
|
for(i=0;i<4;i++) {
|
|
if (!newd) {
|
|
newd = 7;
|
|
celdata = data[0] + data[1]*256;
|
|
data += 2;
|
|
} else {
|
|
newd--;
|
|
}
|
|
|
|
code = (unsigned short)(celdata&0xc000); celdata <<= 2;
|
|
|
|
switch (code) { // code in top two bits of code
|
|
case 0x8000: // 4x4 vq code
|
|
blit4_32( (byte *)&vq4[(*data)*32], status[index], spl );
|
|
data++;
|
|
break;
|
|
case 0xc000: // 2x2 vq code
|
|
blit2_32( (byte *)&vq2[(*data)*8], status[index], spl );
|
|
data++;
|
|
blit2_32( (byte *)&vq2[(*data)*8], status[index]+8, spl );
|
|
data++;
|
|
blit2_32( (byte *)&vq2[(*data)*8], status[index]+spl*2, spl );
|
|
data++;
|
|
blit2_32( (byte *)&vq2[(*data)*8], status[index]+spl*2+8, spl );
|
|
data++;
|
|
break;
|
|
case 0x4000: // motion compensation
|
|
move4_32( status[index] + cin.mcomp[(*data)], status[index], spl );
|
|
data++;
|
|
break;
|
|
}
|
|
index++;
|
|
}
|
|
break;
|
|
case 0x4000: // motion compensation
|
|
move8_32( status[index] + cin.mcomp[(*data)], status[index], spl );
|
|
data++;
|
|
index += 5;
|
|
break;
|
|
case 0x0000:
|
|
index += 5;
|
|
break;
|
|
}
|
|
} while ( status[index] != NULL );
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void ROQ_GenYUVTables( void )
|
|
{
|
|
float t_ub,t_vr,t_ug,t_vg;
|
|
long i;
|
|
|
|
t_ub = (1.77200f/2.0f) * (float)(1<<6) + 0.5f;
|
|
t_vr = (1.40200f/2.0f) * (float)(1<<6) + 0.5f;
|
|
t_ug = (0.34414f/2.0f) * (float)(1<<6) + 0.5f;
|
|
t_vg = (0.71414f/2.0f) * (float)(1<<6) + 0.5f;
|
|
for(i=0;i<256;i++) {
|
|
float x = (float)(2 * i - 255);
|
|
|
|
ROQ_UB_tab[i] = (long)( ( t_ub * x) + (1<<5));
|
|
ROQ_VR_tab[i] = (long)( ( t_vr * x) + (1<<5));
|
|
ROQ_UG_tab[i] = (long)( (-t_ug * x) );
|
|
ROQ_VG_tab[i] = (long)( (-t_vg * x) + (1<<5));
|
|
ROQ_YY_tab[i] = (long)( (i << 6) | (i >> 2) );
|
|
}
|
|
}
|
|
|
|
#define VQ2TO4(a,b,c,d) { \
|
|
*c++ = a[0]; \
|
|
*d++ = a[0]; \
|
|
*d++ = a[0]; \
|
|
*c++ = a[1]; \
|
|
*d++ = a[1]; \
|
|
*d++ = a[1]; \
|
|
*c++ = b[0]; \
|
|
*d++ = b[0]; \
|
|
*d++ = b[0]; \
|
|
*c++ = b[1]; \
|
|
*d++ = b[1]; \
|
|
*d++ = b[1]; \
|
|
*d++ = a[0]; \
|
|
*d++ = a[0]; \
|
|
*d++ = a[1]; \
|
|
*d++ = a[1]; \
|
|
*d++ = b[0]; \
|
|
*d++ = b[0]; \
|
|
*d++ = b[1]; \
|
|
*d++ = b[1]; \
|
|
a += 2; b += 2; }
|
|
|
|
#define VQ2TO2(a,b,c,d) { \
|
|
*c++ = *a; \
|
|
*d++ = *a; \
|
|
*d++ = *a; \
|
|
*c++ = *b; \
|
|
*d++ = *b; \
|
|
*d++ = *b; \
|
|
*d++ = *a; \
|
|
*d++ = *a; \
|
|
*d++ = *b; \
|
|
*d++ = *b; \
|
|
a++; b++; }
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static unsigned short yuv_to_rgb( long y, long u, long v )
|
|
{
|
|
long r,g,b,YY = (long)(ROQ_YY_tab[(y)]);
|
|
|
|
r = (YY + ROQ_VR_tab[v]) >> 9;
|
|
g = (YY + ROQ_UG_tab[u] + ROQ_VG_tab[v]) >> 8;
|
|
b = (YY + ROQ_UB_tab[u]) >> 9;
|
|
|
|
if (r<0)
|
|
r = 0;
|
|
if (g<0)
|
|
g = 0;
|
|
if (b<0)
|
|
b = 0;
|
|
if (r > 31)
|
|
r = 31;
|
|
if (g > 63)
|
|
g = 63;
|
|
if (b > 31)
|
|
b = 31;
|
|
|
|
return (unsigned short)((r<<11)+(g<<5)+(b));
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static unsigned int yuv_to_rgb24( long y, long u, long v )
|
|
{
|
|
long r,g,b,YY = (long)(ROQ_YY_tab[(y)]);
|
|
|
|
r = (YY + ROQ_VR_tab[v]) >> 6;
|
|
g = (YY + ROQ_UG_tab[u] + ROQ_VG_tab[v]) >> 6;
|
|
b = (YY + ROQ_UB_tab[u]) >> 6;
|
|
|
|
if (r<0)
|
|
r = 0;
|
|
if (g<0)
|
|
g = 0;
|
|
if (b<0)
|
|
b = 0;
|
|
if (r > 255)
|
|
r = 255;
|
|
if (g > 255)
|
|
g = 255;
|
|
if (b > 255)
|
|
b = 255;
|
|
|
|
return LittleLong ((r)|(g<<8)|(b<<16)|(255<<24));
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void decodeCodeBook( byte *input, unsigned short roq_flags )
|
|
{
|
|
long i, j, two, four;
|
|
unsigned short *aptr, *bptr, *cptr, *dptr;
|
|
long y0,y1,y2,y3,cr,cb;
|
|
byte *bbptr, *baptr, *bcptr, *bdptr;
|
|
union {
|
|
unsigned int *i;
|
|
unsigned short *s;
|
|
} iaptr, ibptr, icptr, idptr;
|
|
|
|
if (!roq_flags) {
|
|
two = four = 256;
|
|
} else {
|
|
two = roq_flags>>8;
|
|
if (!two) two = 256;
|
|
four = roq_flags&0xff;
|
|
}
|
|
|
|
four *= 2;
|
|
|
|
bptr = (unsigned short *)vq2;
|
|
|
|
if (!cinTable[currentHandle].half) {
|
|
if (!cinTable[currentHandle].smootheddouble) {
|
|
//
|
|
// normal height
|
|
//
|
|
if (cinTable[currentHandle].samplesPerPixel==2) {
|
|
for(i=0;i<two;i++) {
|
|
y0 = (long)*input++;
|
|
y1 = (long)*input++;
|
|
y2 = (long)*input++;
|
|
y3 = (long)*input++;
|
|
cr = (long)*input++;
|
|
cb = (long)*input++;
|
|
*bptr++ = yuv_to_rgb( y0, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y1, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y2, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y3, cr, cb );
|
|
}
|
|
|
|
cptr = (unsigned short *)vq4;
|
|
dptr = (unsigned short *)vq8;
|
|
|
|
for(i=0;i<four;i++) {
|
|
aptr = (unsigned short *)vq2 + (*input++)*4;
|
|
bptr = (unsigned short *)vq2 + (*input++)*4;
|
|
for(j=0;j<2;j++)
|
|
VQ2TO4(aptr,bptr,cptr,dptr);
|
|
}
|
|
} else if (cinTable[currentHandle].samplesPerPixel==4) {
|
|
ibptr.s = bptr;
|
|
for(i=0;i<two;i++) {
|
|
y0 = (long)*input++;
|
|
y1 = (long)*input++;
|
|
y2 = (long)*input++;
|
|
y3 = (long)*input++;
|
|
cr = (long)*input++;
|
|
cb = (long)*input++;
|
|
*ibptr.i++ = yuv_to_rgb24( y0, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( y1, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( y2, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( y3, cr, cb );
|
|
}
|
|
|
|
icptr.s = vq4;
|
|
idptr.s = vq8;
|
|
|
|
for(i=0;i<four;i++) {
|
|
iaptr.s = vq2;
|
|
iaptr.i += (*input++)*4;
|
|
ibptr.s = vq2;
|
|
ibptr.i += (*input++)*4;
|
|
for(j=0;j<2;j++)
|
|
VQ2TO4(iaptr.i, ibptr.i, icptr.i, idptr.i);
|
|
}
|
|
} else if (cinTable[currentHandle].samplesPerPixel==1) {
|
|
bbptr = (byte *)bptr;
|
|
for(i=0;i<two;i++) {
|
|
*bbptr++ = cinTable[currentHandle].gray[*input++];
|
|
*bbptr++ = cinTable[currentHandle].gray[*input++];
|
|
*bbptr++ = cinTable[currentHandle].gray[*input++];
|
|
*bbptr++ = cinTable[currentHandle].gray[*input]; input +=3;
|
|
}
|
|
|
|
bcptr = (byte *)vq4;
|
|
bdptr = (byte *)vq8;
|
|
|
|
for(i=0;i<four;i++) {
|
|
baptr = (byte *)vq2 + (*input++)*4;
|
|
bbptr = (byte *)vq2 + (*input++)*4;
|
|
for(j=0;j<2;j++)
|
|
VQ2TO4(baptr,bbptr,bcptr,bdptr);
|
|
}
|
|
}
|
|
} else {
|
|
//
|
|
// double height, smoothed
|
|
//
|
|
if (cinTable[currentHandle].samplesPerPixel==2) {
|
|
for(i=0;i<two;i++) {
|
|
y0 = (long)*input++;
|
|
y1 = (long)*input++;
|
|
y2 = (long)*input++;
|
|
y3 = (long)*input++;
|
|
cr = (long)*input++;
|
|
cb = (long)*input++;
|
|
*bptr++ = yuv_to_rgb( y0, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y1, cr, cb );
|
|
*bptr++ = yuv_to_rgb( ((y0*3)+y2)/4, cr, cb );
|
|
*bptr++ = yuv_to_rgb( ((y1*3)+y3)/4, cr, cb );
|
|
*bptr++ = yuv_to_rgb( (y0+(y2*3))/4, cr, cb );
|
|
*bptr++ = yuv_to_rgb( (y1+(y3*3))/4, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y2, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y3, cr, cb );
|
|
}
|
|
|
|
cptr = (unsigned short *)vq4;
|
|
dptr = (unsigned short *)vq8;
|
|
|
|
for(i=0;i<four;i++) {
|
|
aptr = (unsigned short *)vq2 + (*input++)*8;
|
|
bptr = (unsigned short *)vq2 + (*input++)*8;
|
|
for(j=0;j<2;j++) {
|
|
VQ2TO4(aptr,bptr,cptr,dptr);
|
|
VQ2TO4(aptr,bptr,cptr,dptr);
|
|
}
|
|
}
|
|
} else if (cinTable[currentHandle].samplesPerPixel==4) {
|
|
ibptr.s = bptr;
|
|
for(i=0;i<two;i++) {
|
|
y0 = (long)*input++;
|
|
y1 = (long)*input++;
|
|
y2 = (long)*input++;
|
|
y3 = (long)*input++;
|
|
cr = (long)*input++;
|
|
cb = (long)*input++;
|
|
*ibptr.i++ = yuv_to_rgb24( y0, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( y1, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( ((y0*3)+y2)/4, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( ((y1*3)+y3)/4, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( (y0+(y2*3))/4, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( (y1+(y3*3))/4, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( y2, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( y3, cr, cb );
|
|
}
|
|
|
|
icptr.s = vq4;
|
|
idptr.s = vq8;
|
|
|
|
for(i=0;i<four;i++) {
|
|
iaptr.s = vq2;
|
|
iaptr.i += (*input++)*8;
|
|
ibptr.s = vq2;
|
|
ibptr.i += (*input++)*8;
|
|
for(j=0;j<2;j++) {
|
|
VQ2TO4(iaptr.i, ibptr.i, icptr.i, idptr.i);
|
|
VQ2TO4(iaptr.i, ibptr.i, icptr.i, idptr.i);
|
|
}
|
|
}
|
|
} else if (cinTable[currentHandle].samplesPerPixel==1) {
|
|
bbptr = (byte *)bptr;
|
|
for(i=0;i<two;i++) {
|
|
y0 = (long)*input++;
|
|
y1 = (long)*input++;
|
|
y2 = (long)*input++;
|
|
y3 = (long)*input; input+= 3;
|
|
*bbptr++ = cinTable[currentHandle].gray[y0];
|
|
*bbptr++ = cinTable[currentHandle].gray[y1];
|
|
*bbptr++ = cinTable[currentHandle].gray[((y0*3)+y2)/4];
|
|
*bbptr++ = cinTable[currentHandle].gray[((y1*3)+y3)/4];
|
|
*bbptr++ = cinTable[currentHandle].gray[(y0+(y2*3))/4];
|
|
*bbptr++ = cinTable[currentHandle].gray[(y1+(y3*3))/4];
|
|
*bbptr++ = cinTable[currentHandle].gray[y2];
|
|
*bbptr++ = cinTable[currentHandle].gray[y3];
|
|
}
|
|
|
|
bcptr = (byte *)vq4;
|
|
bdptr = (byte *)vq8;
|
|
|
|
for(i=0;i<four;i++) {
|
|
baptr = (byte *)vq2 + (*input++)*8;
|
|
bbptr = (byte *)vq2 + (*input++)*8;
|
|
for(j=0;j<2;j++) {
|
|
VQ2TO4(baptr,bbptr,bcptr,bdptr);
|
|
VQ2TO4(baptr,bbptr,bcptr,bdptr);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
//
|
|
// 1/4 screen
|
|
//
|
|
if (cinTable[currentHandle].samplesPerPixel==2) {
|
|
for(i=0;i<two;i++) {
|
|
y0 = (long)*input; input+=2;
|
|
y2 = (long)*input; input+=2;
|
|
cr = (long)*input++;
|
|
cb = (long)*input++;
|
|
*bptr++ = yuv_to_rgb( y0, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y2, cr, cb );
|
|
}
|
|
|
|
cptr = (unsigned short *)vq4;
|
|
dptr = (unsigned short *)vq8;
|
|
|
|
for(i=0;i<four;i++) {
|
|
aptr = (unsigned short *)vq2 + (*input++)*2;
|
|
bptr = (unsigned short *)vq2 + (*input++)*2;
|
|
for(j=0;j<2;j++) {
|
|
VQ2TO2(aptr,bptr,cptr,dptr);
|
|
}
|
|
}
|
|
} else if (cinTable[currentHandle].samplesPerPixel == 1) {
|
|
bbptr = (byte *)bptr;
|
|
|
|
for(i=0;i<two;i++) {
|
|
*bbptr++ = cinTable[currentHandle].gray[*input]; input+=2;
|
|
*bbptr++ = cinTable[currentHandle].gray[*input]; input+=4;
|
|
}
|
|
|
|
bcptr = (byte *)vq4;
|
|
bdptr = (byte *)vq8;
|
|
|
|
for(i=0;i<four;i++) {
|
|
baptr = (byte *)vq2 + (*input++)*2;
|
|
bbptr = (byte *)vq2 + (*input++)*2;
|
|
for(j=0;j<2;j++) {
|
|
VQ2TO2(baptr,bbptr,bcptr,bdptr);
|
|
}
|
|
}
|
|
} else if (cinTable[currentHandle].samplesPerPixel == 4) {
|
|
ibptr.s = bptr;
|
|
for(i=0;i<two;i++) {
|
|
y0 = (long)*input; input+=2;
|
|
y2 = (long)*input; input+=2;
|
|
cr = (long)*input++;
|
|
cb = (long)*input++;
|
|
*ibptr.i++ = yuv_to_rgb24( y0, cr, cb );
|
|
*ibptr.i++ = yuv_to_rgb24( y2, cr, cb );
|
|
}
|
|
|
|
icptr.s = vq4;
|
|
idptr.s = vq8;
|
|
|
|
for(i=0;i<four;i++) {
|
|
iaptr.s = vq2;
|
|
iaptr.i += (*input++)*2;
|
|
ibptr.s = vq2 + (*input++)*2;
|
|
ibptr.i += (*input++)*2;
|
|
for(j=0;j<2;j++) {
|
|
VQ2TO2(iaptr.i,ibptr.i,icptr.i,idptr.i);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void recurseQuad( long startX, long startY, long quadSize, long xOff, long yOff )
|
|
{
|
|
byte *scroff;
|
|
long bigx, bigy, lowx, lowy, useY;
|
|
long offset;
|
|
|
|
offset = cinTable[currentHandle].screenDelta;
|
|
|
|
lowx = lowy = 0;
|
|
bigx = cinTable[currentHandle].xsize;
|
|
bigy = cinTable[currentHandle].ysize;
|
|
|
|
if (bigx > cinTable[currentHandle].CIN_WIDTH) bigx = cinTable[currentHandle].CIN_WIDTH;
|
|
if (bigy > cinTable[currentHandle].CIN_HEIGHT) bigy = cinTable[currentHandle].CIN_HEIGHT;
|
|
|
|
if ( (startX >= lowx) && (startX+quadSize) <= (bigx) && (startY+quadSize) <= (bigy) && (startY >= lowy) && quadSize <= MAXSIZE) {
|
|
useY = startY;
|
|
scroff = cin.linbuf + (useY+((cinTable[currentHandle].CIN_HEIGHT-bigy)>>1)+yOff)*(cinTable[currentHandle].samplesPerLine) + (((startX+xOff))*cinTable[currentHandle].samplesPerPixel);
|
|
|
|
cin.qStatus[0][cinTable[currentHandle].onQuad ] = scroff;
|
|
cin.qStatus[1][cinTable[currentHandle].onQuad++] = scroff+offset;
|
|
}
|
|
|
|
if ( quadSize != MINSIZE ) {
|
|
quadSize >>= 1;
|
|
recurseQuad( startX, startY , quadSize, xOff, yOff );
|
|
recurseQuad( startX+quadSize, startY , quadSize, xOff, yOff );
|
|
recurseQuad( startX, startY+quadSize , quadSize, xOff, yOff );
|
|
recurseQuad( startX+quadSize, startY+quadSize , quadSize, xOff, yOff );
|
|
}
|
|
}
|
|
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void setupQuad( long xOff, long yOff )
|
|
{
|
|
long numQuadCels, i,x,y;
|
|
byte *temp;
|
|
|
|
if (xOff == cin.oldXOff && yOff == cin.oldYOff && cinTable[currentHandle].ysize == (unsigned)cin.oldysize && cinTable[currentHandle].xsize == (unsigned)cin.oldxsize) {
|
|
return;
|
|
}
|
|
|
|
cin.oldXOff = xOff;
|
|
cin.oldYOff = yOff;
|
|
cin.oldysize = cinTable[currentHandle].ysize;
|
|
cin.oldxsize = cinTable[currentHandle].xsize;
|
|
/* Enisform: Not in q3 source
|
|
numQuadCels = (cinTable[currentHandle].CIN_WIDTH*cinTable[currentHandle].CIN_HEIGHT) / (16);
|
|
numQuadCels += numQuadCels/4 + numQuadCels/16;
|
|
numQuadCels += 64; // for overflow
|
|
*/
|
|
|
|
numQuadCels = (cinTable[currentHandle].xsize*cinTable[currentHandle].ysize) / (16);
|
|
numQuadCels += numQuadCels/4;
|
|
numQuadCels += 64; // for overflow
|
|
|
|
cinTable[currentHandle].onQuad = 0;
|
|
|
|
for(y=0;y<(long)cinTable[currentHandle].ysize;y+=16)
|
|
for(x=0;x<(long)cinTable[currentHandle].xsize;x+=16)
|
|
recurseQuad( x, y, 16, xOff, yOff );
|
|
|
|
temp = NULL;
|
|
|
|
for(i=(numQuadCels-64);i<numQuadCels;i++) {
|
|
cin.qStatus[0][i] = temp; // eoq
|
|
cin.qStatus[1][i] = temp; // eoq
|
|
}
|
|
}
|
|
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void readQuadInfo( byte *qData )
|
|
{
|
|
if (currentHandle < 0) return;
|
|
|
|
cinTable[currentHandle].xsize = qData[0]+qData[1]*256;
|
|
cinTable[currentHandle].ysize = qData[2]+qData[3]*256;
|
|
cinTable[currentHandle].maxsize = qData[4]+qData[5]*256;
|
|
cinTable[currentHandle].minsize = qData[6]+qData[7]*256;
|
|
|
|
cinTable[currentHandle].CIN_HEIGHT = cinTable[currentHandle].ysize;
|
|
cinTable[currentHandle].CIN_WIDTH = cinTable[currentHandle].xsize;
|
|
|
|
cinTable[currentHandle].samplesPerLine = cinTable[currentHandle].CIN_WIDTH*cinTable[currentHandle].samplesPerPixel;
|
|
cinTable[currentHandle].screenDelta = cinTable[currentHandle].CIN_HEIGHT*cinTable[currentHandle].samplesPerLine;
|
|
|
|
cinTable[currentHandle].half = qfalse;
|
|
cinTable[currentHandle].smootheddouble = qfalse;
|
|
|
|
cinTable[currentHandle].VQ0 = cinTable[currentHandle].VQNormal;
|
|
cinTable[currentHandle].VQ1 = cinTable[currentHandle].VQBuffer;
|
|
|
|
cinTable[currentHandle].t[0] = cinTable[currentHandle].screenDelta;
|
|
cinTable[currentHandle].t[1] = -cinTable[currentHandle].screenDelta;
|
|
|
|
cinTable[currentHandle].drawX = cinTable[currentHandle].CIN_WIDTH;
|
|
cinTable[currentHandle].drawY = cinTable[currentHandle].CIN_HEIGHT;
|
|
// jic the card sucks
|
|
if ( cls.glconfig.maxTextureSize <= 256) {
|
|
if (cinTable[currentHandle].drawX>256) {
|
|
cinTable[currentHandle].drawX = 256;
|
|
}
|
|
if (cinTable[currentHandle].drawY>256) {
|
|
cinTable[currentHandle].drawY = 256;
|
|
}
|
|
if (cinTable[currentHandle].CIN_WIDTH != 256 || cinTable[currentHandle].CIN_HEIGHT != 256) {
|
|
Com_Printf("HACK: approxmimating cinematic for Rage Pro or Voodoo\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void RoQPrepMcomp( long xoff, long yoff )
|
|
{
|
|
long i, j, x, y, temp, temp2;
|
|
|
|
i=cinTable[currentHandle].samplesPerLine; j=cinTable[currentHandle].samplesPerPixel;
|
|
if ( cinTable[currentHandle].xsize == (cinTable[currentHandle].ysize*4) && !cinTable[currentHandle].half ) { j = j+j; i = i+i; }
|
|
|
|
for(y=0;y<16;y++) {
|
|
temp2 = (y+yoff-8)*i;
|
|
for(x=0;x<16;x++) {
|
|
temp = (x+xoff-8)*j;
|
|
cin.mcomp[(x*16)+y] = cinTable[currentHandle].normalBuffer0-(temp2+temp);
|
|
}
|
|
}
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void initRoQ( void )
|
|
{
|
|
if (currentHandle < 0) return;
|
|
|
|
cinTable[currentHandle].VQNormal = (void (*)(byte *, void *))blitVQQuad32fs;
|
|
cinTable[currentHandle].VQBuffer = (void (*)(byte *, void *))blitVQQuad32fs;
|
|
cinTable[currentHandle].samplesPerPixel = 4;
|
|
ROQ_GenYUVTables();
|
|
RllSetupTable();
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
/*
|
|
static byte* RoQFetchInterlaced( byte *source ) {
|
|
int x, *src, *dst;
|
|
|
|
if (currentHandle < 0) return NULL;
|
|
|
|
src = (int *)source;
|
|
dst = (int *)cinTable[currentHandle].buf2;
|
|
|
|
for(x=0;x<256*256;x++) {
|
|
*dst = *src;
|
|
dst++; src += 2;
|
|
}
|
|
return cinTable[currentHandle].buf2;
|
|
}
|
|
*/
|
|
static void RoQReset( void ) {
|
|
|
|
if (currentHandle < 0) return;
|
|
|
|
FS_FCloseFile( cinTable[currentHandle].iFile );
|
|
FS_FOpenFileRead (cinTable[currentHandle].fileName, &cinTable[currentHandle].iFile, qtrue);
|
|
// let the background thread start reading ahead
|
|
FS_Read (cin.file, 16, cinTable[currentHandle].iFile);
|
|
RoQ_init();
|
|
cinTable[currentHandle].status = FMV_LOOPED;
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void RoQInterrupt(void)
|
|
{
|
|
byte *framedata;
|
|
short sbuf[32768];
|
|
int ssize;
|
|
|
|
if (currentHandle < 0) return;
|
|
|
|
FS_Read( cin.file, cinTable[currentHandle].RoQFrameSize+8, cinTable[currentHandle].iFile );
|
|
if ( cinTable[currentHandle].RoQPlayed >= cinTable[currentHandle].ROQSize ) {
|
|
if (cinTable[currentHandle].holdAtEnd==qfalse) {
|
|
if (cinTable[currentHandle].looping) {
|
|
RoQReset();
|
|
} else {
|
|
cinTable[currentHandle].status = FMV_EOF;
|
|
}
|
|
} else {
|
|
cinTable[currentHandle].status = FMV_IDLE;
|
|
}
|
|
return;
|
|
}
|
|
|
|
framedata = cin.file;
|
|
//
|
|
// new frame is ready
|
|
//
|
|
redump:
|
|
switch(cinTable[currentHandle].roq_id)
|
|
{
|
|
case ROQ_QUAD_VQ:
|
|
if ((cinTable[currentHandle].numQuads&1)) {
|
|
cinTable[currentHandle].normalBuffer0 = cinTable[currentHandle].t[1];
|
|
RoQPrepMcomp( cinTable[currentHandle].roqF0, cinTable[currentHandle].roqF1 );
|
|
cinTable[currentHandle].VQ1( (byte *)cin.qStatus[1], framedata);
|
|
cinTable[currentHandle].buf = cin.linbuf + cinTable[currentHandle].screenDelta;
|
|
} else {
|
|
cinTable[currentHandle].normalBuffer0 = cinTable[currentHandle].t[0];
|
|
RoQPrepMcomp( cinTable[currentHandle].roqF0, cinTable[currentHandle].roqF1 );
|
|
cinTable[currentHandle].VQ0( (byte *)cin.qStatus[0], framedata );
|
|
cinTable[currentHandle].buf = cin.linbuf;
|
|
}
|
|
if (cinTable[currentHandle].numQuads == 0) { // first frame
|
|
Com_Memcpy(cin.linbuf+cinTable[currentHandle].screenDelta, cin.linbuf, cinTable[currentHandle].samplesPerLine*cinTable[currentHandle].ysize);
|
|
}
|
|
cinTable[currentHandle].numQuads++;
|
|
cinTable[currentHandle].dirty = qtrue;
|
|
break;
|
|
case ROQ_CODEBOOK:
|
|
decodeCodeBook( framedata, (unsigned short)cinTable[currentHandle].roq_flags );
|
|
break;
|
|
case ZA_SOUND_MONO:
|
|
if (!cinTable[currentHandle].silent) {
|
|
ssize = RllDecodeMonoToStereo( framedata, sbuf, cinTable[currentHandle].RoQFrameSize, 0, (unsigned short)cinTable[currentHandle].roq_flags);
|
|
S_RawSamples( ssize, 22050, 2, 1, (byte *)sbuf, s_volume->value, qtrue );
|
|
}
|
|
break;
|
|
case ZA_SOUND_STEREO:
|
|
if (!cinTable[currentHandle].silent) {
|
|
if (cinTable[currentHandle].numQuads == -1) {
|
|
S_Update();
|
|
s_rawend = s_soundtime;
|
|
}
|
|
ssize = RllDecodeStereoToStereo( framedata, sbuf, cinTable[currentHandle].RoQFrameSize, 0, (unsigned short)cinTable[currentHandle].roq_flags);
|
|
S_RawSamples( ssize, 22050, 2, 2, (byte *)sbuf, s_volume->value, qtrue );
|
|
}
|
|
break;
|
|
case ROQ_QUAD_INFO:
|
|
if (cinTable[currentHandle].numQuads == -1) {
|
|
readQuadInfo( framedata );
|
|
setupQuad( 0, 0 );
|
|
cinTable[currentHandle].startTime = cinTable[currentHandle].lastTime = Sys_Milliseconds()*com_timescale->value;
|
|
}
|
|
if (cinTable[currentHandle].numQuads != 1) cinTable[currentHandle].numQuads = 0;
|
|
break;
|
|
case ROQ_PACKET:
|
|
cinTable[currentHandle].inMemory = (qboolean)cinTable[currentHandle].roq_flags;
|
|
cinTable[currentHandle].RoQFrameSize = 0; // for header
|
|
break;
|
|
case ROQ_QUAD_HANG:
|
|
cinTable[currentHandle].RoQFrameSize = 0;
|
|
break;
|
|
case ROQ_QUAD_JPEG:
|
|
break;
|
|
default:
|
|
cinTable[currentHandle].status = FMV_EOF;
|
|
break;
|
|
}
|
|
//
|
|
// read in next frame data
|
|
//
|
|
if ( cinTable[currentHandle].RoQPlayed >= cinTable[currentHandle].ROQSize ) {
|
|
if (cinTable[currentHandle].holdAtEnd==qfalse) {
|
|
if (cinTable[currentHandle].looping) {
|
|
RoQReset();
|
|
} else {
|
|
cinTable[currentHandle].status = FMV_EOF;
|
|
}
|
|
} else {
|
|
cinTable[currentHandle].status = FMV_IDLE;
|
|
}
|
|
return;
|
|
}
|
|
|
|
framedata += cinTable[currentHandle].RoQFrameSize;
|
|
cinTable[currentHandle].roq_id = framedata[0] + framedata[1]*256;
|
|
cinTable[currentHandle].RoQFrameSize = framedata[2] + framedata[3]*256 + framedata[4]*65536;
|
|
cinTable[currentHandle].roq_flags = framedata[6] + framedata[7]*256;
|
|
cinTable[currentHandle].roqF0 = (signed char)framedata[7];
|
|
cinTable[currentHandle].roqF1 = (signed char)framedata[6];
|
|
|
|
if (cinTable[currentHandle].RoQFrameSize>65536||cinTable[currentHandle].roq_id==0x1084) {
|
|
Com_DPrintf("roq_size>65536||roq_id==0x1084\n");
|
|
cinTable[currentHandle].status = FMV_EOF;
|
|
if (cinTable[currentHandle].looping) {
|
|
RoQReset();
|
|
}
|
|
return;
|
|
}
|
|
if (cinTable[currentHandle].inMemory && (cinTable[currentHandle].status != FMV_EOF))
|
|
{
|
|
cinTable[currentHandle].inMemory = (qboolean)(((int)cinTable[currentHandle].inMemory)-1);
|
|
framedata += 8;
|
|
goto redump;
|
|
}
|
|
//
|
|
// one more frame hits the dust
|
|
//
|
|
// assert(cinTable[currentHandle].RoQFrameSize <= 65536);
|
|
// r = FS_Read( cin.file, cinTable[currentHandle].RoQFrameSize+8, cinTable[currentHandle].iFile );
|
|
cinTable[currentHandle].RoQPlayed += cinTable[currentHandle].RoQFrameSize+8;
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void RoQ_init( void )
|
|
{
|
|
|
|
cinTable[currentHandle].startTime = cinTable[currentHandle].lastTime = Sys_Milliseconds()*com_timescale->value;
|
|
|
|
cinTable[currentHandle].RoQPlayed = 24;
|
|
|
|
/* get frame rate */
|
|
cinTable[currentHandle].roqFPS = cin.file[ 6] + cin.file[ 7]*256;
|
|
|
|
if (!cinTable[currentHandle].roqFPS) cinTable[currentHandle].roqFPS = 30;
|
|
|
|
|
|
cinTable[currentHandle].numQuads = -1;
|
|
|
|
cinTable[currentHandle].roq_id = cin.file[ 8] + cin.file[ 9]*256;
|
|
cinTable[currentHandle].RoQFrameSize = cin.file[10] + cin.file[11]*256 + cin.file[12]*65536;
|
|
cinTable[currentHandle].roq_flags = cin.file[14] + cin.file[15]*256;
|
|
|
|
if (cinTable[currentHandle].RoQFrameSize > 65536 || !cinTable[currentHandle].RoQFrameSize) {
|
|
return;
|
|
}
|
|
|
|
if (cinTable[currentHandle].hSFX)
|
|
{
|
|
S_StartLocalSound(cinTable[currentHandle].hSFX, CHAN_AUTO);
|
|
}
|
|
}
|
|
|
|
/******************************************************************************
|
|
*
|
|
* Function:
|
|
*
|
|
* Description:
|
|
*
|
|
******************************************************************************/
|
|
|
|
static void RoQShutdown( void ) {
|
|
const char *s;
|
|
|
|
if (!cinTable[currentHandle].buf) {
|
|
if (cinTable[currentHandle].iFile) {
|
|
// assert( 0 && "ROQ handle leak-prevention WAS needed!");
|
|
FS_FCloseFile( cinTable[currentHandle].iFile );
|
|
cinTable[currentHandle].iFile = 0;
|
|
if (cinTable[currentHandle].hSFX) {
|
|
S_CIN_StopSound( cinTable[currentHandle].hSFX );
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (cinTable[currentHandle].status == FMV_IDLE) {
|
|
return;
|
|
}
|
|
|
|
Com_DPrintf("finished cinematic\n");
|
|
cinTable[currentHandle].status = FMV_IDLE;
|
|
|
|
if (cinTable[currentHandle].iFile) {
|
|
FS_FCloseFile( cinTable[currentHandle].iFile );
|
|
cinTable[currentHandle].iFile = 0;
|
|
if (cinTable[currentHandle].hSFX) {
|
|
S_CIN_StopSound( cinTable[currentHandle].hSFX );
|
|
}
|
|
}
|
|
|
|
if (cinTable[currentHandle].alterGameState) {
|
|
cls.state = CA_DISCONNECTED;
|
|
// we can't just do a vstr nextmap, because
|
|
// if we are aborting the intro cinematic with
|
|
// a devmap command, nextmap would be valid by
|
|
// the time it was referenced
|
|
s = Cvar_VariableString( "nextmap" );
|
|
if ( s[0] ) {
|
|
Cbuf_ExecuteText( EXEC_APPEND, va("%s\n", s) );
|
|
Cvar_Set( "nextmap", "" );
|
|
}
|
|
CL_handle = -1;
|
|
}
|
|
cinTable[currentHandle].fileName[0] = 0;
|
|
currentHandle = -1;
|
|
}
|
|
|
|
/*
|
|
==================
|
|
CIN_StopCinematic
|
|
==================
|
|
*/
|
|
|
|
e_status CIN_StopCinematic(int handle) {
|
|
|
|
if (handle < 0 || handle>= MAX_VIDEO_HANDLES || cinTable[handle].status == FMV_EOF) return FMV_EOF;
|
|
currentHandle = handle;
|
|
|
|
Com_DPrintf("trFMV::stop(), closing %s\n", cinTable[currentHandle].fileName);
|
|
|
|
if (!cinTable[currentHandle].buf) {
|
|
if (cinTable[currentHandle].iFile) {
|
|
// assert( 0 && "ROQ handle leak-prevention WAS needed!");
|
|
FS_FCloseFile( cinTable[currentHandle].iFile );
|
|
cinTable[currentHandle].iFile = 0;
|
|
cinTable[currentHandle].fileName[0] = 0;
|
|
if (cinTable[currentHandle].hSFX) {
|
|
S_CIN_StopSound( cinTable[currentHandle].hSFX );
|
|
}
|
|
}
|
|
return FMV_EOF;
|
|
}
|
|
|
|
if (cinTable[currentHandle].alterGameState) {
|
|
if ( cls.state != CA_CINEMATIC ) {
|
|
return cinTable[currentHandle].status;
|
|
}
|
|
}
|
|
cinTable[currentHandle].status = FMV_EOF;
|
|
RoQShutdown();
|
|
|
|
return FMV_EOF;
|
|
}
|
|
|
|
/*
|
|
==================
|
|
SCR_RunCinematic
|
|
|
|
Fetch and decompress the pending frame
|
|
==================
|
|
*/
|
|
|
|
|
|
e_status CIN_RunCinematic (int handle)
|
|
{
|
|
int start = 0;
|
|
int thisTime = 0;
|
|
|
|
if (handle < 0 || handle>= MAX_VIDEO_HANDLES || cinTable[handle].status == FMV_EOF) return FMV_EOF;
|
|
|
|
if (cin.currentHandle != handle) {
|
|
currentHandle = handle;
|
|
cin.currentHandle = currentHandle;
|
|
cinTable[currentHandle].status = FMV_EOF;
|
|
RoQReset();
|
|
}
|
|
|
|
if (cinTable[handle].playonwalls < -1)
|
|
{
|
|
return cinTable[handle].status;
|
|
}
|
|
|
|
currentHandle = handle;
|
|
|
|
if (cinTable[currentHandle].alterGameState) {
|
|
if ( cls.state != CA_CINEMATIC ) {
|
|
return cinTable[currentHandle].status;
|
|
}
|
|
}
|
|
|
|
if (cinTable[currentHandle].status == FMV_IDLE) {
|
|
return cinTable[currentHandle].status;
|
|
}
|
|
|
|
thisTime = Sys_Milliseconds()*com_timescale->value;
|
|
if (cinTable[currentHandle].shader && (abs(thisTime - (double)cinTable[currentHandle].lastTime))>100) {
|
|
cinTable[currentHandle].startTime += thisTime - cinTable[currentHandle].lastTime;
|
|
}
|
|
cinTable[currentHandle].tfps = ((((Sys_Milliseconds()*com_timescale->value) - cinTable[currentHandle].startTime)*cinTable[currentHandle].roqFPS)/1000);
|
|
|
|
start = cinTable[currentHandle].startTime;
|
|
while( (cinTable[currentHandle].tfps != cinTable[currentHandle].numQuads)
|
|
&& (cinTable[currentHandle].status == FMV_PLAY) )
|
|
{
|
|
RoQInterrupt();
|
|
if ((unsigned)start != cinTable[currentHandle].startTime) {
|
|
cinTable[currentHandle].tfps = ((((Sys_Milliseconds()*com_timescale->value)
|
|
- cinTable[currentHandle].startTime)*cinTable[currentHandle].roqFPS)/1000);
|
|
start = cinTable[currentHandle].startTime;
|
|
}
|
|
}
|
|
|
|
cinTable[currentHandle].lastTime = thisTime;
|
|
|
|
if (cinTable[currentHandle].status == FMV_LOOPED) {
|
|
cinTable[currentHandle].status = FMV_PLAY;
|
|
}
|
|
|
|
if (cinTable[currentHandle].status == FMV_EOF) {
|
|
if (cinTable[currentHandle].looping) {
|
|
RoQReset();
|
|
} else {
|
|
RoQShutdown();
|
|
}
|
|
}
|
|
|
|
return cinTable[currentHandle].status;
|
|
}
|
|
|
|
void Menus_CloseAll(void);
|
|
void UI_Cursor_Show(qboolean flag);
|
|
|
|
/*
|
|
==================
|
|
CL_PlayCinematic
|
|
|
|
==================
|
|
*/
|
|
int CIN_PlayCinematic( const char *arg, int x, int y, int w, int h, int systemBits, const char *psAudioFile /* = NULL */ )
|
|
{
|
|
unsigned short RoQID;
|
|
char name[MAX_OSPATH];
|
|
int i;
|
|
|
|
if (strstr(arg, "/") == NULL && strstr(arg, "\\") == NULL) {
|
|
Com_sprintf (name, sizeof(name), "video/%s", arg);
|
|
} else {
|
|
Com_sprintf (name, sizeof(name), "%s", arg);
|
|
}
|
|
COM_DefaultExtension(name,sizeof(name),".roq");
|
|
|
|
if (!(systemBits & CIN_system)) {
|
|
for ( i = 0 ; i < MAX_VIDEO_HANDLES ; i++ ) {
|
|
if (!strcmp(cinTable[i].fileName, name) ) {
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
|
|
Com_DPrintf("CIN_PlayCinematic( %s )\n", arg);
|
|
|
|
memset(&cin, 0, sizeof(cinematics_t) );
|
|
currentHandle = CIN_HandleForVideo();
|
|
|
|
cin.currentHandle = currentHandle;
|
|
|
|
Q_strncpyz(cinTable[currentHandle].fileName, name, MAX_OSPATH);
|
|
|
|
cinTable[currentHandle].ROQSize = 0;
|
|
cinTable[currentHandle].ROQSize = FS_FOpenFileRead (cinTable[currentHandle].fileName, &cinTable[currentHandle].iFile, qtrue);
|
|
|
|
if (cinTable[currentHandle].ROQSize<=0) {
|
|
Com_Printf(S_COLOR_RED"ERROR: playCinematic: %s not found!\n", arg);
|
|
cinTable[currentHandle].fileName[0] = 0;
|
|
return -1;
|
|
}
|
|
|
|
CIN_SetExtents(currentHandle, x, y, w, h);
|
|
CIN_SetLooping(currentHandle, (qboolean)((systemBits & CIN_loop) != 0));
|
|
|
|
cinTable[currentHandle].CIN_HEIGHT = DEFAULT_CIN_HEIGHT;
|
|
cinTable[currentHandle].CIN_WIDTH = DEFAULT_CIN_WIDTH;
|
|
cinTable[currentHandle].holdAtEnd = (qboolean)((systemBits & CIN_hold) != 0);
|
|
cinTable[currentHandle].alterGameState = (qboolean)((systemBits & CIN_system) != 0);
|
|
cinTable[currentHandle].playonwalls = 1;
|
|
cinTable[currentHandle].silent = (qboolean)((systemBits & CIN_silent) != 0);
|
|
cinTable[currentHandle].shader = (qboolean)((systemBits & CIN_shader) != 0);
|
|
if (psAudioFile)
|
|
{
|
|
cinTable[currentHandle].hSFX = S_RegisterSound(psAudioFile);
|
|
}
|
|
else
|
|
{
|
|
cinTable[currentHandle].hSFX = 0;
|
|
}
|
|
cinTable[currentHandle].hCRAWLTEXT = 0;
|
|
|
|
if (cinTable[currentHandle].alterGameState)
|
|
{
|
|
// close the menu
|
|
Con_Close();
|
|
if (cls.uiStarted)
|
|
{
|
|
UI_Cursor_Show(qfalse);
|
|
Menus_CloseAll();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
cinTable[currentHandle].playonwalls = cl_inGameVideo->integer;
|
|
}
|
|
|
|
initRoQ();
|
|
|
|
FS_Read (cin.file, 16, cinTable[currentHandle].iFile);
|
|
|
|
RoQID = (unsigned short)(cin.file[0]) + (unsigned short)(cin.file[1])*256;
|
|
if (RoQID == 0x1084)
|
|
{
|
|
RoQ_init();
|
|
// FS_Read (cin.file, cinTable[currentHandle].RoQFrameSize+8, cinTable[currentHandle].iFile);
|
|
|
|
cinTable[currentHandle].status = FMV_PLAY;
|
|
Com_DPrintf("trFMV::play(), playing %s\n", arg);
|
|
|
|
if (cinTable[currentHandle].alterGameState) {
|
|
cls.state = CA_CINEMATIC;
|
|
}
|
|
|
|
Con_Close();
|
|
|
|
if ( !cinTable[currentHandle].silent )
|
|
s_rawend = s_soundtime;
|
|
|
|
return currentHandle;
|
|
}
|
|
Com_DPrintf("trFMV::play(), invalid RoQ ID\n");
|
|
|
|
RoQShutdown();
|
|
return -1;
|
|
}
|
|
|
|
void CIN_SetExtents (int handle, int x, int y, int w, int h) {
|
|
if (handle < 0 || handle>= MAX_VIDEO_HANDLES || cinTable[handle].status == FMV_EOF) return;
|
|
cinTable[handle].xpos = x;
|
|
cinTable[handle].ypos = y;
|
|
cinTable[handle].width = w;
|
|
cinTable[handle].height = h;
|
|
cinTable[handle].dirty = qtrue;
|
|
}
|
|
|
|
void CIN_SetLooping(int handle, qboolean loop) {
|
|
if (handle < 0 || handle>= MAX_VIDEO_HANDLES || cinTable[handle].status == FMV_EOF) return;
|
|
cinTable[handle].looping = loop;
|
|
}
|
|
|
|
// Text crawl defines
|
|
#define TC_PLANE_WIDTH 250
|
|
#define TC_PLANE_NEAR 90
|
|
#define TC_PLANE_FAR 715
|
|
#define TC_PLANE_TOP 0
|
|
#define TC_PLANE_BOTTOM 1100
|
|
|
|
#define TC_DELAY 9000
|
|
#define TC_STOPTIME 81000
|
|
static void CIN_AddTextCrawl()
|
|
{
|
|
refdef_t refdef;
|
|
polyVert_t verts[4];
|
|
|
|
// Set up refdef
|
|
memset( &refdef, 0, sizeof( refdef ));
|
|
|
|
refdef.rdflags = RDF_NOWORLDMODEL;
|
|
AxisClear( refdef.viewaxis );
|
|
|
|
refdef.override_fov = true;
|
|
refdef.fov_x = 130;
|
|
refdef.fov_y = 130;
|
|
|
|
refdef.x = 0;
|
|
refdef.y = -50;
|
|
refdef.width = cls.glconfig.vidWidth;
|
|
refdef.height = cls.glconfig.vidHeight * 2; // deliberately extend off the bottom of the screen
|
|
|
|
// use to set shaderTime for scrolling shaders
|
|
refdef.time = 0;
|
|
|
|
// Set up the poly verts
|
|
float fadeDown = 1.0;
|
|
if (cls.realtime-CL_iPlaybackStartTime >= (TC_STOPTIME-2500))
|
|
{
|
|
fadeDown = (TC_STOPTIME - (cls.realtime-CL_iPlaybackStartTime))/ 2480.0f;
|
|
if (fadeDown < 0)
|
|
{
|
|
fadeDown = 0;
|
|
}
|
|
if (fadeDown > 1)
|
|
{
|
|
fadeDown = 1;
|
|
}
|
|
}
|
|
for ( int i = 0; i < 4; i++ )
|
|
{
|
|
verts[i].modulate[0] = 255*fadeDown; // gold color?
|
|
verts[i].modulate[1] = 235*fadeDown;
|
|
verts[i].modulate[2] = 127*fadeDown;
|
|
verts[i].modulate[3] = 255*fadeDown;
|
|
}
|
|
|
|
VectorScaleM( verts[2].modulate, 0.1f, verts[2].modulate ); // darken at the top??
|
|
VectorScaleM( verts[3].modulate, 0.1f, verts[3].modulate );
|
|
|
|
#define TIMEOFFSET +(cls.realtime-CL_iPlaybackStartTime-TC_DELAY)*0.000015f -1
|
|
VectorSet( verts[0].xyz, TC_PLANE_NEAR, -TC_PLANE_WIDTH, TC_PLANE_TOP );
|
|
verts[0].st[0] = 1;
|
|
verts[0].st[1] = 1 TIMEOFFSET;
|
|
|
|
VectorSet( verts[1].xyz, TC_PLANE_NEAR, TC_PLANE_WIDTH, TC_PLANE_TOP );
|
|
verts[1].st[0] = 0;
|
|
verts[1].st[1] = 1 TIMEOFFSET;
|
|
|
|
VectorSet( verts[2].xyz, TC_PLANE_FAR, TC_PLANE_WIDTH, TC_PLANE_BOTTOM );
|
|
verts[2].st[0] = 0;
|
|
verts[2].st[1] = 0 TIMEOFFSET;
|
|
|
|
VectorSet( verts[3].xyz, TC_PLANE_FAR, -TC_PLANE_WIDTH, TC_PLANE_BOTTOM );
|
|
verts[3].st[0] = 1;
|
|
verts[3].st[1] = 0 TIMEOFFSET;
|
|
|
|
// render it out
|
|
re.ClearScene();
|
|
re.AddPolyToScene( cinTable[CL_handle].hCRAWLTEXT, 4, verts );
|
|
re.RenderScene( &refdef );
|
|
|
|
//time's up
|
|
if (cls.realtime-CL_iPlaybackStartTime >= TC_STOPTIME)
|
|
{
|
|
// cinTable[currentHandle].holdAtEnd = qfalse;
|
|
cinTable[CL_handle].status = FMV_EOF;
|
|
RoQShutdown();
|
|
SCR_StopCinematic(); // change ROQ from FMV_IDLE to FMV_EOF, and clear some other vars
|
|
}
|
|
}
|
|
|
|
/*
|
|
==================
|
|
CIN_ResampleCinematic
|
|
|
|
Resample cinematic to 256x256 and store in buf2
|
|
==================
|
|
*/
|
|
void CIN_ResampleCinematic(int handle, int *buf2) {
|
|
int ix, iy, *buf3, xm, ym, ll;
|
|
byte *buf;
|
|
|
|
buf = cinTable[handle].buf;
|
|
|
|
xm = cinTable[handle].CIN_WIDTH/256;
|
|
ym = cinTable[handle].CIN_HEIGHT/256;
|
|
ll = 8;
|
|
if (cinTable[handle].CIN_WIDTH==512) {
|
|
ll = 9;
|
|
}
|
|
|
|
buf3 = (int*)buf;
|
|
if (xm==2 && ym==2) {
|
|
byte *bc2, *bc3;
|
|
int ic, iiy;
|
|
|
|
bc2 = (byte *)buf2;
|
|
bc3 = (byte *)buf3;
|
|
for (iy = 0; iy<256; iy++) {
|
|
iiy = iy<<12;
|
|
for (ix = 0; ix<2048; ix+=8) {
|
|
for(ic = ix;ic<(ix+4);ic++) {
|
|
*bc2=(bc3[iiy+ic]+bc3[iiy+4+ic]+bc3[iiy+2048+ic]+bc3[iiy+2048+4+ic])>>2;
|
|
bc2++;
|
|
}
|
|
}
|
|
}
|
|
} else if (xm==2 && ym==1) {
|
|
byte *bc2, *bc3;
|
|
int ic, iiy;
|
|
|
|
bc2 = (byte *)buf2;
|
|
bc3 = (byte *)buf3;
|
|
for (iy = 0; iy<256; iy++) {
|
|
iiy = iy<<11;
|
|
for (ix = 0; ix<2048; ix+=8) {
|
|
for(ic = ix;ic<(ix+4);ic++) {
|
|
*bc2=(bc3[iiy+ic]+bc3[iiy+4+ic])>>1;
|
|
bc2++;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
for (iy = 0; iy<256; iy++) {
|
|
for (ix = 0; ix<256; ix++) {
|
|
buf2[(iy<<8)+ix] = buf3[((iy*ym)<<ll) + (ix*xm)];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
==================
|
|
CIN_DrawCinematic
|
|
|
|
==================
|
|
*/
|
|
void CIN_DrawCinematic (int handle) {
|
|
float x, y, w, h;
|
|
byte *buf;
|
|
|
|
if (handle < 0 || handle>= MAX_VIDEO_HANDLES || cinTable[handle].status == FMV_EOF) return;
|
|
|
|
if (!cinTable[handle].buf) {
|
|
return;
|
|
}
|
|
|
|
x = cinTable[handle].xpos;
|
|
y = cinTable[handle].ypos;
|
|
w = cinTable[handle].width;
|
|
h = cinTable[handle].height;
|
|
buf = cinTable[handle].buf;
|
|
|
|
if (cinTable[handle].dirty && (cinTable[handle].CIN_WIDTH != cinTable[handle].drawX || cinTable[handle].CIN_HEIGHT != cinTable[handle].drawY)) {
|
|
int *buf2;
|
|
|
|
//buf2 = (int *)Hunk_AllocateTempMemory( 256*256*4 );
|
|
buf2 = (int*)Z_Malloc( 256*256*4, TAG_TEMP_WORKSPACE, qfalse );
|
|
|
|
CIN_ResampleCinematic(handle, buf2);
|
|
|
|
re.DrawStretchRaw( x, y, w, h, 256, 256, (byte *)buf2, handle, qtrue);
|
|
cinTable[handle].dirty = qfalse;
|
|
Z_Free(buf2); //Hunk_FreeTempMemory(buf2);
|
|
return;
|
|
}
|
|
|
|
re.DrawStretchRaw( x, y, w, h, cinTable[handle].drawX, cinTable[handle].drawY, buf, handle, cinTable[handle].dirty);
|
|
cinTable[handle].dirty = qfalse;
|
|
}
|
|
|
|
// external vars so I can check if the game is setup enough that I can play the intro video...
|
|
//
|
|
extern qboolean com_fullyInitialized;
|
|
extern qboolean s_soundStarted, s_soundMuted;
|
|
//
|
|
// ... and if the app isn't ready yet (which should only apply for the intro video), then I use these...
|
|
//
|
|
static char sPendingCinematic_Arg [256]={0};
|
|
static char sPendingCinematic_s [256]={0};
|
|
static qboolean gbPendingCinematic = qfalse;
|
|
//
|
|
// This stuff is for EF1-type ingame cinematics...
|
|
//
|
|
static qboolean qbPlayingInGameCinematic = qfalse;
|
|
static qboolean qbInGameCinematicOnStandBy = qfalse;
|
|
static char sInGameCinematicStandingBy[MAX_QPATH];
|
|
static char sTextCrawlFixedCinematic[MAX_QPATH];
|
|
static qboolean qbTextCrawlFixed = qfalse;
|
|
static int stopCinematicCallCount = 0;
|
|
|
|
|
|
|
|
static qboolean CIN_HardwareReadyToPlayVideos(void)
|
|
{
|
|
if (com_fullyInitialized && cls.rendererStarted &&
|
|
cls.soundStarted &&
|
|
cls.soundRegistered
|
|
)
|
|
{
|
|
return qtrue;
|
|
}
|
|
|
|
return qfalse;
|
|
}
|
|
|
|
|
|
static void PlayCinematic(const char *arg, const char *s, qboolean qbInGame)
|
|
{
|
|
qboolean bFailed = qfalse;
|
|
|
|
Cvar_Set( "timescale", "1" ); // jic we were skipping a scripted cinematic, return to normal after playing video
|
|
Cvar_Set( "skippingCinematic", "0" ); // ""
|
|
|
|
if(qbInGameCinematicOnStandBy == qfalse)
|
|
{
|
|
qbTextCrawlFixed = qfalse;
|
|
}
|
|
else
|
|
{
|
|
qbInGameCinematicOnStandBy = qfalse;
|
|
}
|
|
|
|
int bits = qbInGame?0:CIN_system;
|
|
|
|
Com_DPrintf("CL_PlayCinematic_f\n");
|
|
|
|
char sTemp[1024];
|
|
if (strstr(arg, "/") == NULL && strstr(arg, "\\") == NULL) {
|
|
Com_sprintf (sTemp, sizeof(sTemp), "video/%s", arg);
|
|
} else {
|
|
Com_sprintf (sTemp, sizeof(sTemp), "%s", arg);
|
|
}
|
|
COM_DefaultExtension(sTemp,sizeof(sTemp),".roq");
|
|
arg = &sTemp[0];
|
|
|
|
extern qboolean S_FileExists( const char *psFilename );
|
|
if (S_FileExists( arg ))
|
|
{
|
|
SCR_StopCinematic();
|
|
// command-line hack to avoid problems when playing intro video before app is fully setup...
|
|
//
|
|
if (!CIN_HardwareReadyToPlayVideos())
|
|
{
|
|
Q_strncpyz(sPendingCinematic_Arg,arg, 256);
|
|
Q_strncpyz(sPendingCinematic_s , (s&&s[0])?s:"", 256);
|
|
gbPendingCinematic = qtrue;
|
|
return;
|
|
}
|
|
|
|
qbPlayingInGameCinematic = qbInGame;
|
|
|
|
if ((s && s[0] == '1') || Q_stricmp(arg,"video/end.roq")==0) {
|
|
bits |= CIN_hold;
|
|
}
|
|
if (s && s[0] == '2') {
|
|
bits |= CIN_loop;
|
|
}
|
|
|
|
S_StopAllSounds ();
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
//
|
|
// work out associated audio-overlay file, if any...
|
|
//
|
|
extern cvar_t *s_language;
|
|
qboolean bIsForeign = (qboolean)(s_language && Q_stricmp(s_language->string,"english") && Q_stricmp(s_language->string,""));
|
|
const char *psAudioFile = NULL;
|
|
qhandle_t hCrawl = 0;
|
|
if (!Q_stricmp(arg,"video/jk0101_sw.roq"))
|
|
{
|
|
psAudioFile = "music/cinematic_1";
|
|
#ifdef JK2_MODE
|
|
hCrawl = re.RegisterShaderNoMip( va("menu/video/tc_%d", sp_language->integer) );
|
|
if(!hCrawl)
|
|
{
|
|
// failed, so go back to english
|
|
hCrawl = re.RegisterShaderNoMip( "menu/video/tc_0" );
|
|
}
|
|
#else
|
|
hCrawl = re.RegisterShaderNoMip( va("menu/video/tc_%s",se_language->string) );
|
|
if (!hCrawl)
|
|
{
|
|
hCrawl = re.RegisterShaderNoMip( "menu/video/tc_english" );//failed, so go back to english
|
|
}
|
|
#endif
|
|
bits |= CIN_hold;
|
|
}
|
|
else
|
|
if (bIsForeign)
|
|
{
|
|
if (!Q_stricmp(arg,"video/jk05.roq"))
|
|
{
|
|
psAudioFile = "sound/chars/video/cinematic_5";
|
|
bits |= CIN_silent; // knock out existing english track
|
|
}
|
|
else
|
|
if (!Q_stricmp(arg,"video/jk06.roq"))
|
|
{
|
|
psAudioFile = "sound/chars/video/cinematic_6";
|
|
bits |= CIN_silent; // knock out existing english track
|
|
}
|
|
}
|
|
//
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
CL_handle = CIN_PlayCinematic( arg, 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, bits, psAudioFile );
|
|
if (CL_handle >= 0)
|
|
{
|
|
cinTable[CL_handle].hCRAWLTEXT = hCrawl;
|
|
do
|
|
{
|
|
SCR_RunCinematic();
|
|
}
|
|
while (cinTable[currentHandle].buf == NULL && cinTable[currentHandle].status == FMV_PLAY); // wait for first frame (load codebook and sound)
|
|
|
|
if (qbInGame)
|
|
{
|
|
Cvar_SetValue( "cl_paused", 1); // remove-menu call will have unpaused us, so we sometimes need to re-pause
|
|
}
|
|
|
|
CL_iPlaybackStartTime = cls.realtime; // special use to avoid accidentally skipping ingame videos via fast-firing
|
|
}
|
|
else
|
|
{
|
|
// failed to open video...
|
|
//
|
|
bFailed = qtrue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// failed to open video...
|
|
//
|
|
bFailed = qtrue;
|
|
}
|
|
|
|
if (bFailed)
|
|
{
|
|
Com_Printf(S_COLOR_RED "PlayCinematic(): Failed to open \"%s\"\n",arg);
|
|
//S_RestartMusic(); //restart the level music
|
|
SCR_StopCinematic(); // I know this seems pointless, but it clears a bunch of vars as well
|
|
}
|
|
else
|
|
{
|
|
// this doesn't work for now...
|
|
//
|
|
// if (cls.state == CA_ACTIVE){
|
|
// re.InitDissolve(qfalse); // so we get a dissolve between previous screen image and cinematic
|
|
// }
|
|
}
|
|
}
|
|
|
|
|
|
qboolean CL_CheckPendingCinematic(void)
|
|
{
|
|
if ( gbPendingCinematic && CIN_HardwareReadyToPlayVideos() )
|
|
{
|
|
gbPendingCinematic = qfalse; // BEFORE next line, or we get recursion
|
|
PlayCinematic(sPendingCinematic_Arg,sPendingCinematic_s[0]?sPendingCinematic_s:NULL,qfalse);
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
/*
|
|
==================
|
|
CL_CompleteCinematic
|
|
==================
|
|
*/
|
|
void CL_CompleteCinematic( char *args, int argNum ) {
|
|
if ( argNum == 2 )
|
|
Field_CompleteFilename( "video", "roq", qtrue, qfalse );
|
|
}
|
|
|
|
void CL_PlayCinematic_f(void)
|
|
{
|
|
const char *arg, *s;
|
|
|
|
arg = Cmd_Argv( 1 );
|
|
s = Cmd_Argv(2);
|
|
PlayCinematic(arg,s,qfalse);
|
|
}
|
|
|
|
void CL_PlayInGameCinematic_f(void)
|
|
{
|
|
const char *arg = Cmd_Argv( 1 );
|
|
if (cls.state == CA_ACTIVE)
|
|
{
|
|
PlayCinematic(arg,NULL,qtrue);
|
|
}
|
|
else if( !qbInGameCinematicOnStandBy )
|
|
{
|
|
Q_strncpyz(sInGameCinematicStandingBy, arg, MAX_QPATH);
|
|
qbInGameCinematicOnStandBy = qtrue;
|
|
}
|
|
else
|
|
{
|
|
// hack in order to fix text crawl --eez
|
|
Q_strncpyz(sTextCrawlFixedCinematic, arg, MAX_QPATH);
|
|
qbTextCrawlFixed = qtrue;
|
|
}
|
|
}
|
|
|
|
// Externally-called only, and only if cls.state == CA_CINEMATIC (or CL_IsRunningInGameCinematic() == true now)
|
|
//
|
|
void SCR_DrawCinematic (void)
|
|
{
|
|
if (CL_InGameCinematicOnStandBy())
|
|
{
|
|
PlayCinematic(sInGameCinematicStandingBy,NULL,qtrue);
|
|
}
|
|
else if( qbTextCrawlFixed && stopCinematicCallCount > 1)
|
|
{
|
|
PlayCinematic(sTextCrawlFixedCinematic, NULL, qtrue);
|
|
}
|
|
|
|
if (CL_handle >= 0 && CL_handle < MAX_VIDEO_HANDLES) {
|
|
CIN_DrawCinematic(CL_handle);
|
|
if (cinTable[CL_handle].hCRAWLTEXT && (cls.realtime - CL_iPlaybackStartTime >= TC_DELAY))
|
|
{
|
|
CIN_AddTextCrawl();
|
|
}
|
|
}
|
|
}
|
|
|
|
void SCR_RunCinematic (void)
|
|
{
|
|
CL_CheckPendingCinematic();
|
|
|
|
if (CL_handle >= 0 && CL_handle < MAX_VIDEO_HANDLES) {
|
|
e_status Status = CIN_RunCinematic(CL_handle);
|
|
|
|
if (CL_IsRunningInGameCinematic() && Status == FMV_IDLE && !cinTable[CL_handle].holdAtEnd)
|
|
{
|
|
SCR_StopCinematic(); // change ROQ from FMV_IDLE to FMV_EOF, and clear some other vars
|
|
}
|
|
}
|
|
}
|
|
|
|
void SCR_StopCinematic( qboolean bAllowRefusal /* = qfalse */ )
|
|
{
|
|
if (bAllowRefusal)
|
|
{
|
|
if ( (CL_handle >= 0 && CL_handle < MAX_VIDEO_HANDLES)
|
|
&&
|
|
cls.realtime < CL_iPlaybackStartTime + 1200 // 1.2 seconds have to have elapsed
|
|
)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
if ( CL_IsRunningInGameCinematic())
|
|
{
|
|
Com_DPrintf("In-game Cinematic Stopped\n");
|
|
}
|
|
|
|
if (CL_handle >= 0 && CL_handle < MAX_VIDEO_HANDLES &&
|
|
stopCinematicCallCount != 1) { // hello no, don't want this plz
|
|
CIN_StopCinematic(CL_handle);
|
|
S_StopAllSounds ();
|
|
CL_handle = -1;
|
|
if (CL_IsRunningInGameCinematic()){
|
|
re.InitDissolve(qfalse); // dissolve from cinematic to underlying ingame
|
|
}
|
|
}
|
|
|
|
if (cls.state == CA_CINEMATIC)
|
|
{
|
|
Com_DPrintf("Cinematic Stopped\n");
|
|
cls.state = CA_DISCONNECTED;
|
|
}
|
|
|
|
if(sInGameCinematicStandingBy[0] &&
|
|
qbTextCrawlFixed)
|
|
{
|
|
// Hacky fix to help deal with broken text crawl..
|
|
// If we are skipping past the one on standby, DO NOT SKIP THE OTHER ONES!
|
|
stopCinematicCallCount++;
|
|
}
|
|
else if(stopCinematicCallCount == 1)
|
|
{
|
|
stopCinematicCallCount++;
|
|
}
|
|
else
|
|
{
|
|
// Skipping the last one in the list, go ahead and kill it.
|
|
qbTextCrawlFixed = qfalse;
|
|
sTextCrawlFixedCinematic[0] = 0;
|
|
stopCinematicCallCount = 0;
|
|
}
|
|
|
|
if(stopCinematicCallCount != 2)
|
|
{
|
|
qbPlayingInGameCinematic = qfalse;
|
|
qbInGameCinematicOnStandBy = qfalse;
|
|
sInGameCinematicStandingBy[0]=0;
|
|
Cvar_SetValue( "cl_paused", 0 );
|
|
}
|
|
if (cls.state != CA_DISCONNECTED) // cut down on needless calls to music code
|
|
{
|
|
S_RestartMusic(); //restart the level music
|
|
}
|
|
}
|
|
|
|
|
|
void CIN_UploadCinematic(int handle) {
|
|
if (handle >= 0 && handle < MAX_VIDEO_HANDLES) {
|
|
if (!cinTable[handle].buf) {
|
|
return;
|
|
}
|
|
if (cinTable[handle].playonwalls <= 0 && cinTable[handle].dirty) {
|
|
if (cinTable[handle].playonwalls == 0) {
|
|
cinTable[handle].playonwalls = -1;
|
|
} else {
|
|
if (cinTable[handle].playonwalls == -1) {
|
|
cinTable[handle].playonwalls = -2;
|
|
} else {
|
|
cinTable[handle].dirty = qfalse;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Resample the video if needed
|
|
if (cinTable[handle].dirty && (cinTable[handle].CIN_WIDTH != cinTable[handle].drawX || cinTable[handle].CIN_HEIGHT != cinTable[handle].drawY)) {
|
|
int *buf2;
|
|
|
|
buf2 = (int *)Z_Malloc(256*256*4, TAG_TEMP_WORKSPACE, qfalse);
|
|
|
|
CIN_ResampleCinematic(handle, buf2);
|
|
|
|
re.UploadCinematic( 256, 256, (byte *)buf2, handle, qtrue);
|
|
cinTable[handle].dirty = qfalse;
|
|
Z_Free(buf2);
|
|
} else {
|
|
// Upload video at normal resolution
|
|
re.UploadCinematic( cinTable[handle].drawX, cinTable[handle].drawY,
|
|
cinTable[handle].buf, handle, cinTable[handle].dirty);
|
|
cinTable[handle].dirty = qfalse;
|
|
}
|
|
|
|
if (cl_inGameVideo->integer == 0 && cinTable[handle].playonwalls == 1) {
|
|
cinTable[handle].playonwalls--;
|
|
}
|
|
else if (cl_inGameVideo->integer != 0 && cinTable[handle].playonwalls != 1) {
|
|
cinTable[handle].playonwalls = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
qboolean CL_IsRunningInGameCinematic(void)
|
|
{
|
|
return qbPlayingInGameCinematic;
|
|
}
|
|
|
|
qboolean CL_InGameCinematicOnStandBy(void)
|
|
{
|
|
return qbInGameCinematicOnStandBy;
|
|
}
|
|
|
|
|