多邊形裁剪一:Sutherland-Hodgman演算法

yangxi_001發表於2014-07-13

Sutherland-Hodgman演算法

         Sutherland-Hodgman演算法也叫逐邊裁剪法,該演算法是薩瑟蘭德(I.E.Sutherland)和霍德曼(Hodgman)在1974年提出的。這種演算法採用了分割處理、逐邊裁剪的方法。

一,基本思想:

         一次用視窗的一條邊裁剪多邊形。

        考慮視窗的一條邊以及延長線構成的裁剪線該線把平面分成兩個部分:可見一側;不可見一側。多邊形的各條邊的兩端點S、P。它們與裁剪線的位置關係只有四種

情況(1)僅輸出1個頂點P;

情況(2)輸出0個頂點;

情況(3)輸出線段SP與裁剪線的1個交點I;

情況(4)輸出線段SP與裁剪線的1個交點I和1個終點P

二、演算法實現:

  1、已知:多邊形頂點陣列src,頂點個數n,
       定義新多邊形頂點陣列dest

  2、賦初值:用變數flag來標識:
          0表示在內側,1表示在外側。

 3、對多邊形的n條邊進行處理,對當前點號的考慮為:0~n-1。
   for(i=0;i<n;i++)
   {
    if(當前第i個頂點是否在邊界內側?)

  {
     if(flag!=0) /*前一個點在外側嗎?*/
     {
      flag=0;/*從外到內的情況,將標誌置0,作為下一次迴圈的前一點標誌*/
      (dest + j) =求出交點; /*將交點dest放入新多邊形*/

j++
}
     
     (dest + j)= (src + i); /*將當前點srci放入新多邊形*/

j++
}
    else
    {
     if(flag==0) /*前一個點在內側嗎?*/
     {
      flag=1;/*從內到外的情況,將標誌置1,作為下一次迴圈的前一點標誌*/
      (dest + j) =求出交點; /*將交點dest放入新多邊形*/

j++
}
    }
    s= (src + i); /*將當前點作為下次迴圈的前一點*/
   }

三,演算法特點:

  Sutherland-Hodgeman多邊形裁剪演算法具有一般性,被裁剪多邊形可以是任意凸多邊形或凹多邊形,裁剪視窗不侷限於矩形,可以是任意凸多邊形。
  上面的演算法是多邊形相對視窗的一條邊界進行裁剪的實現,對於視窗的每一條邊界依次呼叫該演算法程式,並將前一次裁剪的結果多邊形作為下一次裁剪時的被裁剪多邊形,即可得到完整的多邊形裁剪程式。

  1. //點在有向線段那側  
  2. /* 
  3. 向量叉積法 
  4.  
  5.     為簡單計,測試點表示為P點。假設視窗邊界方向為順時針,如圖中所示,對於其中任一邊界向量,從向量起點A向終點B看過去: 
  6.    
  7.       如果被測試點P在該邊界線右邊(即內側),AB×AP的方向與X-Y平面垂直並指向螢幕裡面,即右手座標系中Z軸的負方向。 
  8.      
  9.         反過來,如果P在該邊界線的左邊(即外側),這時AB×AP的方向與X-Y平面垂直並指向螢幕外面,即右手座標系中Z軸的正方向。 
  10.        
  11.           設:點P(x,y)、點A(xA,yA)、點B(xB,yB), 
  12.             向量AB={(xB-xA),(yB-yA)}, 
  13.             向量AP={(x-xA),(y-yA)}, 
  14.          
  15.             那麼AB×AP的方向可由下式的符號來確定: 
  16.            
  17.             V=(xB-xA)·(y-yA)-(x-xA)·(yB-yA) (3-14)  
  18.              
  19.                 因此,當V≤0時,P在邊界線內側; 
  20.                
  21.                      而V>0時,P在邊界線外側。 
  22.  
  23. */  
  24. static int _RtInSide(RtVector vector, RtPoint point)  
  25. {  
  26.     return (vector.ep.x - vector.sp.x) * (point.y - vector.sp.y) - (vector.ep.y - vector.sp.y) * (point.x - vector.sp.x);  
  27. }  
  28.   
  29. //多邊形點必須是順時針方向  
  30. int rtPrunePSH(RtPoint* src, int num, RtPoint** dest, int* total)  
  31. {  
  32.     int i = 0, j = 0, k = -1, flag = 0;  
  33.     RtPoint start, stop;//被剪裁多邊形  
  34.     RtPoint sp, ep;//剪裁視窗  
  35.     RtPoint* temp = NULL;  
  36.     temp = (RtPoint*)malloc(sizeof(RtPoint) * 3 * (*total));  
  37.     if (temp == NULL) return -1;  
  38.   
  39.       
  40.     sp = *(src + num - 1);  
  41.   
  42.     for ( i = 0; i < num; i++)//剪裁視窗  
  43.     {  
  44.         ep = *(src + i);  
  45.           
  46.         start = *((*dest) + *total - 1);  
  47.         flag = _RtInSide(rtVector(sp, ep), start) > 0 ? 0 : 1;  
  48.   
  49.         for ( j = 0; j < *total; j++)//被剪裁的多邊形  
  50.         {     
  51.             stop = *((*dest) + j);  
  52.             if (_RtInSide(rtVector(sp, ep), stop) <= 0)//當前第i個頂點是否在邊界內側  
  53.             {  
  54.                 if (flag == 0)/*前一個點在外側嗎?*/  
  55.                 {  
  56.                     flag = 1;/*從外到內的情況,將標誌置0,作為下一次迴圈的前一點標誌*/  
  57.                     k++;  
  58.                     CRtPoint<double> point;  
  59.                     CRtPoint<int> st(sp.x, sp.y), et(ep.x, ep.y);                   
  60.                     CRtLine<int> v1(st, et);  
  61.                     st.SetData(start.x, start.y);  
  62.                     et.SetData(stop.x, stop.y);  
  63.                     CRtLine<int> v2(st, et);  
  64.                     v2.Intersect(v1,point);  
  65.                     (temp + k)->x = point[0];/*將交點放入新多邊形*/   
  66.                     (temp + k)->y = point[1];  
  67.                 }  
  68.                 k++;  
  69.                 *(temp + k) =  stop;/*將當前點pi放入新多邊形*/   
  70.             }  
  71.             else  
  72.             {  
  73.                 if (0 != flag)/*前一個點在內側嗎?*/  
  74.                 {  
  75.                     flag = 0;/*從內到外的情況,將標誌置1,作為下一次迴圈的前一點標誌*/  
  76.                     k++;  
  77.                     CRtPoint<double> point;  
  78.                     CRtPoint<int> st(sp.x, sp.y), et(ep.x, ep.y);                   
  79.                     CRtLine<int> v1(st, et);  
  80.                     st.SetData(start.x, start.y);  
  81.                     et.SetData(stop.x, stop.y);  
  82.                     CRtLine<int> v2(st, et);  
  83.                     v2.Intersect(v1,point);  
  84.                     (temp + k)->x = point[0];/*將交點放入新多邊形*/   
  85.                     (temp + k)->y = point[1];  
  86.                 }  
  87.             }  
  88.             start = stop;/*將當前點作為下次迴圈的前一點*/  
  89.         }  
  90.   
  91.         sp = ep;  
  92.           
  93.         *total = k + 1;  
  94.         memcpy(*dest, temp, sizeof(RtPoint) * (*total));  
  95.         k = -1;  
  96.     }  
  97.   
  98.   
  99.     return 0;  
  100. }  

相關文章