Go Back   EQEmulator Home > EQEmulator Forums > Archives > Archive::Development > Archive::Development

Archive::Development Archive area for Development's posts that were moved here after an inactivity period of 90 days.

Reply
 
Thread Tools Display Modes
  #1  
Old 01-09-2004, 02:21 PM
Valdain
Fire Beetle
 
Join Date: Feb 2003
Location: WI
Posts: 20
Default Fix for mobs and pet movement - Updated

I've been messing around with the code to make mobs move smoothly instead of jumping on top of you. I re-wrote portions of the Mob::CalculateNewPosition to use a more refined approach to moving. Also, I changed all references to CalculateNewPosition2 to use CalculateNewPosition.

This was developed and tested on 0.5.3 DR2 code.

In MobAI.cpp, just drop in this code to replace the old function:
Code:
bool Mob::CalculateNewPosition(float x, float y, float z, float speed) { 
   float ratio; 
   float target_distance;

   if(GetID()==0) 
      return true; 

    // if NPC is rooted 
    if (speed == 0.0) { 
        SetHeading(CalculateHeadingToTarget(x, y)); 
      if(moved){ 
         SendPosition(); 
         SetMoving(false); 
         moved=false; 
      } 
      SetRunAnimSpeed(0); 
        return true; 
    } 

   float nx = this->x_pos; 
   float ny = this->y_pos; 
   float nz = this->z_pos; 
   float nh = this->heading; 

   tarx=x; 
   tary=y; 
   tarz=z; 
    
	tar_vx = x - nx; 
    tar_vy = y - ny; 
    tar_vz = z - nz; 

      //If we are at our destination, then we don't need move!
	  if (tar_vx == 0 && tar_vy == 0 && tar_vz==0)
         return false; 

      pRunAnimSpeed = (sint8)(speed*30); 
	  
	  speed *= 2.0f; //The 2.0 is adjustable to get the speed just right.

      target_distance = sqrt (tar_vx*tar_vx + tar_vy*tar_vy + tar_vz*tar_vz); 
      heading = CalculateHeadingToTarget(x, y); 

  	  if(target_distance >= speed)  // If we need to go father than we can in one cycle... 
      { 
	     ratio = (speed/target_distance);
         x_pos += tar_vx * ratio;
         y_pos += tar_vy * ratio;
 		 z_pos += tar_vz * ratio;
	  } 
      else //If we are with-in our one cycle distance... 
      { 
         x_pos = x; 
         y_pos = y; 
         z_pos = z;  
      } 
       
      //OP_MobUpdate 
      APPLAYER* outapp = NULL; 
      PlayerPositionUpdateServer_Struct* spu = NULL; 
      this->SetMoving(true); 
      moved=true; 
      outapp = new APPLAYER(OP_ClientUpdate, sizeof(PlayerPositionUpdateServer_Struct)); 
      spu = (PlayerPositionUpdateServer_Struct*)outapp->pBuffer; 
      delta_x=x_pos-nx; 
      delta_y=y_pos-ny; 
      delta_z=z_pos-nz; 
      delta_heading=heading-nh; 
      MakeSpawnUpdate(spu); 
      spu->heading*=8; 
      entity_list.QueueCloseClients(this, outapp, true, 300); 
      safe_delete(outapp); 
 
    SetAppearance(0, false);
    pLastChange = Timer::GetCurrentTime(); 
    return true; 
}
Another quick fix helps smooth out pet movement.

In MobAI.cpp, in the Mob::AI_Process function around line 892
Code:
		else if (AImovement_timer->Check() && !IsRooted()) {
------------>>>>>    //SetRunAnimSpeed(0);
			if (GetOwnerID()) {
				// we're a pet, do as we're told
By commenting out the SetRunAnimSpeed(0), the pet seems to move a little smoother, don't know why.

UPDATE - This is probably as good as it will get. Until the server can know the Z coordinate for each X,Y (or at least be pretty close), the mob will not handle large vertical changes between 2 way points that are far apart, such as steep stairs. My suggestion is to make a way point at the top and bottom of such things as stairs and hills.

I've tested the code out on Windows with great success. There is still some refining that needs to be done, but this gets us alot closer to properly moving creatures.

-Valdain

[Edit] - Fixed code after talking with Doodman
Reply With Quote
  #2  
Old 01-10-2004, 12:03 AM
Scorpious2k's Avatar
Scorpious2k
Demi-God
 
Join Date: Mar 2003
Location: USA
Posts: 1,067
Default

I'm going to give this a try on the Scorpious2k Server wityh a couple of exceptions:

The Z coord calculation is important. In hilly areas or where a dest is lower than the start position you will find NPCs walking on air, This is not good. And if the dest is lower, either the NPC moves underground or the client goes nuts resetting the position to the surface. For these reasons, I uncommented the line where the Z values are calculated,

My other change for our purposes was to change Mob::CalculateNewPosition2 to
Code:
	return CalculateNewPosition(x, y, z, speed);
for testing purposes rather then change all ocurrences of it. We make mods here with #ifdef... #endif so we can unplug them easily if there are problems... I do a lot of testing. Usually on our miniserver before the live one.

We will be trying this today.
__________________
Maybe I should try making one of these servers...
Reply With Quote
  #3  
Old 01-10-2004, 03:31 AM
Valdain
Fire Beetle
 
Join Date: Feb 2003
Location: WI
Posts: 20
Default

I noticed in my testing that "gravity" is applied to the mob elsewhere and you don't get floating or "buried" symptoms on the client. Let me know how adding the Z calcs back in work for you. I'm just getting familiar with the code base again, and there have been lots of updates since a year ago.

As for the function naming, it really isn't consistant in the code between CalculateNewPosition and CalculateNewPosition2, but the code is the same in both. My main goal was just to improve the function. Feel free to adjust to however needed. I know most coders will #ifdef to allow for easier testing.

Let me know how it goes for you and thanks for taking a look at it!
-Valdain
Reply With Quote
  #4  
Old 01-10-2004, 03:34 AM
Scorpious2k's Avatar
Scorpious2k
Demi-God
 
Join Date: Mar 2003
Location: USA
Posts: 1,067
Default

Quote:
Originally Posted by Valdain
As for the function naming, it really isn't consistant in the code between CalculateNewPosition and CalculateNewPosition2, but the code is the same in both.
I noticed that too. For testing purposes, it was just easier to do it the way I did.

The change is live on our server. Feel free to drop in and check it out. That goes for anyone else who may be curious about the this change.
__________________
Maybe I should try making one of these servers...
Reply With Quote
  #5  
Old 01-10-2004, 04:02 AM
Valdain
Fire Beetle
 
Join Date: Feb 2003
Location: WI
Posts: 20
Default

I notice with the Z calcs back in the mob will do a little fall every once in a while. The right way to fix this is to determine what the Z coord is at the new position and set it that way, but I have no idea how to do that.

To see what I mean, attack a mob, then walk backwards away from it. It seems to be most prenounced when it is chasing you uphill.

-Valdain
Reply With Quote
  #6  
Old 01-12-2004, 02:54 PM
Doodman's Avatar
Doodman
Developer
 
Join Date: Aug 2003
Posts: 246
Default Re: Fix for "jumpy" mobs and pets - Updated

One note:

Code:
...
...
      ratio = (speed/target_distance);

   	  if(target_distance >= speed)  // If we need to go father than we can in one cycle... 
      { 
         x_pos += tar_vx * ratio;
         y_pos += tar_vy * ratio;
		 if(tar_vz>2)
			z_pos += tar_vz * ratio;
		 else
			z_pos = z;
      } 
      else //If we are with-in our one cycle distance... 
      { 
         x_pos = x; 
         y_pos = y; 
         z_pos = z;  
      } 
...
...
Should be:
Code:
...
...
   	  if(target_distance >= speed)  // If we need to go father than we can in one cycle... 
      { 
         ratio = (speed/target_distance);

         x_pos += tar_vx * ratio;
         y_pos += tar_vy * ratio;
		 if(tar_vz>2)
			z_pos += tar_vz * ratio;
		 else
			z_pos = z;
      } 
      else //If we are with-in our one cycle distance... 
      { 
         x_pos = x; 
         y_pos = y; 
         z_pos = z;  
      } 
...
...
Since ratio is only used in there, no point in wasting cycles on the division.
Reply With Quote
  #7  
Old 01-12-2004, 03:07 PM
Doodman's Avatar
Doodman
Developer
 
Join Date: Aug 2003
Posts: 246
Default Re: Fix for "jumpy" mobs and pets - Updated

Quote:
Originally Posted by Valdain
[Edit] - Minor update to handle rare cases where very small distances messed up the function.
Describe "messed up". All that * and / is expensive in there. So, it'll have a big impact.
Reply With Quote
  #8  
Old 01-18-2004, 02:26 AM
Armanthuz
Sarnak
 
Join Date: Apr 2003
Posts: 72
Default

This seemed to work very well for me...


One small problem i noticed thatmay or may not be related is lets say i fight a mob then mem blur him, when mob walks back to spawn spot, i can see him walk about 5-10 paces, the mob usually keeps walking right past spawn spot then dissapears until next reboot.
Reply With Quote
  #9  
Old 01-18-2004, 05:02 AM
Trumpcard
Demi-God
 
Join Date: Jan 2002
Location: Charlotte, NC
Posts: 2,614
Default

Scorp, since you applied this to your server and checked it out, do you mind investigating merging it in?
__________________
Quitters never win, and winners never quit, but those who never win and never quit are idiots.
Reply With Quote
  #10  
Old 01-18-2004, 07:18 AM
Scorpious2k's Avatar
Scorpious2k
Demi-God
 
Join Date: Mar 2003
Location: USA
Posts: 1,067
Default

Quote:
Originally Posted by Trumpcard
Scorp, since you applied this to your server and checked it out, do you mind investigating merging it in?
Will do.
__________________
Maybe I should try making one of these servers...
Reply With Quote
  #11  
Old 01-19-2004, 02:51 AM
Scorpious2k's Avatar
Scorpious2k
Demi-God
 
Join Date: Mar 2003
Location: USA
Posts: 1,067
Default

One little snag....

The original code this was designed to replace has been redone. Mob::CalculateNewPosition and Mob::CalculateNewPosition2 are no longer identical. Major changes have been made while this discussion was ocurring, making a direct merge of the new code impossible.

It looks to me like many of the ideas have been applied, although not as a merge of the code. Here is what we have now:

Code:
bool Mob::CalculateNewPosition(float x, float y, float z, float speed) {
	if(GetID()==0)
		return true;
	/*if(tar_ndx<20 && tarx==x && tary==y && tarz==z){
		tar_ndx++;
		x_pos = x_pos + tar_vx*tar_vector;
		y_pos = y_pos + tar_vy*tar_vector;
		z_pos = z_pos + tar_vz*tar_vector;
		return true;
	}
	else{
		tar_ndx=0;
		tarx=x;
		tary=y;
		tarz=z;
	}*/
    float nx = x_pos;
    float ny = y_pos;
    float nz = z_pos;
	float nh = heading;
	
    // if NPC is rooted
    if (speed == 0.0) {
        SetHeading(CalculateHeadingToTarget(x, y));
		if(moved){
			SendPosition();
			SetMoving(false);
			moved=false;
		}
		SetRunAnimSpeed(0);
        return true;
    }
		float old_test_vector=test_vector;
		tar_vx = x - nx;
		tar_vy = y - ny;
		tar_vz = z - nz;

		if (tar_vx == 0 && tar_vy == 0 && tar_vz == 0)
			return false;
		pRunAnimSpeed = (int8)(speed*43);
		speed *= 2.8;
		// --------------------------------------------------------------------------
		// 2: get unit vector
		// --------------------------------------------------------------------------
		test_vector=sqrt (x*x + y*y + z*z);
		tar_vector = speed / sqrt (tar_vx*tar_vx + tar_vy*tar_vy + tar_vz*tar_vz);
		heading = CalculateHeadingToTarget(x, y);

		if (tar_vector >= 1.0) {
			x_pos = x;
			y_pos = y;
			z_pos = z;
		}
		else {
			x_pos = x_pos + tar_vx*tar_vector;
			y_pos = y_pos + tar_vy*tar_vector;
			z_pos = z_pos + tar_vz*tar_vector;
		}
		//OP_MobUpdate
		if((old_test_vector!=test_vector) || tar_ndx>20){ //send update
			tar_ndx=0;
			APPLAYER* outapp = NULL;
			PlayerPositionUpdateServer_Struct* ppu = NULL;
			this->SetMoving(true);
			moved=true;
			outapp = new APPLAYER(OP_ClientUpdate, sizeof(PlayerPositionUpdateServer_Struct));
			ppu = (PlayerPositionUpdateServer_Struct*)outapp->pBuffer;
			delta_x=(x_pos-nx);
			delta_y=(y_pos-ny);
			delta_z=(z_pos-nz);
			delta_heading=0;//(heading-nh)*8;
			MakeSpawnUpdate(ppu);
			ppu->heading*=8;
			entity_list.QueueCloseClients(this, outapp, true, 300);
			safe_delete(outapp);
		}
		tar_ndx++;
    // now get new heading
	SetAppearance(0, false); // make sure they're standing
    pLastChange = Timer::GetCurrentTime();
    return true;
}
and
Code:
bool Mob::CalculateNewPosition2(float x, float y, float z, float speed) {
	if(GetID()==0)
		return true;
	if(tar_ndx<20 && tarx==x && tary==y && tarz==z){
		x_pos = x_pos + tar_vx*tar_vector;
		y_pos = y_pos + tar_vy*tar_vector;
		z_pos = z_pos + tar_vz*tar_vector;
		tar_ndx++;
		return true;
	}
	else{
		tar_ndx=0;
		tarx=x;
		tary=y;
		tarz=z;
	}
	float nx = this->x_pos;
    float ny = this->y_pos;
    float nz = this->z_pos;
	float nh = this->heading;
	
	tar_vx = x - nx;
	tar_vy = y - ny;
	tar_vz = z - nz;

	if (tar_vx == 0 && tar_vy == 0 && tar_vz == 0)
		return false;
	pRunAnimSpeed = (sint8)(speed*36);
	speed *= 46;
	// --------------------------------------------------------------------------
	// 2: get unit vector
	// --------------------------------------------------------------------------
	tar_vector = speed / sqrt (tar_vx*tar_vx + tar_vy*tar_vy + tar_vz*tar_vz);
	heading = CalculateHeadingToTarget(x, y);

	if (tar_vector >= 1.0) {
		x_pos = x;
		y_pos = y;
		z_pos = z;
	}
	else {
		tar_vector/=20;
		x_pos = x_pos + tar_vx*tar_vector;
		y_pos = y_pos + tar_vy*tar_vector;
		z_pos = z_pos + tar_vz*tar_vector;
	}
	APPLAYER* outapp = NULL;
	PlayerPositionUpdateServer_Struct* spu = NULL;
	this->SetMoving(true);
	moved=true;
	outapp = new APPLAYER(OP_ClientUpdate, sizeof(PlayerPositionUpdateServer_Struct));
	spu = (PlayerPositionUpdateServer_Struct*)outapp->pBuffer;
	delta_x=x_pos-nx;
	delta_y=y_pos-ny;
	delta_z=z_pos-nz;
	delta_heading=heading-nh;
	MakeSpawnUpdate(spu);
	spu->heading*=8;
	entity_list.QueueCloseClients(this, outapp, true, 300);
	safe_delete(outapp);
	SetAppearance(0, false);
    pLastChange = Timer::GetCurrentTime();
    return true;
}
I think you can see a lot of the same ideas being applied. One interesting addition is the supression of update packets here, which should help.

So, should we try to merge in the logic from each version? Merge the code anyway? Test both versions and pick the best?

Opinions?
__________________
Maybe I should try making one of these servers...
Reply With Quote
Reply


Posting Rules
You may not post new threads
You may not post replies
You may not post attachments
You may not edit your posts

BB code is On
Smilies are On
[IMG] code is On
HTML code is Off

Forum Jump

   

All times are GMT -4. The time now is 04:55 AM.


 

Everquest is a registered trademark of Daybreak Game Company LLC.
EQEmulator is not associated or affiliated in any way with Daybreak Game Company LLC.
Except where otherwise noted, this site is licensed under a Creative Commons License.
       
Powered by vBulletin®, Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
Template by Bluepearl Design and vBulletin Templates - Ver3.3