Page 1 of 2
					
				Weld one entity to another on collision
				Posted: Sun Sep 09, 2012 2:14 am
				by Telanor
				Am I missing something, or is there no way to get the Entity associated with the collidable passed in from the PairTouched event?  I'm trying to make arrows weld to whatever they hit, but the Weld constraint wants 2 entities.  Do I have to put the entity in the tag for all the collidables or is there a better way?  Also, how can I make it so an arrow penetrates the collidable a little bit so it doesnt just stick to the surface?
			 
			
					
				Re: Weld one entity to another on collision
				Posted: Sun Sep 09, 2012 3:03 am
				by Norbo
				Am I missing something, or is there no way to get the Entity associated with the collidable passed in from the PairTouched event? I'm trying to make arrows weld to whatever they hit, but the Weld constraint wants 2 entities. Do I have to put the entity in the tag for all the collidables or is there a better way?
EntityCollidable is a subclass of Collidable. If a cast attempt succeeds, then you can use the EntityCollidable.Entity property to grab the entity associated with the entity.
Edit:
CollidablePairHandlers also expose two entity properties. Note that these can return null if there's no entity associated with the collidable.
Also, how can I make it so an arrow penetrates the collidable a little bit so it doesnt just stick to the surface?
Scoot the arrow entity forward a bit upon impact before welding it. "Forward" in this case depends on how the arrow is oriented in local space. Transform the direction the arrow points in local space by the orientation of the entity to get the world space forward vector to scoot along. If it's one of the three primary axes of the orthonormal basis defining the orientation, you can just grab it straight out of the entity.OrientationMatrix.Forward etc. properties.
When doing this, collision rules should be set to prevent the arrow from undergoing collision response with the hit object.
If this is a purely graphical effect, I'd recommend removing the arrow from the simulation and just using a purely graphical proxy. It will be quite a bit more efficient.
 
			
					
				Re: Weld one entity to another on collision
				Posted: Mon Sep 10, 2012 5:45 am
				by Telanor
				Ah.  I knew there had to be a better way.  It is a purely graphical effect, but if it attaches to a moving entity, I'd need some way to make it move with that entity.  I'm having 2 other issues though.  When the arrow collides with a surface, if I don't weld it, it bounces off and begins spinning wildly.  The problem is, if I do weld it, it still starts to rotate a bit before its stopped by the weld (or in the case when it collides with static terrain, I just remove the physics entity entirely).  What ends up happening is that the arrow gets stuck at weird angles:
			
		
				
			 
- RuinValor-2012-09-10-01-40-32-74.jpg (36.65 KiB) Viewed 12909 times
 
The second issue is minor but when you shoot the arrow up (it doesn't even have to be shot straight up), it comes back down with the arrow head facing upwards.  Ive tried constructing the arrow as a compound object made of 2 boxes, with the arrowhead one weighing more, but it didn't help.
 
			
					
				Re: Weld one entity to another on collision
				Posted: Mon Sep 10, 2012 5:09 pm
				by Norbo
				 It is a purely graphical effect, but if it attaches to a moving entity, I'd need some way to make it move with that entity.
On impact, compute the local space transform of the arrow relative to the entity. To find where to put the arrow graphic each frame, multiply the local space transform by the world transform of the entity. This will be infinitely rigid (compared to a merely mostly rigid weld), and it will be a lot faster.
When the arrow collides with a surface, if I don't weld it, it bounces off and begins spinning wildly. The problem is, if I do weld it, it still starts to rotate a bit before its stopped by the weld (or in the case when it collides with static terrain, I just remove the physics entity entirely). What ends up happening is that the arrow gets stuck at weird angles:
Assuming that the bounce is simply caused by the large impact forces plus the low arrow inertia as opposed to some problem like the collision shape not matching the graphic, this could be solved by making the arrow not respond to collisions with anything. It just generates contacts. So, you'll know it hit something without it bouncing around. You may want to incorporate your own form of collision response for the arrows to stop them from continuing to fly through the obstacle during the end-of-timestep position update. This could be done by computing the relative linear velocity between the arrow and the impacted entity (if any), and then computing the component of that relative linear velocity which is along the direction of motion of the arrow. That component is then subtracted out of the arrow's linear velocity in one of the immediate events so that the change is visible by the solver and position update.
The second issue is minor but when you shoot the arrow up (it doesn't even have to be shot straight up), it comes back down with the arrow head facing upwards. Ive tried constructing the arrow as a compound object made of 2 boxes, with the arrowhead one weighing more, but it didn't help.
One option would be to control the orientation of the arrow based upon the linear velocity. Align the local forward vector of the arrow with the linear velocity direction. Beware of the zero-velocity singularity.
 
			
					
				Re: Weld one entity to another on collision
				Posted: Tue Sep 11, 2012 6:40 am
				by Telanor
				Ok, so I turned off the solver for the arrow and when it collides, I set its linear velocity to zero.  This mostly works, but sometimes the arrows go through the ground or at least penetrate it pretty far.  I assume this is exactly what you were talking about with the custom collision response, however I'm not sure I understand your solution.  You said calculate the relative velocity between the arrow and its target, compute the amount along the motion vector and subtract that from the linear velocity.  If the target isn't moving, wouldn't the result just be the negative of the linear velocity?   If that's the case you'd simply end up with zero velocity, which is what I've already done...  I'm doing all of this inside the PairTouching event, is that the correct event to use?
As for the orientation trick, that worked perfectly, thanks.
			 
			
					
				Re: Weld one entity to another on collision
				Posted: Tue Sep 11, 2012 4:33 pm
				by Norbo
				You said calculate the relative velocity between the arrow and its target, compute the amount along the motion vector and subtract that from the linear velocity. If the target isn't moving, wouldn't the result just be the negative of the linear velocity? If that's the case you'd simply end up with zero velocity, which is what I've already done... I'm doing all of this inside the PairTouching event, is that the correct event to use?
That's all correct. DetectingInitialCollision might be a bit more direct than PairTouching, but in this case they should be functionally equivalent.
If the arrow is moving fast relative to the time step duration, then the arrow could still penetrate pretty far into the target even with full collision response. This is because the engine takes discrete steps forward in time. To mitigate this, you can set the arrow's entity.PositionUpdateMode = PositionUpdateMode.Continuous. This will prevent full tunneling (except in the case of significant angular motion), but the object could still go through a little less than halfway because the primary form of motion clamping CCD is applied between a core shape and the other object (and vice versa).
However, by default, the MotionSettings.CCDFilter will avoid testing any pair of objects which don't undergo collision response. This would need to be modified. The default filter looks like this:
Code: Select all
        static bool DefaultCCDFilter(Entity entity, Collidable other, CollidablePairHandler pair)
        {
            return pair.broadPhaseOverlap.collisionRule < CollisionRule.NoSolver;
        }
Making an allowance for particular pairs involving an arrow or simply returning true would allow it to run.
 
			
					
				Re: Weld one entity to another on collision
				Posted: Wed Sep 12, 2012 12:37 am
				by Telanor
				Modifying the CCDFilter to return true for arrows fixed it.  Now the arrows stop exactly on the surface every time.  Everything is working quite nicely now, thanks for the help!
			 
			
					
				Re: Weld one entity to another on collision
				Posted: Sun Jan 13, 2013 8:06 am
				by Mokgra
				Norbo mentioned removing the arrow's Box from the Space as a simple solution.  I'm attempting that...  
I am hooked onto the arrow's InitialCollisionDetected event.  When that fires I do space.Remove(sender.Entity).
This works fine for a while, but if I get going shooting arrows rapid style I get an exception:
"The object does not belong to this space; cannot remove it."
What else would I need to do to prevent this from happening?
			 
			
					
				Re: Weld one entity to another on collision
				Posted: Sun Jan 13, 2013 8:25 am
				by Telanor
				I get that a lot.  Make sure that you're using Space.SpaceObjectBuffer.Remove and that you don't try to remove the same object multiple times
			 
			
					
				Re: Weld one entity to another on collision
				Posted: Sun Jan 13, 2013 8:47 am
				by Norbo
				Removing objects multiple times is indeed the cause. Since InitialCollisionDetected is deferred, the object could accumulate quite a few instances of the event due to multiple collisions in one frame. Trying to remove the object from each event will fail on the second attempt. This is more noticeable when firing many arrows since individual objects are more likely to have more than one event in one frame.
[In contrast to deferred events, immediate events execute in-line and would technically allow you to call Space.Remove before the next event ran, but it would fail due to asynchronous access and bookkeeping corruption. Removing objects from beneath the feet of the system while it's running does not work well 

 Space.SpaceObjectBuffer.Remove allows you to safely enqueue an object for removal from an immediate event, though later immediate events will still execute so this is similar to using deferred events.]
 
			
					
				Re: Weld one entity to another on collision
				Posted: Mon Jan 14, 2013 10:37 pm
				by Mokgra
				Another solution I found was to check if the entity.space property was null. Is there any snafus with that (other than efficiency and extra events firing)?
So I can remove the entity from just the Space.SpaceObjectBuffer? Or do I need to do both Space and Space.SpaceObjectBuffer?
			 
			
					
				Re: Weld one entity to another on collision
				Posted: Mon Jan 14, 2013 10:48 pm
				by Norbo
				Another solution I found was to check if the entity.space property was null. Is there any snafus with that (other than efficiency and extra events firing)?
That'll work fine.
So I can remove the entity from just the Space.SpaceObjectBuffer? Or do I need to do both Space and Space.SpaceObjectBuffer?
Adding/removing with the Space.SpaceObjectBuffer just enqueues and object for addition to or removal from the Space later on. It's just deferring the call to Space.Add or Space.Remove to a time when it is safe to do so (the beginning of the next time step); that's why Space.SpaceObjectBuffer is only needed in dangerous situations like asynchronous access. So, it is unnecessary and incorrect to call both Space.Remove and Space.SpaceObjectBuffer.Remove, because that would end up calling Space.Remove twice.
 
			
					
				Re: Weld one entity to another on collision
				Posted: Wed Jan 23, 2013 7:07 am
				by Mokgra
				One option would be to control the orientation of the arrow based upon the linear velocity. Align the local forward vector of the arrow with the linear velocity direction. Beware of the zero-velocity singularity.
How would I go about doing this exactly?   I've tried calculating a set of angle for orientation based on the linear velocity, but I can't seem to get the math right.  FYI the yaw for my arrow needs a (yaw - PI/2) modifier to start straight (pun intended).
 
			
					
				Re: Weld one entity to another on collision
				Posted: Wed Jan 23, 2013 7:32 am
				by Norbo
				The easiest way would be to just grab the Forward (or Right, or Left) vector from the entity's OrientationMatrix. There's a property for them. This is equivalent to taking something like (0,0,1) and transforming it by the rotation matrix, except because a rotation matrix is an orthonormal basis, you can just read the appropriate direction right out of the matrix's components without any extra math.
			 
			
					
				Re: Weld one entity to another on collision
				Posted: Wed Jan 23, 2013 8:13 am
				by Norbo
				Oops, the above post was a bit backward as I was doing three things at once 
 
To 
generate an orientation matrix from a direction, you have to decide on the remaining degree of freedom. Any method of creating a complete basis from one direction will do the trick. For example, taking the cross product of Vector3.Up and the DesiredForwardDirection gives you another axis in the basis (so long as the DesiredForwardDirection is not parallel to Vector3.Up, in which case you'd need to use Vector3.Right or something). Normalize that new axis, then take the cross product of it and the DesiredForwardDirection again to get the third axis. This set of three vectors defines an orthonormal basis, which is a rotation matrix. Note that the inverse of a rotation matrix is its transpose; if things end up doing the opposite of what you want, this is probably related.
Another less involved option would be to use Toolbox.GetQuaternionBetweenNormalizedVectors with a parameter of the current 'forward' direction of the entity and the other parameter of DesiredForwardDirection (from the velocity). Then, transform the current entity rotation by the change returned. (Beware the multiplication order!)