首页  编辑  

游戏中的边界碰撞

Tags: /超级猛料/Picture.图形图像编程/控件和绘图/   Date Created:

游戏中的边界碰撞

Author: Erwin Haantjes

http://www.swissdelphicenter.ch/torry/showcode.php?id=1791

detect simple collision and transparency?

(*

Simple collision detection and transparency, by Erwin Haantjes

DelphiApplicationDiscussion@hotmail.com

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

It's not uncommon that collision detection in games is one of the most called functions.

There are several ways of doing collision detection and this article describes a method

for 2D games using bounding boxes (rectangles).

Perfect collision detection down to the pixel is of course what (theoretically) is the best.

However, checking each pixel against each other for all objects every frame in a game

is slow, very slow. So a simplier way is needed, this is where bounding boxes comes in.

Imagine that you draw a rectangle, like a frame, around an object (the player spaceship

for examle). Then do the same thing for all the other objects in the game. Now calculate

if these rectangles intercept each other to determine a collision, instead of checking

each pixels. This method is much faster, not as precise, but usually exact enough.

+------------+

|      XX    |

|    XXXXXX  |

|XXXXXXXXXXXX|

|    XXXXXX+-|----------+

|      XX  |X|XX    XXXX|

+------------+ XX  XX   |

          |    XXXX    |

          |   XX  XX   |

          |XXXX    XXXX|

          +------------+

A problem is that sometimes a collision is detected when visually it seems like the object

didn't touch. This usually occurs when any of the objects is shaped like a sphere. To

compensate this we use an offset when checking if the bounding boxes intercept. By making

the bounding boxes slightly smaller than the object itself, detection usually seems accurate

enough. As mentioned before, we sacrifice precision for speed, but this is a way to compensate

the precision to make it "feel" more exact.

Here is the function which does the real collision detection of two objects. In this example

the player's coordinates are R1, while the enemy's coordinates are R2. You can use an

offset for the X and Y coordinates to precize the detection (especially when you using

shaped objects).

*)

function CheckBoundryCollision( R1, R2 : TRect; pR : PRect = nil; OffSetY : LongInt = 0; OffSetX : LongInt = 0): boolean; {overload;}

// Simple collision test based on rectangles.

begin

  // Rectangle R1 can be the rectangle of the character (player)

  // Rectangle R2 can be the rectangle of the enemy

 if( pR <> nil ) then

  begin

   // Simple collision test based on rectangles. We use here the

   // API function IntersectRect() which returns the intersection rectangle.

   with( R1 ) do

    R1:=Rect( Left+OffSetX, Top+OffSetY, Right-(OffSetX * 2), Bottom-(OffSetY * 2));

   with( R2 ) do

    R2:=Rect( Left+OffSetX, Top+OffSetY, Right-(OffSetX * 2), Bottom-(OffSetY * 2));

   Result:=IntersectRect( pR^, R1, R2 );

  end

 else begin

       // Simple collision test based on rectangles. We can also use the

       // API function IntersectRect() but i believe this is much faster.

       Result:=( NOT ((R1.Bottom - (OffSetY * 2) < R2.Top + OffSetY)

         or(R1.Top + OffSetY > R2.Bottom - (OffSetY * 2))

          or( R1.Right - (OffSetX * 2) < R2.Left + OffSetX)

           or( R1.Left + OffSetX > R2.Right - (OffSetX * 2))));

      end;

end;

(*

Also you can make an overloaded function with the same name based on handles (to make it easier).

*)

function CheckBoundryCollision( h1, h2 : Hwnd; pR : PRect = nil; OffSetY : LongInt = 0; OffSetX : LongInt = 0): boolean; {overload;}

var

R1 : TRect;

R2 : TRect;

begin

FillChar( R1, SizeOf( R1 ), 0 );

FillChar( R2, SizeOf( R2 ), 0 );

 // Note: You will get here the REAL screen coordinates.

 // It doesn't matter if a control is placed inside a panel or something.

GetWindowRect( h1, R1 );

GetWindowRect( h2, R2 );

Result:=CheckBoundryCollision( R1, R2, pR, OffsetY, OffSetX );

end;

As an example, the above function may be used in a (threated timer) loop like this to check the player against multiple enemies:

 for i := 0 to Length( Enemies ) - 1 do

  if( CheckCollision( Player.Handle, Enemies[i].Handle, nil, 4, 4 )) then

   GameOver;

(*

Bottom line is that usually it's enough if it "looks" real, and less important that

it "is" real. It's a balance between how exact we want it, and how fast we need it to perform.

I have used this methods in a simple game and it just works fine. Let's talk about transparency.

You can use Regions to create special shaped Wincontrols. For example: A TPanel with a TImage

on it (Panel1 and Image1). You can shape the panel to the image using the following examples:

*)

function  CreateRegion( Bitmap : TBitmap; Rect : TRect; TransColor : TColor ) : hRgn;

var

i1, Count   : LongInt;

x,y         : LongInt;

c1          : Cardinal;

c,t         : TColor;

Msg         : TMsg;

begin

Result:=0;

if( NOT Assigned( Bitmap )) then

 Exit;

Count :=0;

{if( TransColor = clAutomatic ) then

 }t:=Bitmap.Canvas.Pixels[ Rect.Left, Rect.Top ]

{else t:=TransColor};

with( Bitmap.canvas ) do

 for y := Rect.Top to Rect.Bottom do

   begin

     // Sort of the same like Application.ProcessMessages but doesn't require

     // the bulky Forms unit.

    PeekMessage( Msg, 0, 0, 0, PM_REMOVE );

    x:=Rect.Left;

    while( x <= Rect.Right ) do

     begin

      C:=Pixels[ x, y ];

      if( C <> t ) then

       begin

        i1 := x;

         while( C <> t ) do

          begin

           Inc(i1);

           C:=Pixels[ i1, y ];

           if( i1 >= Rect.Right ) then

            Break;

          end;

         c1:=CreateRectRgn( x-Rect.Left,y-Rect.Top, i1-Rect.Left, (y-Rect.Top) + 1 );

         if( count = 0 ) then

          Result:=c1

         else begin

               CombineRgn( Result, Result, c1, RGN_OR );

               DeleteObject( c1 );

              end;

      Inc( Count );

      x := i1;

      Continue;

     end;

     Inc(x);

   end;

 end;

if( Count = 0 ) and ( Result > 0 ) then

 begin

  DeleteObject( Result );

  Result:=0;

 end;

end;

// Put this somewhere in your code (for example in the FormCreate event)

begin

 Image1.Left:=0;

 Image1.Top:=0;

 Image1.AutoSize:=TRUE;

 Panel1.BevelInner:=bvNone;

 Panel1.BevelOuter:=bvNone;

 Panel1.Width:=Image1.Width;

 Panel1.Height:=Image1.Height;

  // Create a region of the bitmap that is inside the image and set the region for the window.

  // Note: In this example the transparent color is black. Make sure the transparent (background) color is

  //       black! To be sure that you using the right color, you can also try this: Image1.Picture.Bitmap.Pixels[ 0, 0 ]

 with( Panel1 ) do

  SetWindowRgn( Handle, CreateRegion( Image1.Picture.Bitmap, Rect( 1, 1, Width, Height ), clBlack ));

end;

 

(*

Note: When you move the panel it is possible that the bitmap starts to flicker. You can create an

     Message handler for the message WN_EREASEBKGND to avoid the flicker. See the help of Delphi how to

     create new components and/or message handlers if do not know. Here is an example:

     procedure TMyPanel.WMEraseBkgnd(var Message: TWmEraseBkgnd);

     begin

      if( csDesigning in ComponentState ) then

       inherited

      else Message.Result:=1; // Fake erase

     end;

*)

(*

NOTE: Absolutly no warranty to this code. This is just an example. It is possible that you need to make some

improvements to the code to give it a more proffesional behaviour.

This article was aimed towards beginners wanting get into game programming, I'll hope it was helpful.

Regards and good luck,

Erwin Haantjes.

*)