图形学算法之Nicholl-Lee-Nicholl算法

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了图形学算法之Nicholl-Lee-Nicholl算法相关的知识,希望对你有一定的参考价值。

在本学期的所学的裁剪算法中,有一个叫做Nicholl-Lee-Nicholl的算法。

在实践中,我遇到了一个小问题,因为教程中所讲的只有起点在剪裁框内、左侧和左上的情况,并说其余情况均可以通过点的平移或旋转变换等方式变换到上述3个区域内。

但考虑到裁剪框并不都是正方形,所以上方区域和左侧区域也不都是完全映射的,所以并不能简单的旋转平移点,而是整个屏幕都旋转,这样的话还不如为上方区域单独做算法。

最终实现代码如下:

  1 /* Nicholl-Lee-Nicholl Line Clipping Algorithm */
  2 void NichollLeeNichollClip()
  3 {
  4     // 变量
  5     clipped_line_num = 0;
  6     float m, k;
  7     float mLT, mLB, mRB, mRT;    // 生成起点到四个剪裁窗口顶点的斜率
  8     // 针对每条线
  9     for (int i = 0; i < line_num; i++)
 10     {
 11         // 生成剪裁线
 12         line clip_line = l[i];
 13         // 区域码赋值
 14         getRegionCode(clip_line.original);
 15         getRegionCode(clip_line.terminal);
 16         // 判断框内线条(两端点区域码都为00000)
 17         if ((clip_line.original.code | clip_line.terminal.code) == 0x0)
 18         {
 19             clipped_l[clipped_line_num] = clip_line;
 20             clipped_line_num++;
 21             continue;
 22         }
 23         // 判断框外线条(两端点区域码按为与为真)
 24         if (clip_line.original.code & clip_line.terminal.code)
 25         {
 26             continue;
 27         }
 28         // 判断起点是否在已有算法区域的对称区域
 29         int before_code; // 变换之前的区域码
 30         before_code = clip_line.original.code;
 31         // 右侧变换
 32         if (clip_line.original.code == winRightBitCode)
 33         {
 34             clip_line.original.x = clip_left + clip_right - clip_line.original.x;
 35             clip_line.terminal.x = clip_left + clip_right - clip_line.terminal.x;
 36         }
 37         // 右上侧变换
 38         if (clip_line.original.code == ( winRightBitCode | winTopBitCode ) )
 39         {
 40             clip_line.original.x = clip_left + clip_right - clip_line.original.x;
 41             clip_line.terminal.x = clip_left + clip_right - clip_line.terminal.x;
 42         }
 43         // 左下侧变换
 44         if (clip_line.original.code == ( winLeftBitCode | winBottomBitCode ) )
 45         {
 46             clip_line.original.y = clip_top + clip_bottom - clip_line.original.y;
 47             clip_line.terminal.y = clip_top + clip_bottom - clip_line.terminal.y;
 48         }
 49         // 右下侧变换
 50         if (clip_line.original.code == ( winRightBitCode | winBottomBitCode ) )
 51         {
 52             clip_line.original.x = clip_left + clip_right - clip_line.original.x;
 53             clip_line.terminal.x = clip_left + clip_right - clip_line.terminal.x;
 54             clip_line.original.y = clip_top + clip_bottom - clip_line.original.y;
 55             clip_line.terminal.y = clip_top + clip_bottom - clip_line.terminal.y;
 56         }
 57         // 下侧变换
 58         if (clip_line.original.code == winBottomBitCode)
 59         {
 60             clip_line.original.y = clip_top + clip_bottom - clip_line.original.y;
 61             clip_line.terminal.y = clip_top + clip_bottom - clip_line.terminal.y;
 62         }
 63         // 重新加载区域码
 64         getRegionCode(clip_line.original);
 65         getRegionCode(clip_line.terminal);
 66         // 以下情况均要用到斜率
 67     mLT=(float)(clip_top-clip_line.original.y)/(clip_left-clip_line.original.x);
 68     mLB=(float)(clip_bottom-clip_line.original.y)/(clip_left-clip_line.original.x);
 69     mRB=(float)(clip_bottom-clip_line.original.y)/(clip_right-clip_line.original.x);
 70     mRT=(float)(clip_top-clip_line.original.y)/(clip_right-clip_line.original.x);
 71 m=(float)(clip_line.terminal.y-clip_line.original.y)/(clip_line.terminal.x-clip_line.original.x);
 72     k = 1 / m;
 73 
 74         // 判断起点位置
 75         // 情况一:在框内
 76         if (clip_line.original.code == 0x0)
 77         {
 78             // left
 79             if ((mLT <= m && m < mLB) && clip_line.terminal.code & winLeftBitCode)
 80             {
 81                 clip_line.terminal.y = clip_line.terminal.y + round(m * (clip_left - clip_line.terminal.x));
 82                 clip_line.terminal.x = clip_left;
 83             }
 84             // bottom
 85             if ((m >= mLB || m < mRB) && clip_line.terminal.code & winBottomBitCode)
 86             {
 87                 clip_line.terminal.x = clip_line.terminal.x + round(k * (clip_bottom - clip_line.terminal.y));
 88                 clip_line.terminal.y = clip_bottom;
 89             }
 90             // right
 91             if ((mRB <= m && m < mRT) && clip_line.terminal.code & winRightBitCode)
 92             {
 93                 clip_line.terminal.y = clip_line.terminal.y + round(m * (clip_right - clip_line.terminal.x));
 94                 clip_line.terminal.x = clip_right;
 95             }
 96             // top
 97             if ((m >= mRT || m < mLT) && clip_line.terminal.code & winTopBitCode)
 98             {
 99                 clip_line.terminal.x = clip_line.terminal.x + round(k * (clip_top - clip_line.terminal.y));
100                 clip_line.terminal.y = clip_top;
101             }
102         }
103         // 情况二:在框左侧
104         elseif (clip_line.original.code == winLeftBitCode)
105         {
106             if (m < mLB || m >= mLT) // 区域外
107                 continue;
108             else
109             {
110                 // 起点剪裁 L
111                 clip_line.original.y = clip_line.original.y + round(m * (clip_left - clip_line.original.x);
112                 clip_line.original.x = clip_left;
113                 if (clip_line.terminal.code != 0x0)
114                 {
115                     if (mLB <= m && m < mRB) // LB
116                     {
117                         clip_line.terminal.x = clip_line.terminal.x + round(k * (clip_bottom - clip_line.terminal.y));
118                         clip_line.terminal.y = clip_bottom;
119                     }
120                     elseif (mRB <= m && m < mRT) // LR
121                     {
122                         clip_line.terminal.y = clip_line.terminal.y + round(m * (clip_right - clip_line.terminal.x));
123                         clip_line.terminal.x = clip_right;
124                     }
125                     elseif (mRT <= m && m < mLT) // LT
126                     {
127                         clip_line.terminal.x = clip_line.terminal.x + round(k * (clip_top - clip_line.terminal.y));
128                         clip_line.terminal.y = clip_top;
129                     }
130                 }
131             }
132         }
133         // 情况三:在框左上侧
134         elseif (clip_line.original.code == ( winLeftBitCode | winTopBitCode ) )
135         {
136             if (m < mLB || m >= mRT) // 区域外
137                 continue;
138             else
139             {
140                 // 起点靠近左裁剪线
141                 if (clip_left - clip_line.original.x < clip_line.original.y - clip_top)
142                 {
143                     if (mLB <= m && m < mLT)
144                     {
145                         // L
146                         clip_line.original.y = clip_line.original.y + round(m * (clip_left - clip_line.original.x));
147                         clip_line.original.x = clip_left;
148                         if (clip_line.terminal.code != 0x0)    // LB
149                         {
150                             clip_line.terminal.x = clip_line.terminal.x + round(k * (clip_bottom - clip_line.terminal.y));
151                             clip_line.terminal.y = clip_bottom;
152                         }
153                     }
154                     elseif (mLT <= m && m < mRB)
155                     {
156                         // T
157                         clip_line.original.x = clip_line.original.x + round(k * (clip_top - clip_line.original.y));
158                         clip_line.original.y = clip_top;
159                         if (clip_line.terminal.code != 0x0)    // TB
160                         {
161                             clip_line.terminal.x = clip_line.terminal.x + round(k * (clip_bottom - clip_line.terminal.y));
162                             clip_line.terminal.y = clip_bottom;
163                         }
164                     }
165                     elseif (mRB <= m && m < mRT)
166                     {
167                         // T
168                         clip_line.original.x = clip_line.original.x + round(k * (clip_top - clip_line.original.y));
169                         clip_line.original.y = clip_top;
170                         if (clip_line.terminal.code != 0x0)    // TR
171                         {
172                             clip_line.terminal.y = clip_line.terminal.y + round(m * (clip_right - clip_line.terminal.x));
173                             clip_line.terminal.x = clip_right;
174                         }
175                     }
176                 }
177                 // 起点靠近上裁剪线
178                 else
179                 {
180                     if (mLB <= m && m < mRB)
181                     {
182                         // L
183                         clip_line.original.y = clip_line.original.y + round(m * (clip_left - clip_line.original.x));
184                         clip_line.original.x = clip_left;
185                         if (clip_line.terminal.code != 0x0)    // LB
186                         {
187                             clip_line.terminal.x = clip_line.terminal.x + round(k * (clip_bottom - clip_line.terminal.y));
188                             clip_line.terminal.y = clip_bottom;
189                         }
190                     }
191                     elseif (mRB <= m && m < mLT)
192                     {
193                         // L
194                         clip_line.original.y = clip_line.original.y + round(m * (clip_left - clip_line.original.x));
195                         clip_line.original.x = clip_left;
196                         if (clip_line.terminal.code != 0x0)    // LR
197                         {
198                             clip_line.terminal.y = clip_line.terminal.y + round(m * (clip_right - clip_line.terminal.x));
199                             clip_line.terminal.x = clip_right;
200                         }
201                     }
202                     elseif (mLT <= m && m < mRT)
203                     {
204                         // T
205                         clip_line.original.x = clip_line.original.x + round(k * (clip_top - clip_line.original.y));
206                         clip_line.original.y = clip_top;
207                         if (clip_line.terminal.code != 0x0)    // TR
208                         {
209                             clip_line.terminal.y = clip_line.terminal.y + round(m * (clip_right - clip_line.terminal.x));
210                             clip_line.terminal.x = clip_right;
211                         }
212                     }
213                 }
214             }
215 
216         }
217         // 情况四:在框上侧
218         elseif (clip_line.original.code == winTopBitCode)
219         {
220             if ((m > 0 && m < mLT) || (m < 0 && m >= mRT)) // 区域外
221                 continue;
222             else
223             {
224                 // T
225                 clip_line.original.x = clip_line.original.x + round(k * (clip_top - clip_line.original.y));
226                 clip_line.original.y = clip_top;
227                 if (clip_line.terminal.code != 0x0)
228                 {
229                     if (mLT <= m && m < mLB) // TL
230                     {
231                         clip_line.terminal.y = clip_line.terminal.y + round(m * (clip_left - clip_line.terminal.x));
232                         clip_line.terminal.x = clip_left;
233                     }
234                     elseif (m >= mLB || m <= mRB) // TB
235                     {
236                         clip_line.terminal.x = clip_line.terminal.x + round(k * (clip_bottom - clip_line.terminal.y));
237                         clip_line.terminal.y = clip_bottom;
238                     }
239                     elseif (mRB <= m && m < mRT) // TR
240                     {
241                         clip_line.terminal.y = clip_line.terminal.y + round(m * (clip_right - clip_line.terminal.x));
242                         clip_line.terminal.x = clip_right;
243                     }
244                 }
245             }
246         }
247         // 恢复对称区域的变化
248         // 右侧变换
249         if (before_code == winRightBitCode)
250         {
251             clip_line.original.x = clip_left + clip_right - clip_line.original.x;
252             clip_line.terminal.x = clip_left + clip_right - clip_line.terminal.x;
253         }
254         // 右上侧变换
255         if (before_code == ( winRightBitCode | winTopBitCode ) )
256         {
257             clip_line.original.x = clip_left + clip_right - clip_line.original.x;
258             clip_line.terminal.x = clip_left + clip_right - clip_line.terminal.x;
259         }
260         // 左下侧变换
261         if (before_code == ( winLeftBitCode | winBottomBitCode ) )
262         {
263             clip_line.original.y = clip_top + clip_bottom - clip_line.original.y;
264             clip_line.terminal.y = clip_top + clip_bottom - clip_line.terminal.y;
265         }
266         // 右下侧变换
267         if (before_code == ( winRightBitCode | winBottomBitCode ) )
268         {
269             clip_line.original.x = clip_left + clip_right - clip_line.original.x;
270             clip_line.terminal.x = clip_left + clip_right - clip_line.terminal.x;
271             clip_line.original.y = clip_top + clip_bottom - clip_line.original.y;
272             clip_line.terminal.y = clip_top + clip_bottom - clip_line.terminal.y;
273         }
274         // 下侧变换
275         if (before_code == winBottomBitCode)
276         {
277             clip_line.original.y = clip_top + clip_bottom - clip_line.original.y;
278             clip_line.terminal.y = clip_top + clip_bottom - clip_line.terminal.y;
279         }
280         // 将剪裁好的直线存入数组
281         clipped_l[clipped_line_num] = clip_line;
282         clipped_line_num++;
283     }
284 }

 

以上是关于图形学算法之Nicholl-Lee-Nicholl算法的主要内容,如果未能解决你的问题,请参考以下文章

计算机图形学之扫描转换直线-DDA,Bresenham,中点画线算法

求计算机图形学中的直线绘制函数法、DDA算法、中点法和Bresenham算法的优缺点以及比较.

计算机图形学:直线段扫描转换算法

菜鸟学四轴控制器之2:逐点比较法

03 光栅图形学算法-- 裁剪算法

图形学计算机图形学知识点提纲11