1
0
forked from 0ad/0ad
0ad/source/lib/input.cpp
janwas 2e5d9452aa part2: misc source/lib fixes/improvements
move all except user-specified config choices out of config.h and into
appropriate headers
CPU_IA32 -> ARCH_IA32
wsdl: disable use of ddraw (will soon be replaced by WMI)

use shared_ptr without namespace qualifier (it's in tr1)
debug_warn -> debug_assert(0)
LIB_API to allow building as DLL
smart pointers: reduce use of .get()
cache_adt: use map instead of hash_map (avoids needing a hashCompare
class). also remove spurious warning
code_annotation.h: better cassert implementation
move FPS measuring portion of timer.cpp into frequency_filter
move include of memory headers into mmgr.h (to avoid errors, we must
ensure they are included if mmgr is used)
posix_filesystem.h: move definition of mkdir to wfilesystem
stl: disable iterator checks in release mode
wmi: fix COM init bug, use smart pointers
wutil: add code to get DLL module handle (if compiled as such), add
WinScopedLock
timer: fix handling of raw ticks

This was SVN commit r5517.
2007-12-20 20:09:19 +00:00

163 lines
2.8 KiB
C++

/**
* =========================================================================
* File : input.cpp
* Project : 0 A.D.
* Description : SDL input redirector; dispatches to multiple handlers and
* : allows record/playback of evs.
* =========================================================================
*/
// license: GPL; see lib/license.txt
#include "precompiled.h"
#include "input.h"
#include <stdio.h>
#include <stdlib.h>
#include "lib/external_libraries/sdl.h"
const uint MAX_HANDLERS = 8;
static InHandler handler_stack[MAX_HANDLERS];
static uint handler_stack_top = 0;
void in_add_handler(InHandler handler)
{
debug_assert(handler);
if(handler_stack_top >= MAX_HANDLERS)
WARN_ERR_RETURN(ERR::LIMIT);
handler_stack[handler_stack_top++] = handler;
}
// send ev to each handler until one returns IN_HANDLED
static void dispatch_ev(const SDL_Event_* ev)
{
for(int i = (int)handler_stack_top-1; i >= 0; i--)
{
debug_assert(handler_stack[i] && ev);
InReaction ret = handler_stack[i](ev);
// .. done, return
if(ret == IN_HANDLED)
return;
// .. next handler
else if(ret == IN_PASS)
continue;
// .. invalid return value
else
debug_assert(0); // invalid handler return value
}
}
//-----------------------------------------------------------------------------
static enum
{
INIT, // first call to in_record() or in_playback(): register cleanup routine
IDLE,
RECORD,
PLAYBACK
}
state = INIT;
static FILE* f;
u32 game_ticks;
static u32 time_adjust = 0;
static u32 next_ev_time;
void in_stop()
{
if(f)
{
fclose(f);
f = 0;
}
state = IDLE;
}
LibError in_record(const char* fn)
{
if(state == INIT)
atexit(in_stop);
in_stop();
f = fopen(fn, "wb");
if(!f)
WARN_RETURN(ERR::FAIL);
fwrite(&game_ticks, sizeof(u32), 1, f);
state = RECORD;
return INFO::OK;
}
LibError in_playback(const char* fn)
{
if(state == INIT)
atexit(in_stop);
in_stop();
f = fopen(fn, "rb");
if(!f)
WARN_RETURN(ERR::FAIL);
u32 rec_start_time;
fread(&rec_start_time, sizeof(u32), 1, f);
time_adjust = game_ticks-rec_start_time;
fread(&next_ev_time, sizeof(u32), 1, f);
next_ev_time += time_adjust;
state = PLAYBACK;
return INFO::OK;
}
void in_dispatch_event(const SDL_Event_* ev)
{
if(state == RECORD)
{
fwrite(&game_ticks, sizeof(u32), 1, f);
fwrite(ev, sizeof(SDL_Event_), 1, f);
}
dispatch_ev(ev);
}
void in_dispatch_recorded_events()
{
SDL_Event_ ev;
while(state == PLAYBACK && next_ev_time <= game_ticks)
{
fread(&ev, sizeof(SDL_Event_), 1, f);
// do this before dispatch_ev(),
// in case a handler calls in_stop() (setting f to 0)
if(!fread(&next_ev_time, sizeof(u32), 1, f))
{
in_stop();
exit(0x73c07d);
// TODO: 'disconnect'?
}
next_ev_time += time_adjust;
in_dispatch_event(&ev);
}
}