Simple Kreis/Rechteck kollisionsüberprüfung!



  • Hi,

    ich möchte eine möglichst sehr kleine Funktion haben für eine Kreis/Rechteck kollisionsabfrage!

    Also ich habe 3 Strukturen:

    struct rect
    {
        int x;
        int y;
        int width;  // X wird NICHT draufgerechnet!
        int height; // Y wird NICHT draufgerechnet!
    };
    
    struct circle
    {
        int x;
        int y;
        int r;
    };
    
    struct point
    {
        int x;
        int y;
    };
    

    Leider bietet die WinAPI oder der C++ Standard keine Funktion wie "bool intersectsCircleRect (point* schnittpunkt, const rect &rect_, const circle &circle_)" 😞

    Hat jemand so ne Funktion bzw. kann mir jemand so eine machen? (Sorry falls ich zu dreist bin aber ich hab 0 peil :()

    - Patrick



  • weist du, wie man eine kollision von linie mit kreis macht?
    kreisgleichung+gradengleichung sollten dein problem lösen.
    (welcher punkt der geraden erfüllt die kreisgleichung...)



  • Danke für den Tipp

    - Patrick



  • falls dich der genaue schnittpunkt nicht interessiert, und du jemanden erschlagen musst... geht auch das hier:

    inline int sqr(const int x){ return x*x; }
    bool intersectsCircleRect(const rect &rect_, const circle &circle_)
    {
      return(((int)(sqr(rect_.x-circle_.x)+sqr(rect_.y-circle_.y)<=sqr(circle_.r))+
    	      (int)(sqr(rect_.x+rect_.width-circle_.x)+sqr(rect_.y-circle_.y)<=sqr(circle_.r))+
              (int)(sqr(rect_.x-circle_.x)+sqr(rect_.y+rect_.height-circle_.y)<=sqr(circle_.r))+
              (int)(sqr(rect_.x+rect_.width-circle_.x)+sqr(rect_.y+rect_.height-circle_.y)<=sqr(circle_.r)))&3||
         ( ((rect_.y-circle_.y)*(rect_.y+rect_.height-circle_.y)<=0)&&
          ((abs(rect_.x-circle_.x)<=circle_.r)||(abs(rect_.x+rect_.width-circle_.x)<=circle_.r)))||
         ( ((rect_.x-circle_.x)*(rect_.x+rect_.width-circle_.x)<=0)&&
          ((abs(rect_.y-circle_.y)<=circle_.r)||(abs(rect_.y+rect_.height-circle_.y)<=circle_.r)));
    }
    


  • sehr genau ist die funktion aber nicht:

    [cpp HDC hdc = GetDC(hWnd);

    CIRCLE c;
    c.x = 100;
    c.y = 100;
    c.r = 50;

    RECTANGLE rc;
    rc.x = 145;
    rc.y = 0;
    rc.width = 60;
    rc.height = 150;

    Ellipse(hdc, c.x - c.r, c.y - c.r, c.x + c.r, c.y + c.r);
    Rectangle(hdc, rc.x, rc.y, rc.width+rc.x, rc.height+rc.y);

    ReleaseDC(hWnd, hdc); [/cpp]



  • ja, da hab ich wohl geschlampt :p - geht so wohl doch besser:

    inline int sqr(const int x){ return x*x; }
    inline bool intersectsCircleLineX(const circle &c, const int x, const int y1,const int y2)
    {
    	register int r2;
    	return (r2=sqr(c.r)-sqr(c.x-x))>=0&&((r2-sqr(c.y-y1))*(r2-sqr(c.y-y2))<=0||((c.y-y1)*(c.y-y2)<=0&&r2-sqr(c.y-y1)<=0));
    }
    inline bool intersectsCircleLineY(const circle &c, const int x1, const int x2,const int y)
    {
    	register int r2;
    	return (r2=sqr(c.r)-sqr(c.y-y))>=0&&((r2-sqr(c.x-x1))*(r2-sqr(c.x-x2))<=0||((c.x-x1)*(c.x-x2)<=0&&r2-sqr(c.x-x1)<=0));
    }
    bool intersectsCircleRect(const rect &r, const circle &c)
    {
    	return intersectsCircleLineX(c,r.x,r.y,r.y+r.height)||
    		intersectsCircleLineX(c,r.x+r.width,r.y,r.y+r.height)||
    		intersectsCircleLineY(c,r.x,r.x+r.width,r.y)||
    		intersectsCircleLineY(c,r.x,r.x+r.width,r.y+r.height);
    }
    


  • Wieder geschlampt? Mit den selben Koordianten gehts noch immer nicht und obwohl sich kreis und Rect schneiden sagt er "no collision"



  • dann liegt der fehler diesmal nicht bei mir...



  • doch, könnte sogar sein, dass der fehler bei dir liegt, die sache mit dem schneidend er linie mit dem kreis sieht mir sehr abenteuerlich aus

    @threadersteller ich muss das ganze heute abend mal machen, ma schaun ob ichs heute noch poste...



  • also wenn ich diese funktion mit eben diesen beispieldaten füttere liefert sie mir hier true zurück, so wie es sein soll

    die idee des schnitttests ist eigentlich simpel; erst wird geprüft, ob die geraden, die das rechteck definieren, dem kreis überhaupt nah genug kommen; wenn das der fall ist, gibt es zwei möglichkeiten für den schnitt, endweder liegt ein endpunkt im kreis und einer ausserhalb oder beide liegen ausserhalb (ich teste nur einen punkt, denn falls der andere punkt innen liegt, kommen wir gar nicht soweit) aber auf verschiedenen seiten der gerade y=circle_.y bzw. x=circle_.x



  • hi,

    hier das testprogramm:

    int MIN(int x, int y) 
    { 
       return (x < y) ? x : y; 
    }
    
    inline  int abs (int value)
    {
        return ((value < 0) ? -value : value); 
    }
    
    struct RECTANGLE 
    { 
       unsigned int x;  // upper left 
       unsigned int y; 
       unsigned int width;  // lower right 
       unsigned int height; 
    }; 
    
    struct CIRCLE 
    { 
       unsigned int x; 
       unsigned int y; 
       unsigned int r; 
    };
    inline int sqr(const int x){ return x*x; } 
    inline bool intersectsCircleLineX(const CIRCLE &c, const int x, const int y1,const int y2) 
    { 
        register int r2; 
        return (r2=sqr(c.r)-sqr(c.x-x))>=0&&((r2-sqr(c.y-y1))*(r2-sqr(c.y-y2))<=0||((c.y-y1)*(c.y-y2)<=0&&r2-sqr(c.y-y1)<=0)); 
    } 
    inline bool intersectsCircleLineY(const CIRCLE &c, const int x1, const int x2,const int y) 
    { 
        register int r2; 
        return (r2=sqr(c.r)-sqr(c.y-y))>=0&&((r2-sqr(c.x-x1))*(r2-sqr(c.x-x2))<=0||((c.x-x1)*(c.x-x2)<=0&&r2-sqr(c.x-x1)<=0)); 
    } 
    bool intersectsCircleRect(const RECTANGLE &r, const CIRCLE &c) 
    { 
        return intersectsCircleLineX(c,r.x,r.y,r.y+r.height)|| 
            intersectsCircleLineX(c,r.x+r.width,r.y,r.y+r.height)|| 
            intersectsCircleLineY(c,r.x,r.x+r.width,r.y)|| 
            intersectsCircleLineY(c,r.x,r.x+r.width,r.y+r.height); 
    }
    
    #define WIN32_LEAN_AND_MEAN
    
    #include <windows.h>
    #include "resource.h"
    
    const char name[] = "SplashScreen";
    const unsigned long window_style = WS_SYSMENU | WS_MINIMIZEBOX | WS_OVERLAPPED;
    
    HBITMAP g_hBitmap;
    
    ///////////////////////////////////////////////////////////////////////////////
    LRESULT CALLBACK WndProc(HWND hWnd, unsigned int message, unsigned int wParam, long lParam)
    {
    	switch(message)
    	{
    	case WM_CREATE:
    		{
    			RECT rc = {0, 0, 640, 480};
    
    			::AdjustWindowRectEx (&rc, window_style, false, 0);
    
    			int centerX = (::GetSystemMetrics(SM_CXFULLSCREEN)/2) - (abs(rc.left)+rc.right )/2;
    			int centerY = (::GetSystemMetrics(SM_CYFULLSCREEN)/2) - (abs(rc.top )+rc.bottom)/2; 
    
    			::MoveWindow (hWnd, centerX, centerY, (rc.right - rc.left), (rc.bottom - rc.top), true);
    
    			g_hBitmap = LoadBitmap(GetModuleHandle(NULL), MAKEINTRESOURCE(IDB_BITMAP1));
    		} break;
    
    	case WM_PAINT:
    		{
    			PAINTSTRUCT ps;
    			HDC hDC, hBitmapDC;
    			HBITMAP hOldBitmap;
    			BITMAP bm;
    
    			hDC = BeginPaint(hWnd, &ps);
    
    			//-- Create a DC for the bitmap and select it
    			//   into the created DC.
    			hBitmapDC = CreateCompatibleDC(hDC);
    			hOldBitmap = (HBITMAP)SelectObject(hBitmapDC, (HBITMAP)g_hBitmap);
    
    			//-- Display the bitmap into window
    			GetObject((HBITMAP)g_hBitmap, sizeof(BITMAP), &bm);	
    			BitBlt(hDC, 0, 0, bm.bmWidth, bm.bmHeight, hBitmapDC, 0, 0, SRCCOPY);
    
    			//-- Delete the temporary bitmap DC
    			SelectObject(hBitmapDC, hOldBitmap);
    			DeleteDC(hBitmapDC);
    
    			EndPaint(hWnd, &ps);
    
      HDC hdc = GetDC(hWnd); 
    
       CIRCLE c; 
       c.x = 100; 
       c.y = 100; 
       c.r   = 50; 
    
       RECTANGLE rc; 
       rc.x = 145; 
       rc.y = 0; 
       rc.width = 60; 
       rc.height = 150; 
    
       Ellipse(hdc, c.x - c.r, c.y - c.r, c.x + c.r, c.y + c.r); 
       Rectangle(hdc, rc.x, rc.y, rc.width+rc.x, rc.height+rc.y); 
    
       ReleaseDC(hWnd, hdc); 
    
       if( intersectsCircleRect(rc, c) ) 
          MessageBoxA(hWnd, "COLLISION", "Notify", MB_OK|MB_ICONINFORMATION); 
       else 
          MessageBoxA(hWnd, "NO COLLISION", "Notify", MB_OK|MB_ICONINFORMATION);
    		} break;
    
    	case WM_DESTROY:
    		{
    			if(g_hBitmap)
    				DeleteObject(g_hBitmap);
    
    			PostQuitMessage(0);
    		} break;
    	}
    
    	return DefWindowProc(hWnd, message, wParam, lParam);	
    }
    
    int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow)
    {
    		// Läuft die Anwendung schon?
    	HANDLE isRunning = ::CreateFileMapping(INVALID_HANDLE_VALUE, NULL, PAGE_READONLY, 0, 32, name);
    
    	if (isRunning && ::GetLastError () == ERROR_ALREADY_EXISTS)
    	{
    		::MessageBox (NULL, "Läuft schon!", "", MB_OK);
    		return 0;
    	}
    
    	MSG msg;
    	HWND hWnd;
    	WNDCLASSEX wc;
    
    	wc.cbSize = sizeof(wc);
    	wc.style = CS_VREDRAW | CS_HREDRAW;
    	wc.lpfnWndProc = WndProc;
    	wc.cbClsExtra = 0;
    	wc.cbWndExtra = 0;
    	wc.hInstance = hInstance;
    	wc.hIcon = LoadIcon(NULL, MAKEINTRESOURCE(IDI_APPLICATION));
    	wc.hCursor = LoadCursor(NULL, IDC_ARROW);
    	wc.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
    	wc.lpszMenuName = NULL;
    	wc.lpszClassName = name;
    	wc.hIconSm = LoadIcon(NULL, MAKEINTRESOURCE(IDI_APPLICATION));
    
    	RegisterClassEx(&wc);
    
    	hWnd = CreateWindowEx(NULL,
    								 name,
    								 name,
    								 window_style,
    								 0,
    								 0,
    								 640,
    								 480,
    								 NULL,
    								 NULL,
    								 hInstance,
    								 NULL);
    
    	ShowWindow(hWnd, iCmdShow);
    	UpdateWindow(hWnd);
    
    	while(GetMessage(&msg, NULL, 0, 0))
    	{
    		TranslateMessage(&msg);
    		DispatchMessage(&msg);
    	}
    
    	::UnregisterClass (name, hInstance);
    
    	return (int) msg.wParam;
    }
    


  • Vieleicht hilft dir das HIER weiter



  • Eigentlich sollte das nicht all zu schwer sein:

    2 Rechteck stoßen zusammen wenn ihre Projektion auf die X Axe und die Y Axe überlappen:

    A 0AA00
    A 0AA00
    0 00000
    B 000BB
    B 000BB
    
      0AABB
    

    Nein, keine überlappt

    A 00AA0
    A 00AA0
    0 00000
    B 000BB
    B 000BB
    
      00ACB
    

    Nein, nur die X Axe

    0 00000
    0 00000
    A 00AA0
    C 00ACB
    B 000BB
    
      00ACB
    

    Ja, X Axe und Y Axe.

    Dieser Trick funz auch ganz ähnlich mit Kreisen, nur muss du hier die Seiten des Rechtecks einzeln betrachten, und jeweils die "Breite" des Kreises mit gleicher X oder Y Koordinate. Beispiel für:

    00000
    00A00
    0AAA0
    00ABB
    000BB
    

    Erste vertikale Seite:

    |
    0 00000
    0 00A00
    A 0AAA0
    B 00ABB
    B 000BB
    

    Zweite vertikale Seite:

    |
    0 00000
    0 00A00
    0 0AAA0
    B 00ABB
    B 000BB
    

    Erste horizontale Seite:

    00000
     00A00
     0AAA0
    -00ABB
     000BB
    
     00ABB
    

    Zweite horizontale Seite:

    00000
     00A00
     0AAA0
     00ABB
    -000BB
    
     000BB
    

    Und ein Beipiel von es eine Kollision gibt:

    00000
    00A00
    0AACB
    00ABB
    00000
    

    Erste vertikale Seite:

    |
    0 00000
    0 00ABB
    C 0AACB
    B 00ABB
    0 00000
    

    Zweite vertikale Seite:

    |
    0 00000
    B 00ABB
    B 0AACB
    B 00ABB
    0 000B0
    

    Erste horizontale Seite:

    00000
    -00ABB
     0AACB
     00ABB
     00000
    
     00ABB
    

    Zweite horizontale Seite:

    00000
     00ABB
     0AACB
    -00ABB
     00000
    
     00ABB
    

    Einziges Problem: Es werden nur die Ränder des Rechtecks getestet, wenn der Kreis komplet im Rechteck liegt wird diese Methode fehlschlagen. Allerdings kann diesen Fall ja sehr einfach einzeln behandlen, man braucht nur den Kreis viereckig zu machen. Also aus:

    00000
    00A00
    0AAA0
    00A00
    00000
    

    wird

    00000
    0AAA0
    0AAA0
    0AAA0
    00000
    

    und schon musst du nur 2 Rechtecke testen.

    Keine Ahnung ob man das besser hin kriegen könnte.



  • tja... damit gehts...

    inline int sqr(const int x){ return x*x; }
    inline bool intersectsCircleLineX(const CIRCLE &c, const int x, const int y1,const int y2)
    {
        int r2=sqr(c.r)-sqr(c.x-x);
    	int aa=r2-sqr(c.y-y1);
    	int bb=r2-sqr(c.y-y2);
    	int cc=(c.y-y1);
    	int dd=(c.y-y2);
    	return (r2>=0)&&((aa*bb<=0)||((cc*dd<=0)&&(aa<=0)));
    //    return ((r2=sqr(c.r)-sqr(c.x-x))>=0)&&(((r2-sqr(c.y-y1))*(r2-sqr(c.y-y2))<=0)||(((c.y-y1)*(c.y-y2)<=0)&&(r2-sqr(c.y-y1)<=0)));
    }
    inline bool intersectsCircleLineY(const CIRCLE &c, const int x1, const int x2,const int y)
    {
        int r2=sqr(c.r)-sqr(c.y-y);
    	int aa=r2-sqr(c.x-x1);
    	int bb=r2-sqr(c.x-x2);
    	int cc=(c.x-x1);
    	int dd=(c.x-x2);
    	return (r2>=0)&&((aa*bb<=0)||((cc*dd<=0)&&(aa<=0)));
    //    return (r2=sqr(c.r)-sqr(c.y-y))>=0&&((r2-sqr(c.x-x1))*(r2-sqr(c.x-x2))<=0||((c.x-x1)*(c.x-x2)<=0&&r2-sqr(c.x-x1)<=0));
    }
    bool intersectsCircleRect(const RECTANGLE &r, const CIRCLE &c)
    {
        return intersectsCircleLineX(c,r.x,r.y,r.y+r.height)||
            intersectsCircleLineX(c,r.x+r.width,r.y,r.y+r.height)||
            intersectsCircleLineY(c,r.x,r.x+r.width,r.y)||
            intersectsCircleLineY(c,r.x,r.x+r.width,r.y+r.height);
    }
    

    das ist derselbe algorithmus - k.A. wieso es nicht in einer zeile geht... für hinweise bin ich dankbar



  • oh na klar... das alte unsigned/signed problem...

    inline int sqr(const int x){ return x*x; }
    inline bool intersectsCircleLineX(const CIRCLE &c, const int x, const int y1,const int y2)
    {
        register int r2;
        return ((r2=sqr(c.r)-sqr(c.x-x))>=0)&&(((r2-sqr(c.y-y1))*(r2-sqr(c.y-y2))<=0)||(((int)(c.y-y1)*(int)(c.y-y2)<=0)&&(r2-sqr(c.y-y1)<=0)));
    }
    inline bool intersectsCircleLineY(const CIRCLE &c, const int x1, const int x2,const int y)
    {
        register int r2;
        return ((r2=sqr(c.r)-sqr(c.y-y))>=0)&&(((r2-sqr(c.x-x1))*(r2-sqr(c.x-x2))<=0)||(((int)(c.x-x1)*(int)(c.x-x2)<=0)&&(r2-sqr(c.x-x1)<=0)));
    }
    bool intersectsCircleRect(const RECTANGLE &r, const CIRCLE &c)
    {
        return intersectsCircleLineX(c,r.x,r.y,r.y+r.height)||
            intersectsCircleLineX(c,r.x+r.width,r.y,r.y+r.height)||
            intersectsCircleLineY(c,r.x,r.x+r.width,r.y)||
            intersectsCircleLineY(c,r.x,r.x+r.width,r.y+r.height);
    }
    

    ich hatte es mit den definitionen, die oben angegeben waren getestet - und die waren signed; die windows strukturen benutzen aber unsigned - also hatten wir beide recht 🙂


Anmelden zum Antworten