mirror of
https://github.com/yquake2/yquake2remaster.git
synced 2025-04-03 17:01:17 +00:00
game: load dynamic entities definition
Based on Anachronox: * https://github.com/hogsy/chronon/blob/master/game/g_spawn.cpp
This commit is contained in:
parent
7203b0ed78
commit
f64ab5bfb7
3 changed files with 274 additions and 0 deletions
|
@ -44,6 +44,77 @@ static spawn_t spawns[] = {
|
|||
#include "savegame/tables/spawnfunc_list.h"
|
||||
};
|
||||
|
||||
/* Definition of dynamic object */
|
||||
typedef struct
|
||||
{
|
||||
char classname[MAX_QPATH];
|
||||
char model_path[MAX_QPATH];
|
||||
vec3_t scale;
|
||||
char entity_type[MAX_QPATH];
|
||||
vec3_t mins;
|
||||
vec3_t maxs;
|
||||
char noshadow[MAX_QPATH];
|
||||
int solidflag;
|
||||
float walk_speed;
|
||||
float run_speed;
|
||||
int speed;
|
||||
int lighting;
|
||||
int blending;
|
||||
char target_sequence[MAX_QPATH];
|
||||
int misc_value;
|
||||
int no_mip;
|
||||
char spawn_sequence[MAX_QPATH];
|
||||
char description[MAX_QPATH];
|
||||
} dynamicentity_t;
|
||||
|
||||
static dynamicentity_t *dynamicentities;
|
||||
static int ndynamicentities;
|
||||
|
||||
static void
|
||||
DynamicSpawn(edict_t *self, dynamicentity_t *data)
|
||||
{
|
||||
self->movetype = MOVETYPE_NONE;
|
||||
self->solid = SOLID_BBOX;
|
||||
self->s.modelindex = gi.modelindex("<model_path>"/*data->model_path*/);
|
||||
|
||||
VectorCopy(data->mins, self->mins);
|
||||
VectorCopy(data->maxs, self->maxs);
|
||||
|
||||
gi.linkentity(self);
|
||||
}
|
||||
|
||||
static int
|
||||
DynamicSpawnSearch(const char *name)
|
||||
{
|
||||
int start, end;
|
||||
|
||||
start = 0;
|
||||
end = ndynamicentities - 1;
|
||||
|
||||
while (start <= end)
|
||||
{
|
||||
int i, res;
|
||||
|
||||
i = start + (end - start) / 2;
|
||||
|
||||
res = Q_stricmp(dynamicentities[i].classname, name);
|
||||
if (res == 0)
|
||||
{
|
||||
return i;
|
||||
}
|
||||
else if (res < 0)
|
||||
{
|
||||
start = i + 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
end = i - 1;
|
||||
}
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
static qboolean
|
||||
Spawn_CheckCoop_MapHacks(edict_t *ent)
|
||||
{
|
||||
|
@ -143,6 +214,14 @@ ED_CallSpawn(edict_t *ent)
|
|||
}
|
||||
}
|
||||
|
||||
i = DynamicSpawnSearch(ent->classname);
|
||||
if (i >= 0)
|
||||
{
|
||||
DynamicSpawn(ent, &dynamicentities[i]);
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
gi.dprintf("%s doesn't have a spawn function\n", ent->classname);
|
||||
}
|
||||
|
||||
|
@ -1596,3 +1675,194 @@ Widowlegs_Spawn(vec3_t startpos, vec3_t angles)
|
|||
ent->nextthink = level.time + FRAMETIME;
|
||||
gi.linkentity(ent);
|
||||
}
|
||||
|
||||
static char *
|
||||
DynamicStringParse(char *line, char *field, int size)
|
||||
{
|
||||
char *next_section, *current_section;
|
||||
|
||||
/* search line end */
|
||||
current_section = line;
|
||||
next_section = strchr(line, '|');
|
||||
if (next_section)
|
||||
{
|
||||
*next_section = 0;
|
||||
line = next_section + 1;
|
||||
}
|
||||
|
||||
/* copy current line state */
|
||||
strncpy(field, current_section, size);
|
||||
|
||||
return line;
|
||||
}
|
||||
|
||||
static char *
|
||||
DynamicIntParse(char *line, int *field)
|
||||
{
|
||||
char *next_section;
|
||||
|
||||
next_section = strchr(line, '|');
|
||||
if (next_section)
|
||||
{
|
||||
*next_section = 0;
|
||||
*field = (int)strtol(line, (char **)NULL, 10);
|
||||
line = next_section + 1;
|
||||
}
|
||||
|
||||
return line;
|
||||
}
|
||||
|
||||
static char *
|
||||
DynamicFloatParse(char *line, float *field, int size)
|
||||
{
|
||||
char *next_section;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < size; i++)
|
||||
{
|
||||
next_section = strchr(line, '|');
|
||||
if (next_section)
|
||||
{
|
||||
*next_section = 0;
|
||||
field[i] = (float)strtod(line, (char **)NULL);
|
||||
line = next_section + 1;
|
||||
}
|
||||
}
|
||||
return line;
|
||||
}
|
||||
|
||||
static int
|
||||
DynamicSort(const void *p1, const void *p2)
|
||||
{
|
||||
dynamicentity_t *ent1, *ent2;
|
||||
|
||||
ent1 = (dynamicentity_t*)p1;
|
||||
ent2 = (dynamicentity_t*)p2;
|
||||
return Q_stricmp(ent1->classname, ent2->classname);
|
||||
}
|
||||
|
||||
void
|
||||
DynamicSpawnInit(void)
|
||||
{
|
||||
char *buf, *raw;
|
||||
int len, curr_pos;
|
||||
|
||||
buf = NULL;
|
||||
len = 0;
|
||||
|
||||
/* load the file */
|
||||
len = gi.FS_LoadFile("models/entity.dat", (void **)&raw);
|
||||
if (len > 1)
|
||||
{
|
||||
buf = malloc(len + 1);
|
||||
memcpy(buf, raw, len);
|
||||
buf[len] = 0;
|
||||
gi.FS_FreeFile(raw);
|
||||
}
|
||||
|
||||
/* definition lines count */
|
||||
if (buf)
|
||||
{
|
||||
char *curr;
|
||||
|
||||
/* get lines count */
|
||||
curr = buf;
|
||||
while(*curr)
|
||||
{
|
||||
size_t linesize = 0;
|
||||
|
||||
linesize = strcspn(curr, "\n\r");
|
||||
if (*curr && strncmp(curr, "//", 2) &&
|
||||
*curr != '\n' && *curr != '\r' && *curr != ';')
|
||||
{
|
||||
ndynamicentities ++;
|
||||
}
|
||||
curr += linesize;
|
||||
if (curr >= (buf + len))
|
||||
{
|
||||
break;
|
||||
}
|
||||
/* skip our endline */
|
||||
curr++;
|
||||
}
|
||||
}
|
||||
|
||||
if (ndynamicentities)
|
||||
{
|
||||
dynamicentities = gi.TagMalloc(ndynamicentities * sizeof(*dynamicentities), TAG_GAME);
|
||||
memset(dynamicentities, 0, ndynamicentities * sizeof(*dynamicentities));
|
||||
}
|
||||
curr_pos = 0;
|
||||
|
||||
/* load definitons count */
|
||||
if (buf)
|
||||
{
|
||||
char *curr;
|
||||
|
||||
/* get lines count */
|
||||
curr = buf;
|
||||
while(*curr)
|
||||
{
|
||||
size_t linesize = 0;
|
||||
|
||||
if (curr_pos >= ndynamicentities)
|
||||
{
|
||||
break;
|
||||
}
|
||||
|
||||
linesize = strcspn(curr, "\n\r");
|
||||
curr[linesize] = 0;
|
||||
if (*curr && strncmp(curr, "//", 2) &&
|
||||
*curr != '\n' && *curr != '\r' && *curr != ';')
|
||||
{
|
||||
char *line;
|
||||
|
||||
line = curr;
|
||||
line = DynamicStringParse(line, dynamicentities[curr_pos].classname, MAX_QPATH);
|
||||
line = DynamicStringParse(line, dynamicentities[curr_pos].model_path, MAX_QPATH);
|
||||
line = DynamicFloatParse(line, dynamicentities[curr_pos].scale, 3);
|
||||
line = DynamicStringParse(line, dynamicentities[curr_pos].entity_type, MAX_QPATH);
|
||||
line = DynamicFloatParse(line, dynamicentities[curr_pos].mins, 3);
|
||||
line = DynamicFloatParse(line, dynamicentities[curr_pos].maxs, 3);
|
||||
line = DynamicStringParse(line, dynamicentities[curr_pos].noshadow, MAX_QPATH);
|
||||
line = DynamicIntParse(line, &dynamicentities[curr_pos].solidflag);
|
||||
line = DynamicFloatParse(line, &dynamicentities[curr_pos].walk_speed, 1);
|
||||
line = DynamicFloatParse(line, &dynamicentities[curr_pos].run_speed, 1);
|
||||
line = DynamicIntParse(line, &dynamicentities[curr_pos].speed);
|
||||
line = DynamicIntParse(line, &dynamicentities[curr_pos].lighting);
|
||||
line = DynamicIntParse(line, &dynamicentities[curr_pos].blending);
|
||||
line = DynamicStringParse(line, dynamicentities[curr_pos].target_sequence, MAX_QPATH);
|
||||
line = DynamicIntParse(line, &dynamicentities[curr_pos].misc_value);
|
||||
line = DynamicIntParse(line, &dynamicentities[curr_pos].no_mip);
|
||||
line = DynamicStringParse(line, dynamicentities[curr_pos].spawn_sequence, MAX_QPATH);
|
||||
line = DynamicStringParse(line, dynamicentities[curr_pos].description, MAX_QPATH);
|
||||
|
||||
curr_pos ++;
|
||||
}
|
||||
curr += linesize;
|
||||
if (curr >= (buf + len))
|
||||
{
|
||||
break;
|
||||
}
|
||||
/* skip our endline */
|
||||
curr++;
|
||||
}
|
||||
|
||||
ndynamicentities = curr_pos;
|
||||
|
||||
free(buf);
|
||||
}
|
||||
|
||||
/* save last used position */
|
||||
ndynamicentities = curr_pos;
|
||||
|
||||
if (!curr_pos)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
gi.dprintf("Found %d dynamic definitions\n", ndynamicentities);
|
||||
|
||||
/* sort definitions */
|
||||
qsort(dynamicentities, ndynamicentities, sizeof(dynamicentity_t), DynamicSort);
|
||||
}
|
||||
|
|
|
@ -796,6 +796,7 @@ edict_t *findradius2(edict_t *from, vec3_t org, float rad);
|
|||
|
||||
/* g_spawn.c */
|
||||
void ED_CallSpawn(edict_t *ent);
|
||||
void DynamicSpawnInit(void);
|
||||
|
||||
/* g_combat.c */
|
||||
qboolean OnSameTeam(edict_t *ent1, edict_t *ent2);
|
||||
|
|
|
@ -267,6 +267,9 @@ InitGame(void)
|
|||
/* initilize localization */
|
||||
LocalizationInit();
|
||||
|
||||
/* initilize dynamic object spawn */
|
||||
DynamicSpawnInit();
|
||||
|
||||
/* items */
|
||||
InitItems();
|
||||
|
||||
|
|
Loading…
Reference in a new issue