#include "npc.hpp" #include #include #include #include #include #include "../common/cxxstdio.hpp" #include "../common/db.hpp" #include "../common/extract.hpp" #include "../common/nullpo.hpp" #include "../common/socket.hpp" #include "../common/timer.hpp" #include "battle.hpp" #include "clif.hpp" #include "itemdb.hpp" #include "map.hpp" #include "mob.hpp" #include "pc.hpp" #include "script.hpp" #include "skill.hpp" #include "../poison.hpp" static std::list npc_srcs; static int npc_id = START_NPC_NUM; static int npc_warp, npc_shop, npc_script, npc_mob; int npc_get_new_npc_id(void) { return npc_id++; } struct event_data { dumb_ptr nd; int pos; }; static Map ev_db; static DMap> npcname_db; // used for clock-based event triggers // only tm_min, tm_hour, and tm_mday are used static struct tm ev_tm_b; /*========================================== * NPCの無効化/有効化 * npc_enable * npc_enable_sub 有効時にOnTouchイベントを実行 *------------------------------------------ */ static void npc_enable_sub(dumb_ptr bl, dumb_ptr nd) { dumb_ptr sd; nullpo_retv(bl); assert (bl->bl_type == BL::PC); { sd = bl->as_player(); // not if disabled if (nd->flag & 1) return; NpcEvent aname; aname.npc = nd->name; aname.label = stringish("OnTouch"); if (sd->areanpc_id == nd->bl_id) return; sd->areanpc_id = nd->bl_id; npc_event(sd, aname, 0); } } int npc_enable(NpcName name, bool flag) { dumb_ptr nd = npcname_db.get(name); if (nd == NULL) return 0; if (flag) { // 有効化 nd->flag &= ~1; clif_spawnnpc(nd); } else { // 無効化 nd->flag |= 1; clif_clearchar(nd, BeingRemoveWhy::GONE); } int xs = 0, ys = 0; if (dumb_ptr nd_ = nd->is_script()) { xs = nd_->scr.xs; ys = nd_->scr.ys; } if (flag && (xs > 0 || ys > 0)) map_foreachinarea(std::bind(npc_enable_sub, ph::_1, nd), nd->bl_m, nd->bl_x - xs, nd->bl_y - ys, nd->bl_x + xs, nd->bl_y + ys, BL::PC); return 0; } /*========================================== * NPCを名前で探す *------------------------------------------ */ dumb_ptr npc_name2id(NpcName name) { return npcname_db.get(name); } /*========================================== * イベントキューのイベント処理 *------------------------------------------ */ int npc_event_dequeue(dumb_ptr sd) { nullpo_ret(sd); sd->npc_id = 0; if (!sd->eventqueuel.empty()) { if (!pc_addeventtimer(sd, std::chrono::milliseconds(100), sd->eventqueuel.front())) { PRINTF("npc_event_dequeue(): Event timer is full.\n"); return 0; } sd->eventqueuel.pop_front(); return 1; } return 0; } int npc_delete(dumb_ptr nd) { nullpo_retr(1, nd); if (nd->bl_prev == NULL) return 1; clif_clearchar(nd, BeingRemoveWhy::DEAD); map_delblock(nd); return 0; } void npc_timer_event(NpcEvent eventname) { struct event_data *ev = ev_db.search(eventname); dumb_ptr nd; // int xs,ys; if ((ev == NULL || (nd = ev->nd) == NULL)) { PRINTF("npc_event: event not found [%s]\n", eventname); return; } run_script(ScriptPointer(nd->scr.script.get(), ev->pos), nd->bl_id, nd->bl_id); } /*========================================== * 全てのNPCのOn*イベント実行 *------------------------------------------ */ static void npc_event_doall_sub(NpcEvent key, struct event_data *ev, int *c, ScriptLabel name, int rid, int argc, argrec_t *argv) { ScriptLabel p = key.label; nullpo_retv(ev); if (name == p) { run_script_l(ScriptPointer(ev->nd->scr.script.get(), ev->pos), rid, ev->nd->bl_id, argc, argv); (*c)++; } } int npc_event_doall_l(ScriptLabel name, int rid, int argc, argrec_t *args) { int c = 0; for (auto& pair : ev_db) npc_event_doall_sub(pair.first, &pair.second, &c, name, rid, argc, args); return c; } static void npc_event_do_sub(NpcEvent key, struct event_data *ev, int *c, NpcEvent name, int rid, int argc, argrec_t *argv) { nullpo_retv(ev); if (name == key) { run_script_l(ScriptPointer(ev->nd->scr.script.get(), ev->pos), rid, ev->nd->bl_id, argc, argv); (*c)++; } } int npc_event_do_l(NpcEvent name, int rid, int argc, argrec_t *args) { int c = 0; if (!name.npc) { return npc_event_doall_l(name.label, rid, argc, args); } for (auto& pair : ev_db) npc_event_do_sub(pair.first, &pair.second, &c, name, rid, argc, args); return c; } /*========================================== * 時計イベント実行 *------------------------------------------ */ static void npc_event_do_clock(TimerData *, tick_t) { struct tm t = TimeT::now(); ScriptLabel buf; if (t.tm_min != ev_tm_b.tm_min) { SNPRINTF(buf, 24, "OnMinute%02d", t.tm_min); npc_event_doall(buf); SNPRINTF(buf, 24, "OnClock%02d%02d", t.tm_hour, t.tm_min); npc_event_doall(buf); } if (t.tm_hour != ev_tm_b.tm_hour) { SNPRINTF(buf, 24, "OnHour%02d", t.tm_hour); npc_event_doall(buf); } if (t.tm_mday != ev_tm_b.tm_mday) { SNPRINTF(buf, 24, "OnDay%02d%02d", t.tm_mon + 1, t.tm_mday); npc_event_doall(buf); } ev_tm_b = t; } /*========================================== * OnInitイベント実行(&時計イベント開始) *------------------------------------------ */ int npc_event_do_oninit(void) { int c = npc_event_doall(stringish("OnInit")); PRINTF("npc: OnInit Event done. (%d npc)\n", c); Timer(gettick() + std::chrono::milliseconds(100), npc_event_do_clock, std::chrono::seconds(1) ).detach(); return 0; } /*========================================== * タイマーイベント実行 *------------------------------------------ */ static void npc_timerevent(TimerData *, tick_t tick, int id, interval_t data) { dumb_ptr nd = map_id2bl(id)->as_npc()->as_script(); assert (nd != NULL); assert (nd->npc_subtype == NpcSubtype::SCRIPT); assert (nd->scr.nexttimer != nd->scr.timer_eventv.end()); nd->scr.timertick = tick; auto te = nd->scr.nexttimer; // nd->scr.timerid = nullptr; // er, isn't this the same as nd->scr.timer = te->timer? interval_t t = nd->scr.timer += data; assert (t == te->timer); ++nd->scr.nexttimer; if (nd->scr.nexttimer != nd->scr.timer_eventv.end()) { interval_t next = nd->scr.nexttimer->timer - t; nd->scr.timerid = Timer(tick + next, std::bind(npc_timerevent, ph::_1, ph::_2, id, next)); } run_script(ScriptPointer(nd->scr.script.get(), te->pos), 0, nd->bl_id); } /*========================================== * タイマーイベント開始 *------------------------------------------ */ void npc_timerevent_start(dumb_ptr nd) { nullpo_retv(nd); if (nd->scr.timer_eventv.empty()) return; if (nd->scr.nexttimer != nd->scr.timer_eventv.end()) return; if (nd->scr.timer == nd->scr.timer_eventv.back().timer) return; npc_timerevent_list phony {}; phony.timer = nd->scr.timer; // find the first element such that el.timer > phony.timer; auto jt = std::upper_bound(nd->scr.timer_eventv.begin(), nd->scr.timer_eventv.end(), phony, [](const npc_timerevent_list& l, const npc_timerevent_list& r) { return l.timer < r.timer; } ); nd->scr.nexttimer = jt; nd->scr.timertick = gettick(); if (jt == nd->scr.timer_eventv.end()) // shouldn't happen? return; interval_t next = jt->timer - nd->scr.timer; nd->scr.timerid = Timer(gettick() + next, std::bind(npc_timerevent, ph::_1, ph::_2, nd->bl_id, next)); } /*========================================== * タイマーイベント終了 *------------------------------------------ */ void npc_timerevent_stop(dumb_ptr nd) { nullpo_retv(nd); if (nd->scr.nexttimer != nd->scr.timer_eventv.end()) { nd->scr.nexttimer = nd->scr.timer_eventv.end(); nd->scr.timer += gettick() - nd->scr.timertick; nd->scr.timerid.cancel(); } } /*========================================== * タイマー値の所得 *------------------------------------------ */ interval_t npc_gettimerevent_tick(dumb_ptr nd) { nullpo_retr(interval_t::zero(), nd); interval_t tick = nd->scr.timer; // Couldn't we just check the truthiness of the timer? // Or would that be affected by the (new!) detach logic? // Of course, you'd be slightly crazy to check the tick when you are // called with it. if (nd->scr.nexttimer != nd->scr.timer_eventv.end()) tick += gettick() - nd->scr.timertick; return tick; } /*========================================== * タイマー値の設定 *------------------------------------------ */ void npc_settimerevent_tick(dumb_ptr nd, interval_t newtimer) { nullpo_retv(nd); bool flag = nd->scr.nexttimer != nd->scr.timer_eventv.end(); npc_timerevent_stop(nd); nd->scr.timer = newtimer; if (flag) npc_timerevent_start(nd); } /*========================================== * イベント型のNPC処理 *------------------------------------------ */ int npc_event(dumb_ptr sd, NpcEvent eventname, int mob_kill) { struct event_data *ev = ev_db.search(eventname); dumb_ptr nd; int xs, ys; if (sd == NULL) { PRINTF("npc_event nullpo?\n"); } if (ev == NULL && eventname.label == stringish("OnTouch")) return 1; if (ev == NULL || (nd = ev->nd) == NULL) { if (mob_kill) { { return 0; } } else { if (battle_config.error_log) PRINTF("npc_event: event not found [%s]\n", eventname); return 0; } } xs = nd->scr.xs; ys = nd->scr.ys; if (xs >= 0 && ys >= 0) { if (nd->bl_m != sd->bl_m) return 1; if (xs > 0 && (sd->bl_x < nd->bl_x - xs / 2 || nd->bl_x + xs / 2 < sd->bl_x)) return 1; if (ys > 0 && (sd->bl_y < nd->bl_y - ys / 2 || nd->bl_y + ys / 2 < sd->bl_y)) return 1; } if (sd->npc_id != 0) { sd->eventqueuel.push_back(eventname); return 1; } if (nd->flag & 1) { // 無効化されている npc_event_dequeue(sd); return 0; } sd->npc_id = nd->bl_id; sd->npc_pos = run_script(ScriptPointer(nd->scr.script.get(), ev->pos), sd->bl_id, nd->bl_id); return 0; } static void npc_command_sub(NpcEvent key, struct event_data *ev, NpcName npcname, XString command) { if (ev->nd->name == npcname && key.label.startswith("OnCommand")) { XString temp = key.label.xslice_t(9); if (command == temp) run_script(ScriptPointer(ev->nd->scr.script.get(), ev->pos), 0, ev->nd->bl_id); } } int npc_command(dumb_ptr, NpcName npcname, XString command) { for (auto& pair : ev_db) npc_command_sub(pair.first, &pair.second, npcname, command); return 0; } /*========================================== * 接触型のNPC処理 *------------------------------------------ */ int npc_touch_areanpc(dumb_ptr sd, map_local *m, int x, int y) { int i, f = 1; int xs, ys; nullpo_retr(1, sd); if (sd->npc_id) return 1; for (i = 0; i < m->npc_num; i++) { if (m->npc[i]->flag & 1) { // 無効化されている f = 0; continue; } switch (m->npc[i]->npc_subtype) { case NpcSubtype::WARP: xs = m->npc[i]->as_warp()->warp.xs; ys = m->npc[i]->as_warp()->warp.ys; break; case NpcSubtype::MESSAGE: assert (0 && "I'm pretty sure these are never put on a map"); xs = 0; ys = 0; break; case NpcSubtype::SCRIPT: xs = m->npc[i]->as_script()->scr.xs; ys = m->npc[i]->as_script()->scr.ys; break; default: continue; } if (x >= m->npc[i]->bl_x - xs / 2 && x < m->npc[i]->bl_x - xs / 2 + xs && y >= m->npc[i]->bl_y - ys / 2 && y < m->npc[i]->bl_y - ys / 2 + ys) break; } if (i == m->npc_num) { if (f) { if (battle_config.error_log) PRINTF("npc_touch_areanpc : some bug \n"); } return 1; } switch (m->npc[i]->npc_subtype) { case NpcSubtype::WARP: skill_stop_dancing(sd, 0); pc_setpos(sd, m->npc[i]->as_warp()->warp.name, m->npc[i]->as_warp()->warp.x, m->npc[i]->as_warp()->warp.y, BeingRemoveWhy::GONE); break; case NpcSubtype::MESSAGE: assert (0 && "I'm pretty sure these NPCs are never put on a map."); break; case NpcSubtype::SCRIPT: { NpcEvent aname; aname.npc = m->npc[i]->name; aname.label = stringish("OnTouch"); if (sd->areanpc_id == m->npc[i]->bl_id) return 1; sd->areanpc_id = m->npc[i]->bl_id; if (npc_event(sd, aname, 0) > 0) npc_click(sd, m->npc[i]->bl_id); break; } } return 0; } /*========================================== * 近くかどうかの判定 *------------------------------------------ */ static int npc_checknear(dumb_ptr sd, int id) { dumb_ptr nd; nullpo_ret(sd); nd = map_id_as_npc(id); assert (nd != NULL); assert (nd->bl_type == BL::NPC); if (nd->npc_class < 0) // イベント系は常にOK return 0; // エリア判定 if (nd->bl_m != sd->bl_m || nd->bl_x < sd->bl_x - AREA_SIZE - 1 || nd->bl_x > sd->bl_x + AREA_SIZE + 1 || nd->bl_y < sd->bl_y - AREA_SIZE - 1 || nd->bl_y > sd->bl_y + AREA_SIZE + 1) return 1; return 0; } /*========================================== * クリック時のNPC処理 *------------------------------------------ */ int npc_click(dumb_ptr sd, int id) { dumb_ptr nd; nullpo_retr(1, sd); if (sd->npc_id != 0) { if (battle_config.error_log) PRINTF("npc_click: npc_id != 0\n"); return 1; } if (npc_checknear(sd, id)) { clif_scriptclose(sd, id); return 1; } nd = map_id_as_npc(id); if (nd->flag & 1) // 無効化されている return 1; sd->npc_id = id; switch (nd->npc_subtype) { case NpcSubtype::SHOP: clif_npcbuysell(sd, id); npc_event_dequeue(sd); break; case NpcSubtype::SCRIPT: sd->npc_pos = run_script(ScriptPointer(nd->as_script()->scr.script.get(), 0), sd->bl_id, id); break; case NpcSubtype::MESSAGE: if (nd->as_message()->message) { clif_scriptmes(sd, id, nd->as_message()->message); clif_scriptclose(sd, id); } break; } return 0; } /*========================================== * *------------------------------------------ */ int npc_scriptcont(dumb_ptr sd, int id) { dumb_ptr nd; nullpo_retr(1, sd); if (id != sd->npc_id) return 1; if (npc_checknear(sd, id)) { clif_scriptclose(sd, id); return 1; } nd = map_id_as_npc(id); if (!nd /* NPC was disposed? */ || nd->npc_subtype == NpcSubtype::MESSAGE) { clif_scriptclose(sd, id); npc_event_dequeue(sd); return 0; } sd->npc_pos = run_script(ScriptPointer(nd->as_script()->scr.script.get(), sd->npc_pos), sd->bl_id, id); return 0; } /*========================================== * *------------------------------------------ */ int npc_buysellsel(dumb_ptr sd, int id, int type) { dumb_ptr nd; nullpo_retr(1, sd); if (npc_checknear(sd, id)) return 1; nd = map_id_as_npc(id); if (nd->npc_subtype != NpcSubtype::SHOP) { if (battle_config.error_log) PRINTF("no such shop npc : %d\n", id); sd->npc_id = 0; return 1; } if (nd->flag & 1) // 無効化されている return 1; sd->npc_shopid = id; if (type == 0) { clif_buylist(sd, nd->as_shop()); } else { clif_selllist(sd); } return 0; } /*========================================== * *------------------------------------------ */ // TODO enumify return type int npc_buylist(dumb_ptr sd, int n, const uint16_t *item_list) { dumb_ptr nd; double z; int i, j, w, itemamount = 0, new_stacks = 0; nullpo_retr(3, sd); nullpo_retr(3, item_list); if (npc_checknear(sd, sd->npc_shopid)) return 3; nd = map_id_as_npc(sd->npc_shopid); if (nd->npc_subtype != NpcSubtype::SHOP) return 3; for (i = 0, w = 0, z = 0; i < n; i++) { for (j = 0; j < nd->as_shop()->shop_items.size(); j++) { if (nd->as_shop()->shop_items[j].nameid == item_list[i * 2 + 1]) break; } if (j == nd->as_shop()->shop_items.size()) return 3; z += static_cast(nd->as_shop()->shop_items[j].value) * item_list[i * 2]; itemamount += item_list[i * 2]; switch (pc_checkadditem(sd, item_list[i * 2 + 1], item_list[i * 2])) { case ADDITEM::EXIST: break; case ADDITEM::NEW: if (itemdb_isequip(item_list[i * 2 + 1])) new_stacks += item_list[i * 2]; else new_stacks++; break; case ADDITEM::OVERAMOUNT: return 2; } w += itemdb_weight(item_list[i * 2 + 1]) * item_list[i * 2]; } if (z > static_cast(sd->status.zeny)) return 1; // zeny不足 if (w + sd->weight > sd->max_weight) return 2; // 重量超過 if (pc_inventoryblank(sd) < new_stacks) return 3; // 種類数超過 if (sd->trade_partner != 0) return 4; // cant buy while trading pc_payzeny(sd, static_cast(z)); for (i = 0; i < n; i++) { struct item_data *item_data; if ((item_data = itemdb_exists(item_list[i * 2 + 1])) != NULL) { int amount = item_list[i * 2]; struct item item_tmp {}; item_tmp.nameid = item_data->nameid; item_tmp.identify = 1; // npc販売アイテムは鑑定済み if (amount > 1 && (item_data->type == ItemType::WEAPON || item_data->type == ItemType::ARMOR || item_data->type == ItemType::_7 || item_data->type == ItemType::_8)) { for (j = 0; j < amount; j++) { pc_additem(sd, &item_tmp, 1); } } else { pc_additem(sd, &item_tmp, amount); } } } return 0; } /*========================================== * *------------------------------------------ */ int npc_selllist(dumb_ptr sd, int n, const uint16_t *item_list) { double z; int i, itemamount = 0; nullpo_retr(1, sd); nullpo_retr(1, item_list); if (npc_checknear(sd, sd->npc_shopid)) return 1; for (i = 0, z = 0; i < n; i++) { int nameid; if (item_list[i * 2] - 2 < 0 || item_list[i * 2] - 2 >= MAX_INVENTORY) return 1; nameid = sd->status.inventory[item_list[i * 2] - 2].nameid; if (nameid == 0 || sd->status.inventory[item_list[i * 2] - 2].amount < item_list[i * 2 + 1]) return 1; if (sd->trade_partner != 0) return 2; // cant sell while trading z += static_cast(itemdb_value_sell(nameid)) * item_list[i * 2 + 1]; itemamount += item_list[i * 2 + 1]; } if (z > MAX_ZENY) z = MAX_ZENY; pc_getzeny(sd, static_cast(z)); for (i = 0; i < n; i++) { int item_id = item_list[i * 2] - 2; pc_delitem(sd, item_id, item_list[i * 2 + 1], 0); } return 0; } // // 初期化関係 // /*========================================== * 読み込むnpcファイルのクリア *------------------------------------------ */ static void npc_clearsrcfile(void) { npc_srcs.clear(); } /*========================================== * 読み込むnpcファイルの追加 *------------------------------------------ */ void npc_addsrcfile(FString name) { if (name == "clear") { npc_clearsrcfile(); return; } npc_srcs.push_back(name); } /*========================================== * 読み込むnpcファイルの削除 *------------------------------------------ */ void npc_delsrcfile(FString name) { if (name == "all") { npc_clearsrcfile(); return; } for (auto it = npc_srcs.begin(); it != npc_srcs.end(); ++it) { if (*it == name) { npc_srcs.erase(it); return; } } } /*========================================== * warp行解析 *------------------------------------------ */ int npc_parse_warp(XString w1, XString, NpcName w3, XString w4) { int x, y, xs, ys, to_x, to_y; int i, j; MapName mapname, to_mapname; dumb_ptr nd; if (!extract(w1, record<','>(&mapname, &x, &y)) || !extract(w4, record<','>(&xs, &ys, &to_mapname, &to_x, &to_y))) { PRINTF("bad warp line : %s\n", w3); return 1; } map_local *m = map_mapname2mapid(mapname); nd.new_(); nd->bl_id = npc_get_new_npc_id(); nd->n = map_addnpc(m, nd); nd->bl_prev = nd->bl_next = NULL; nd->bl_m = m; nd->bl_x = x; nd->bl_y = y; nd->dir = DIR::S; nd->flag = 0; nd->name = w3; if (!battle_config.warp_point_debug) nd->npc_class = WARP_CLASS; else nd->npc_class = WARP_DEBUG_CLASS; nd->speed = std::chrono::milliseconds(200); nd->option = Option::ZERO; nd->opt1 = Opt1::ZERO; nd->opt2 = Opt2::ZERO; nd->opt3 = Opt3::ZERO; nd->warp.name = to_mapname; xs += 2; ys += 2; nd->warp.x = to_x; nd->warp.y = to_y; nd->warp.xs = xs; nd->warp.ys = ys; for (i = 0; i < ys; i++) { for (j = 0; j < xs; j++) { int x_lo = x - xs / 2; int y_lo = y - ys / 2; int xc = x_lo + j; int yc = y_lo + i; MapCell t = map_getcell(m, xc, yc); if (bool(t & MapCell::UNWALKABLE)) continue; map_setcell(m, xc, yc, t | MapCell::NPC_NEAR); } } // PRINTF("warp npc %s %d read done\n",mapname,nd->bl_id); npc_warp++; nd->bl_type = BL::NPC; nd->npc_subtype = NpcSubtype::WARP; map_addblock(nd); clif_spawnnpc(nd); npcname_db.put(nd->name, nd); return 0; } static bool extract(XString xs, npc_item_list *itv) { XString name_or_id; if (!extract(xs, record<':'>(&name_or_id, &itv->value))) return false; struct item_data *id = nullptr; if (extract(name_or_id, &itv->nameid) && itv->nameid > 0) goto return_true; id = itemdb_searchname(stringish(name_or_id.rstrip())); if (id == NULL) return false; itv->nameid = id->nameid; goto return_true; return_true: if (itv->value < 0) { if (id == NULL) id = itemdb_search(itv->nameid); itv->value = id->value_buy * abs(itv->value); } return true; } /*========================================== * shop行解析 *------------------------------------------ */ static int npc_parse_shop(XString w1, XString, NpcName w3, ZString w4a) { int x, y; DIR dir; MapName mapname; dumb_ptr nd; ZString::iterator w4comma; int npc_class; int dir_; // TODO use enum directly in extract if (!extract(w1, record<','>(&mapname, &x, &y, &dir_)) || dir_ < 0 || dir_ >= 8 || (w4comma = std::find(w4a.begin(), w4a.end(), ',')) == w4a.end() || !extract(w4a.xislice_h(w4comma), &npc_class)) { PRINTF("bad shop line : %s\n", w3); return 1; } dir = static_cast(dir_); map_local *m = map_mapname2mapid(mapname); nd.new_(); ZString w4b = w4a.xislice_t(w4comma + 1); if (!extract(w4b, vrec<','>(&nd->shop_items))) { PRINTF("bad shop items : %s\n", w3); PRINTF(" somewhere --> %s\n", w4b); nd->shop_items.clear(); } if (nd->shop_items.empty()) { nd.delete_(); return 1; } nd->bl_prev = nd->bl_next = NULL; nd->bl_m = m; nd->bl_x = x; nd->bl_y = y; nd->bl_id = npc_get_new_npc_id(); nd->dir = dir; nd->flag = 0; nd->name = w3; nd->npc_class = npc_class; nd->speed = std::chrono::milliseconds(200); nd->option = Option::ZERO; nd->opt1 = Opt1::ZERO; nd->opt2 = Opt2::ZERO; nd->opt3 = Opt3::ZERO; npc_shop++; nd->bl_type = BL::NPC; nd->npc_subtype = NpcSubtype::SHOP; nd->n = map_addnpc(m, nd); map_addblock(nd); clif_spawnnpc(nd); npcname_db.put(nd->name, nd); return 0; } /*========================================== * NPCのラベルデータコンバート *------------------------------------------ */ static void npc_convertlabel_db(ScriptLabel lname, int pos, dumb_ptr nd) { nullpo_retv(nd); struct npc_label_list eln {}; eln.name = lname; eln.pos = pos; nd->scr.label_listv.push_back(std::move(eln)); } /*========================================== * script行解析 *------------------------------------------ */ static int npc_parse_script(XString w1, XString w2, NpcName w3, ZString w4, XString first_line, FILE *fp, int *lines) { int x, y; DIR dir = DIR::S; map_local *m; int xs = 0, ys = 0, npc_class = 0; // [Valaris] thanks to fov MapName mapname; std::unique_ptr script = NULL; dumb_ptr nd; int evflag = 0; if (w1 == "-") { x = 0; y = 0; m = nullptr; } else { int dir_; // TODO use enum directly in extract if (!extract(w1, record<','>(&mapname, &x, &y, &dir_)) || dir_ < 0 || dir_ >= 8 || (w2 == "script" && !w4.contains(','))) { PRINTF("bad script line : %s\n", w3); return 1; } dir = static_cast(dir_); m = map_mapname2mapid(mapname); } if (w2 == "script") { // may be empty MString srcbuf; srcbuf += first_line.xislice_t(std::find(first_line.begin(), first_line.end(), '{')); // Note: it was a bug that this was missing. I think. int startline = *lines; // while (!srcbuf.rstrip().endswith('}')) while (true) { auto it = std::find_if_not(srcbuf.rbegin(), srcbuf.rend(), [](char c){ return c == ' ' || c == '\n'; }); if (it != srcbuf.rend() && *it == '}') break; char line_[1024]; if (!fgets(line_, 1020, fp)) // eof break; (*lines)++; if (feof(fp)) break; ZString line(ZString::really_construct_from_a_pointer, line_, nullptr); if (!srcbuf) { // may be a no-op srcbuf += line.xislice_t(std::find(line.begin(), line.end(), '{')); // safe to execute more than once // But will usually only happen once startline = *lines; } else srcbuf += line; } script = parse_script(FString(srcbuf), startline); if (script == NULL) // script parse error? return 1; } else { assert(0 && "duplicate() is no longer supported!\n"); return 0; } nd.new_(); if (m == nullptr) { } else if (extract(w4, record<','>(&npc_class, &xs, &ys))) { if (xs >= 0) xs = xs * 2 + 1; if (ys >= 0) ys = ys * 2 + 1; if (npc_class >= 0) { for (int i = 0; i < ys; i++) { for (int j = 0; j < xs; j++) { int x_lo = x - xs / 2; int y_lo = y - ys / 2; int xc = x_lo + j; int yc = y_lo + i; MapCell t = map_getcell(m, xc, yc); if (bool(t & MapCell::UNWALKABLE)) continue; map_setcell(m, xc, yc, t | MapCell::NPC_NEAR); } } } nd->scr.xs = xs; nd->scr.ys = ys; } else { npc_class = atoi(w4.c_str()); nd->scr.xs = 0; nd->scr.ys = 0; } if (npc_class < 0 && m != nullptr) { evflag = 1; } if (w3.contains(':')) { assert(false && "feature removed"); abort(); } { nd->name = w3; } nd->bl_prev = nd->bl_next = NULL; nd->bl_m = m; nd->bl_x = x; nd->bl_y = y; nd->bl_id = npc_get_new_npc_id(); nd->dir = dir; nd->flag = 0; nd->npc_class = npc_class; nd->speed = std::chrono::milliseconds(200); nd->scr.script = std::move(script); nd->option = Option::ZERO; nd->opt1 = Opt1::ZERO; nd->opt2 = Opt2::ZERO; nd->opt3 = Opt3::ZERO; //PRINTF("script npc %s %d %d read done\n",mapname,nd->bl_id,nd->class); npc_script++; nd->bl_type = BL::NPC; nd->npc_subtype = NpcSubtype::SCRIPT; if (m != nullptr) { nd->n = map_addnpc(m, nd); map_addblock(nd); if (evflag) { struct event_data ev {}; ev.nd = nd; ev.pos = 0; NpcEvent npcev; npcev.npc = nd->name; npcev.label = ScriptLabel(); ev_db.insert(npcev, ev); } else clif_spawnnpc(nd); } npcname_db.put(nd->name, nd); for (auto& pair : scriptlabel_db) npc_convertlabel_db(pair.first, pair.second, nd); for (npc_label_list& el : nd->scr.label_listv) { ScriptLabel lname = el.name; int pos = el.pos; if (lname.startswith("On")) { struct event_data ev {}; ev.nd = nd; ev.pos = pos; NpcEvent buf; buf.npc = nd->name; buf.label = lname; ev_db.insert(buf, ev); } } //----------------------------------------- // ラベルデータからタイマーイベント取り込み for (npc_label_list& el : nd->scr.label_listv) { int t_ = 0; ScriptLabel lname = el.name; int pos = el.pos; if (lname.startswith("OnTimer") && extract(lname.xslice_t(7), &t_)) { interval_t t = static_cast(t_); npc_timerevent_list tel {}; tel.timer = t; tel.pos = pos; auto it = std::lower_bound(nd->scr.timer_eventv.begin(), nd->scr.timer_eventv.end(), tel, [](const npc_timerevent_list& l, const npc_timerevent_list& r) { return l.timer < r.timer; } ); assert (it == nd->scr.timer_eventv.end() || it->timer != tel.timer); nd->scr.timer_eventv.insert(it, std::move(tel)); } } nd->scr.nexttimer = nd->scr.timer_eventv.end(); // nd->scr.timerid = nullptr; return 0; } /*========================================== * function行解析 *------------------------------------------ */ static int npc_parse_function(XString, XString, XString w3, ZString, XString first_line, FILE *fp, int *lines) { MString srcbuf; srcbuf += first_line.xislice_t(std::find(first_line.begin(), first_line.end(), '{')); int startline = *lines; while (true) { auto it = std::find_if_not(srcbuf.rbegin(), srcbuf.rend(), [](char c){ return c == ' ' || c == '\n'; }); if (it != srcbuf.rend() && *it == '}') break; char line_[1024]; if (!fgets(line_, 1020, fp)) break; (*lines)++; if (feof(fp)) break; ZString line(ZString::really_construct_from_a_pointer, line_, nullptr); if (!srcbuf) { srcbuf += line.xislice_t(std::find(line.begin(), line.end(), '{')); startline = *lines; } else srcbuf += line; } std::unique_ptr script = parse_script(FString(srcbuf), startline); if (script == NULL) { // script parse error? return 1; } userfunc_db.put(w3, std::move(script)); return 0; } /*========================================== * mob行解析 *------------------------------------------ */ static int npc_parse_mob(XString w1, XString, MobName w3, ZString w4) { int x, y, xs, ys, mob_class, num; int i; MapName mapname; NpcEvent eventname; dumb_ptr md; xs = ys = 0; int delay1_ = 0, delay2_ = 0; if (!extract(w1, record<',', 3>(&mapname, &x, &y, &xs, &ys)) || !extract(w4, record<',', 2>(&mob_class, &num, &delay1_, &delay2_, &eventname))) { PRINTF("bad monster line : %s\n", w3); return 1; } interval_t delay1 = std::chrono::milliseconds(delay1_); interval_t delay2 = std::chrono::milliseconds(delay2_); map_local *m = map_mapname2mapid(mapname); if (num > 1 && battle_config.mob_count_rate != 100) { if ((num = num * battle_config.mob_count_rate / 100) < 1) num = 1; } for (i = 0; i < num; i++) { md.new_(); md->bl_prev = NULL; md->bl_next = NULL; md->bl_m = m; md->bl_x = x; md->bl_y = y; if (w3 == ENGLISH_NAME) md->name = mob_db[mob_class].name; else if (w3 == JAPANESE_NAME) md->name = mob_db[mob_class].jname; else md->name = w3; md->n = i; md->mob_class = mob_class; md->bl_id = npc_get_new_npc_id(); md->spawn.m = m; md->spawn.x0 = x; md->spawn.y0 = y; md->spawn.xs = xs; md->spawn.ys = ys; md->spawn.delay1 = delay1; md->spawn.delay2 = delay2; really_memzero_this(&md->state); // md->timer = nullptr; md->target_id = 0; md->attacked_id = 0; md->lootitemv.clear(); md->npc_event = eventname; md->bl_type = BL::MOB; map_addiddb(md); mob_spawn(md->bl_id); npc_mob++; } //PRINTF("warp npc %s %d read done\n",mapname,nd->bl_id); return 0; } /*========================================== * マップフラグ行の解析 *------------------------------------------ */ static int npc_parse_mapflag(XString w1, XString, XString w3, ZString w4) { MapName mapname, savemap; int savex, savey; mapname = stringish(w1); if (!mapname) return 1; map_local *m = map_mapname2mapid(mapname); if (m == nullptr) return 1; if (w3 == "nosave") { if (w4 == "SavePoint") { m->save.map_ = stringish("SavePoint"); m->save.x = -1; m->save.y = -1; } else if (extract(w4, record<','>(&savemap, &savex, &savey))) { m->save.map_ = savemap; m->save.x = savex; m->save.y = savey; } m->flag.nosave = 1; } else if (w3 == "nomemo") { m->flag.nomemo = 1; } else if (w3 == "noteleport") { m->flag.noteleport = 1; } else if (w3 == "nowarp") { m->flag.nowarp = 1; } else if (w3 == "nowarpto") { m->flag.nowarpto = 1; } else if (w3 == "noreturn") { m->flag.noreturn = 1; } else if (w3 == "monster_noteleport") { m->flag.monster_noteleport = 1; } else if (w3 == "nobranch") { m->flag.nobranch = 1; } else if (w3 == "nopenalty") { m->flag.nopenalty = 1; } else if (w3 == "pvp") { m->flag.pvp = 1; } else if (w3 == "pvp_noparty") { m->flag.pvp_noparty = 1; } else if (w3 == "pvp_nocalcrank") { m->flag.pvp_nocalcrank = 1; } else if (w3 == "nozenypenalty") { m->flag.nozenypenalty = 1; } else if (w3 == "notrade") { m->flag.notrade = 1; } else if (battle_config.pk_mode && w3 == "nopvp") { // nopvp for pk mode [Valaris] m->flag.nopvp = 1; m->flag.pvp = 0; } else if (w3 == "noicewall") { // noicewall [Valaris] m->flag.noicewall = 1; } else if (w3 == "snow") { // snow [Valaris] m->flag.snow = 1; } else if (w3 == "fog") { // fog [Valaris] m->flag.fog = 1; } else if (w3 == "sakura") { // sakura [Valaris] m->flag.sakura = 1; } else if (w3 == "leaves") { // leaves [Valaris] m->flag.leaves = 1; } else if (w3 == "rain") { // rain [Valaris] m->flag.rain = 1; } else if (w3 == "no_player_drops") { // no player drops [Jaxad0127] m->flag.no_player_drops = 1; } else if (w3 == "town") { // town/safe zone [remoitnane] m->flag.town = 1; } return 0; } dumb_ptr npc_spawn_text(map_local *m, int x, int y, int npc_class, NpcName name, FString message) { dumb_ptr retval; retval.new_(); retval->bl_id = npc_get_new_npc_id(); retval->bl_x = x; retval->bl_y = y; retval->bl_m = m; retval->bl_type = BL::NPC; retval->npc_subtype = NpcSubtype::MESSAGE; retval->name = name; if (message) retval->message = message; retval->npc_class = npc_class; retval->speed = std::chrono::milliseconds(200); clif_spawnnpc(retval); map_addblock(retval); map_addiddb(retval); if (retval->name && retval->name[0]) npcname_db.put(retval->name, retval); return retval; } static void npc_free_internal(dumb_ptr nd_) { if (nd_->npc_subtype == NpcSubtype::SCRIPT) { dumb_ptr nd = nd_->as_script(); nd->scr.timer_eventv.clear(); { nd->scr.script.reset(); nd->scr.label_listv.clear(); } } else if (nd_->npc_subtype == NpcSubtype::MESSAGE) { dumb_ptr nd = nd_->as_message(); nd->message = FString(); } nd_.delete_(); } static void npc_propagate_update(dumb_ptr nd) { int xs = 0, ys = 0; if (dumb_ptr nd_ = nd->is_script()) { xs = nd_->scr.xs; ys = nd_->scr.ys; } map_foreachinarea(std::bind(npc_enable_sub, ph::_1, nd), nd->bl_m, nd->bl_x - xs, nd->bl_y - ys, nd->bl_x + xs, nd->bl_y + ys, BL::PC); } void npc_free(dumb_ptr nd) { clif_clearchar(nd, BeingRemoveWhy::GONE); npc_propagate_update(nd); map_deliddb(nd); map_delblock(nd); npc_free_internal(nd); } /*========================================== * npc初期化 *------------------------------------------ */ int do_init_npc(void) { // other fields unused ev_tm_b.tm_min = -1; ev_tm_b.tm_hour = -1; ev_tm_b.tm_mday = -1; for (; !npc_srcs.empty(); npc_srcs.pop_front()) { FString nsl = npc_srcs.front(); FILE *fp = fopen(nsl.c_str(), "r"); if (fp == NULL) { PRINTF("file not found : %s\n", nsl); exit(1); } int lines = 0; char line_[1024]; while (fgets(line_, 1020, fp)) { // because it's still fgets line_[strlen(line_) - 1] = '\0'; ZString zline(ZString::really_construct_from_a_pointer, line_, nullptr); XString w1, w2, w3, w4x; ZString w4z; lines++; if (!zline) continue; if (zline.startswith("//")) continue; if (!extract(zline, record<'|', 3>(&w1, &w2, &w3, &w4x)) || !w1 || !w2 || !w3) { FPRINTF(stderr, "%s:%d: Broken script line: %s\n", nsl, lines, zline); continue; } if (&*w4x.end() == &*zline.end()) { w4z = zline.xrslice_t(w4x.size()); } assert(bool(w4x) == bool(w4z)); if (w1 != "-" && w1 != "function") { auto comma = std::find(w1.begin(), w1.end(), ','); MapName mapname = stringish(w1.xislice_h(comma)); map_local *m = map_mapname2mapid(mapname); if (m == nullptr) { // "mapname" is not assigned to this server FPRINTF(stderr, "%s:%d: Map not found: %s\n", nsl, lines, mapname); continue; } } if (w2 == "warp") { NpcName npcname = stringish(w3); npc_parse_warp(w1, w2, npcname, w4z); } else if (w2 == "shop") { NpcName npcname = stringish(w3); npc_parse_shop(w1, w2, npcname, w4z); } else if (w2 == "script") { if (w1 == "function") { npc_parse_function(w1, w2, w3, w4z, w4x, fp, &lines); } else { NpcName npcname = stringish(w3); npc_parse_script(w1, w2, npcname, w4z, w4x, fp, &lines); } } else if (w2 == "monster") { MobName mobname = stringish(w3); npc_parse_mob(w1, w2, mobname, w4z); } else if (w2 == "mapflag") { npc_parse_mapflag(w1, w2, w3, w4z); } else { PRINTF("odd script line: %s\n", zline); } } fclose(fp); PRINTF("\rLoading NPCs [%d]: %-54s", npc_id - START_NPC_NUM, nsl); fflush(stdout); } PRINTF("\rNPCs Loaded: %d [Warps:%d Shops:%d Scripts:%d Mobs:%d] %20s\n", npc_id - START_NPC_NUM, npc_warp, npc_shop, npc_script, npc_mob, ""); if (script_errors) { PRINTF("Cowardly refusing to continue after %d errors\n", script_errors); abort(); } return 0; }