diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/map/mob.c | 63 | ||||
-rw-r--r-- | src/map/mob.h | 2 | ||||
-rw-r--r-- | src/map/unit.c | 11 |
3 files changed, 58 insertions, 18 deletions
diff --git a/src/map/mob.c b/src/map/mob.c index 7de4fd50a..86d60a152 100644 --- a/src/map/mob.c +++ b/src/map/mob.c @@ -875,19 +875,32 @@ static int mob_ai_sub_hard_lootsearch(struct block_list *bl,va_list ap) return 0; } -static int mob_ai_sub_hard_warpsearch(struct block_list *bl,va_list ap) +static int mob_warpchase_sub(struct block_list *bl,va_list ap) { struct mob_data* md; - struct block_list **target; + struct block_list *target; + struct npc_data **target_nd; + struct npc_data *nd; + int *min_distance; + int cur_distance; md=va_arg(ap,struct mob_data *); - target= va_arg(ap,struct block_list**); - - if (*target) return 0; - - if(bl->subtype == WARP) - { - *target = bl; + target= va_arg(ap, struct block_list*); + target_nd= va_arg(ap, struct npc_data**); + min_distance= va_arg(ap, int*); + + if(bl->subtype != WARP) + return 0; //Not a warp + nd = (TBL_NPC*) bl; + + if(nd->u.warp.mapindex != map[target->m].index) + return 0; //Does not lead to the same map. + + cur_distance = distance_blxy(target, nd->u.warp.x, nd->u.warp.y); + if (cur_distance < *min_distance) + { //Pick warp that leads closest to target. + *target_nd = nd; + *min_distance = cur_distance; return 1; } return 0; @@ -1073,6 +1086,29 @@ int mob_randomwalk(struct mob_data *md,unsigned int tick) return 1; } +int mob_warpchase(struct mob_data *md, struct block_list *target) +{ + struct npc_data *warp = NULL; + int distance = AREA_SIZE; + if (!(target && battle_config.mob_ai&0x40 && battle_config.mob_warp&1)) + return 0; //Can't warp chase. + + if (target->m == md->bl.m && check_distance_bl(&md->bl, target, AREA_SIZE)) + return 0; //No need to do a warp chase. + + if (md->ud.walktimer != -1 && + map_getcell(md->bl.m,md->ud.to_x,md->ud.to_y,CELL_CHKNPC)) + return 1; //Already walking to a warp. + + //Search for warps within mob's viewing range. + map_foreachinrange (mob_warpchase_sub, &md->bl, + md->db->range2, BL_NPC, md, target, &warp, &distance); + + if (warp && unit_walktobl(&md->bl, &warp->bl, 0, 1)) + return 1; + return 0; +} + /*========================================== * AI of MOB whose is near a Player *------------------------------------------*/ @@ -1128,13 +1164,8 @@ static int mob_ai_sub_hard(struct block_list *bl,va_list ap) ((((TBL_PC*)tbl)->state.gangsterparadise && !(mode&MD_BOSS)) || ((TBL_PC*)tbl)->invincible_timer != INVALID_TIMER) )) { //Unlock current target. - if (tbl && tbl->m != md->bl.m && battle_config.mob_ai&0x40) - { //Chase to a nearby warp [Skotlex] - tbl = NULL; - map_foreachinrange (mob_ai_sub_hard_warpsearch, &md->bl, - view_range, BL_NPC, md, &tbl); - if (tbl) unit_walktobl(&md->bl, tbl, 0, 1); - } + if (mob_warpchase(md, tbl)) + return 0; //Chasing this target. mob_unlocktarget(md, tick-(battle_config.mob_ai&0x8?3000:0)); //Imediately do random walk. tbl = NULL; } diff --git a/src/map/mob.h b/src/map/mob.h index 93fe8225e..25a314d75 100644 --- a/src/map/mob.h +++ b/src/map/mob.h @@ -163,7 +163,7 @@ int mob_spawn_guardian(const char* mapname, short x, short y, const char* mobnam int mob_guardian_guildchange(struct block_list *bl,va_list ap); //Change Guardian's ownership. [Skotlex] int mob_randomwalk(struct mob_data *md,unsigned int tick); - +int mob_warpchase(struct mob_data *md, struct block_list *target); int mob_target(struct mob_data *md,struct block_list *bl,int dist); int mob_unlocktarget(struct mob_data *md,int tick); struct mob_data* mob_spawn_dataset(struct spawn_data *data); diff --git a/src/map/unit.c b/src/map/unit.c index b92804459..3befc3f2d 100644 --- a/src/map/unit.c +++ b/src/map/unit.c @@ -233,6 +233,8 @@ static int unit_walktoxy_timer(int tid,unsigned int tick,int id,int data) if (!tbl || !status_check_visibility(bl, tbl)) { //Cancel chase. ud->to_x = bl->x; ud->to_y = bl->y; + if (tbl && bl->type == BL_MOB) //See if the mob can do a warp chase. + mob_warpchase((TBL_MOB*)bl, tbl); return 0; } if (tbl->m == bl->m && check_distance_bl(bl, tbl, ud->chaserange)) @@ -1367,9 +1369,16 @@ static int unit_attack_timer_sub(struct block_list* src, int tid, unsigned int t if(src == NULL || src->prev == NULL || target==NULL || target->prev == NULL) return 0; - if(src->m != target->m || status_isdead(src) || status_isdead(target) || !status_check_skilluse(src, target, 0, 0)) + if(status_isdead(src) || status_isdead(target) || !status_check_skilluse(src, target, 0, 0)) return 0; // can't attack under these conditions + if (src->m != target->m) + { + if (src->type == BL_MOB && mob_warpchase((TBL_MOB*)src, target)) + return 1; // Follow up. + return 0; + } + if(ud->skilltimer != -1 && !(sd && pc_checkskill(sd,SA_FREECAST) > 0)) return 0; // can't attack while casting |