Skip to content

Commit

Permalink
Add *duplicate/remove npc script command
Browse files Browse the repository at this point in the history
- `npc_duplicate()` duplicate any existing duplicated NPC.
- `npc_duplicate_remove()` will remove any existing NPC other than source NPC.
  • Loading branch information
Emistry committed Jul 5, 2020
1 parent 1dfebbd commit 88f0c3d
Show file tree
Hide file tree
Showing 2 changed files with 181 additions and 0 deletions.
16 changes: 16 additions & 0 deletions doc/script_commands.txt
Original file line number Diff line number Diff line change
Expand Up @@ -7182,6 +7182,22 @@ otherwise it's displayed to the entire npc area.

---------------------------------------

*npc_duplicate("<source_npc_name>", "<new_npc_name>", "<new_npc_hidden_name>", "<mapname>", <map_x>, <map_y>, <dir>{, <sprite_id>{, <map_xs>, <map_ys>}});

Duplicate any existing NPC based on <source_npc_name>.
Return 1 on success, 0 if failed.

---------------------------------------

*npc_duplicate_remove({"<npc_name>", {<flag>}});

Remove any duplicated NPC from source NPC or depend on <npc_name> if specified.
If <npc_name> are source NPC, it will only remove all duplicated NPC.
If <flag> are specified, it will unload monster too. (default: 1)
Return 1 on success, 0 if failed.

---------------------------------------

*doevent("<NPC object name>::<event label>")

This command will start a new execution thread in a specified NPC object
Expand Down
165 changes: 165 additions & 0 deletions src/map/script.c
Original file line number Diff line number Diff line change
Expand Up @@ -13039,6 +13039,169 @@ static BUILDIN(cloakoffnpc)
return true;
}


/*
* Create a duplicate of source NPC.
* npc_duplicate("<source_npc_name>", "<new_npc_name>", "<new_npc_hidden_name>", "<mapname>", <map_x>, <map_y>, <dir>{, <sprite_id>{, <map_xs>, <map_ys>}});
* Return 1 on success, 0 if failed.
*/
static BUILDIN(npc_duplicate)
{
int txs = -1, tys = -1;

if (script_hasdata(st, 10))
txs = script_getnum(st, 10);
if (script_hasdata(st, 11))
tys = script_getnum(st, 11);

if (txs < -1)
txs = -1;
if (tys < -1)
tys = -1;

if (txs == -1 && tys != -1)
txs = 0;
if (txs != -1 && tys == -1)
tys = 0;

const char *dup_name = script_getstr(st, 3);
const char *dup_hidden_name = script_getstr(st, 4);

size_t len_dup = strlen(dup_name);
size_t len_dup_hid = strlen(dup_hidden_name);

if (len_dup + len_dup_hid + 1 > NAME_LENGTH - 1) {
ShowError("buildin_npc_duplicate: NPC name '%s#%s' is too long (max %d chars).\n", dup_name, dup_hidden_name, NAME_LENGTH);
script_pushint(st, 0);
return false;
}

if (strlen(dup_name) > NAME_LENGTH) {
ShowError("buildin_npc_duplicate: NPC duplicated name '%s' is too long (max %d chars).\n", dup_name, NAME_LENGTH);
script_pushint(st, 0);
return false;
}

char targetname[NAME_LENGTH] = "";
strcat(targetname, dup_name);

if (strlen(dup_hidden_name) != 0) {
strncat(targetname, "#", 1);
strncat(targetname, dup_hidden_name, strlen(dup_hidden_name));
}

if (strlen(targetname) > NAME_LENGTH) {
ShowError("buildin_npc_duplicate: NPC name '%s' is too long (max %d chars).\n", targetname, NAME_LENGTH);
script_pushint(st, 0);
return false;
} else if (npc->name2id(targetname) != NULL) {
ShowError("buildin_npc_duplicate: NPC named '%s' already exists.\n", targetname);
script_pushint(st, 0);
return false;
}

const char *npc_name = script_getstr(st, 2);
struct npc_data *nd_source = npc->name2id(npc_name);
int tclass_ = nd_source->class_;

if (script_hasdata(st, 9))
tclass_ = script_getnum(st, 9);
if (tclass_ < -1)
tclass_ = FAKE_NPC;

if (nd_source == NULL) {
ShowError("buildin_npc_duplicate: Source NPC '%s' not found.\n", npc_name);
script_pushint(st, 0);
return false;
}

if (nd_source->src_id != 0) {
ShowError("buildin_npc_duplicate: Source NPC '%s' is a duplicated NPC.\n", nd_source->name);
script_pushint(st, 0);
return false;
}

const char *tmap = script_getstr(st, 5);
int tmapid = map->mapname2mapid(tmap);
if (tmapid < 0) {
ShowError("buildin_npc_duplicate: Target map '%s' not found.\n", tmap);
script_pushint(st, 0);
return false;
}

if (map->list[tmapid].npc_num >= MAX_NPC_PER_MAP) {
ShowError("buildin_npc_duplicate: Exceeded MAX NPC per map (%d).\n", MAX_NPC_PER_MAP);
return false;
}

int tx = script_getnum(st, 6);
int ty = script_getnum(st, 7);
int tdir = script_getnum(st, 8);

if (tclass_ != FAKE_NPC && tclass_ != HIDDEN_WARP_CLASS) {
if (map->getcell(tmapid, NULL, tx, ty, CELL_CHKNOPASS) > 0) {
ShowError("buildin_npc_duplicate: Invalid NPC Location. %s,%d,%d\n", tmap, tx, ty);
script_pushint(st, 0);
return false;
}
}

if (tdir >= UNIT_DIR_MAX) {
ShowWarning("buildin_npc_duplicate: Invalid NPC direction %d. Default to %d.\n", tdir, (tdir % UNIT_DIR_MAX));
tdir %= UNIT_DIR_MAX; // trim spin-over
}
else if (tdir <= UNIT_DIR_UNDEFINED) {
ShowWarning("buildin_npc_duplicate: Invalid NPC direction %d. Default to %d.\n", tdir, UNIT_DIR_SOUTH);
tdir = UNIT_DIR_SOUTH;
}

struct npc_data *nd_target = npc->create_npc(nd_source->subtype, tmapid, tx, ty, tdir, tclass_);

safestrncpy(nd_target->name, targetname, sizeof(nd_target->name));
safestrncpy(nd_target->exname, targetname, sizeof(nd_target->exname));

if (npc->duplicate_sub(nd_target, nd_source, txs, tys, NPO_ONINIT) == true)
script_pushint(st, 1);
else
script_pushint(st, 0);

return true;
}

/*
* npc_duplicate_remove({"<npc_name>", {<flag>}});
* Return 1 on success, 0 if failed.
*/
static BUILDIN(npc_duplicate_remove)
{
struct npc_data *nd = map->id2nd(st->oid);

if (script_hasdata(st, 2))
nd = npc->name2id(script_getstr(st, 2));

if (nd == NULL) {
if (script_hasdata(st, 2)) {
ShowError("buildin_npc_duplicate_remove: NPC '%s' not found.\n", script_getstr(st, 2));
} else {
ShowError("buildin_npc_duplicate_remove: NPC not found.\n");
}
script_pushint(st, 0);
return false;
}

int flag = 1;
if (script_hasdata(st, 3))
flag = script_getnum(st, 3);

if (nd->src_id == 0) // remove all dupicates for this source npc
npc->unload_duplicates(nd, (flag != 0));
else // just remove this duplicate NPC
npc->unload(nd, true, (flag != 0));

script_pushint(st, 1);
return true;
}

/* Starts a status effect on the target unit or on the attached player.
*
* sc_start <effect_id>,<duration>,<val1>{,<rate>,<flag>,{<unit_id>}};
Expand Down Expand Up @@ -27413,6 +27576,8 @@ static void script_parse_builtin(void)
BUILDIN_DEF(hideonnpc,"s"),
BUILDIN_DEF(cloakonnpc,"s?"),
BUILDIN_DEF(cloakoffnpc,"s?"),
BUILDIN_DEF(npc_duplicate, "ssssiii???"),
BUILDIN_DEF(npc_duplicate_remove, "??"),
BUILDIN_DEF(sc_start,"iii???"),
BUILDIN_DEF2(sc_start,"sc_start2","iiii???"),
BUILDIN_DEF2(sc_start,"sc_start4","iiiiii???"),
Expand Down

0 comments on commit 88f0c3d

Please sign in to comment.