/Vtl/img/Img_ScanConvert.hpp

http://saliency.codeplex.com · C++ Header · 235 lines · 159 code · 36 blank · 40 comment · 26 complexity · 63e76d8f7e6a4738addedd2c35d7faf1 MD5 · raw file

  1. /*************************************************************************\
  2. Microsoft Research Asia
  3. Copyright (c) 2003 Microsoft Corporation
  4. Module Name:
  5. Vis Lib: Scan convert a concaved non-simple polygon
  6. Abstract:
  7. Notes:
  8. Code is adabed from Acrylic, June 9, 2005
  9. History:
  10. Created on 2005 June, 9 by t-yinli@microsoft.com
  11. \*************************************************************************/
  12. #pragma once
  13. #include "xtl/Xtl_Pointer.hpp"
  14. #include "xtl/Xtl_Functor.hpp"
  15. #include <algorithm>
  16. #include <vector>
  17. namespace img
  18. {
  19. /// function prototype for scan converter to process inside scanlines
  20. /// Usage example: Callback(int yScanline, int xBegin, int xEnd, INT_PTR hint);
  21. /// Here, always xBegin < xEnd.
  22. typedef void (*ScanConvertCallback)(int, int, int, INT_PTR);
  23. /// Scan convert a concave non-simple polygon
  24. /// Polygon can be clockwise or counterclockwise
  25. /// Algorithm does uniform point sampling at pixel centers.
  26. /// callback is invoked for each scan line segment inside polygon
  27. /// Inside test done by Jordan's rule: a point is inside if an
  28. /// emanating ray intersects the polygon an odd number of times.
  29. /// NOTE: P must be able to be accessed by .X and .Y.
  30. /// It can be Gdiplus::Point, Gdiplus::PointF, img::ShortXY
  31. template<class P> HRESULT ScanConvertPolygon(
  32. const P* pts,
  33. size_t count,
  34. const Gdiplus::Rect& clip,
  35. ScanConvertCallback callback,
  36. INT_PTR hint);
  37. #pragma region -- Scan conver implementation
  38. namespace impl
  39. {
  40. template<class P>
  41. class ScanConvertHelper
  42. {
  43. private:
  44. INT n; /// number of points in polygon
  45. const P* pt; /// vertices of polygon
  46. struct Edge
  47. {
  48. double x; /// x coordinate of edge's intersection with current scanline
  49. double dx; /// change in x with respect to y
  50. INT i; /// edge index: edge i goes from pt[i] to pt[i+1]
  51. };
  52. std::vector<Edge> active;
  53. INT nact; /// number of active edges
  54. struct CompareIndex
  55. {
  56. CompareIndex(const P* pt) : m_polygon(pt) {}
  57. bool operator() (INT a, INT b)
  58. {
  59. return m_polygon[a].Y < m_polygon[b].Y;
  60. }
  61. private:
  62. const P* m_polygon;
  63. };
  64. struct CompareEdge
  65. {
  66. bool operator() (const Edge& a, const Edge& b)
  67. {
  68. return a.x < b.x;
  69. }
  70. };
  71. void InsertActiveEdge(INT i, INT y) /// append edge i to end of active list
  72. {
  73. const P *p, *q;
  74. INT j = (i < n-1) ? i+1 : 0;
  75. if (pt[i].Y < pt[j].Y)
  76. {
  77. p = &pt[i];
  78. q = &pt[j];
  79. }
  80. else
  81. {
  82. p = &pt[j];
  83. q = &pt[i];
  84. }
  85. /// initialize x position at intersection of edge with scanline y
  86. const double dy = ((double)q->Y - (double)p->Y);
  87. const double dx = ((double)q->X - (double)p->X) / dy;
  88. active[nact].dx = dx;
  89. active[nact].x = dx * ((double) y - (double) p->Y) + (double) p->X;
  90. active[nact].i = i;
  91. ++ nact;
  92. }
  93. void DeleteActiveEdge (INT i) /// remove edge i from active list
  94. {
  95. INT j;
  96. for (j=0; (j < nact) && (active[j].i != i); j++) {}
  97. if (j >= nact) return; /// edge not in active list; happens at clip.GetTop()
  98. -- nact;
  99. memmove(&active[j], &active[j+1], (nact-j) * sizeof(active[0]));
  100. }
  101. public:
  102. HRESULT Action(
  103. const P* pts,
  104. size_t count,
  105. const Gdiplus::Rect& clip,
  106. ScanConvertCallback callback,
  107. INT_PTR hint)
  108. {
  109. if(count <= 0) return E_INVALIDARG;
  110. if(pts == NULL) return E_INVALIDARG;
  111. if(callback == NULL) return E_INVALIDARG;
  112. n = (INT) count;
  113. pt = pts;
  114. /// initialize index array as natrual order
  115. std::vector<INT> idx(n);
  116. active.resize(n);
  117. for(INT i = 0; i < n; i ++)
  118. {
  119. idx[i] = i;
  120. }
  121. /// Sort idx by the order of pt[idx[k]].Y
  122. std::sort(&idx[0], &idx[0] + n, CompareIndex(pt));
  123. nact = 0; /// number of active edges, init as empty
  124. INT k = 0; /// idx[k] is the next vertex to precess
  125. INT y0 = max(clip.GetTop(), xtl::FloorCast<INT>(pt[idx[0]].Y)); /// the topmost scanline
  126. INT y1 = min(clip.GetBottom(), xtl::CeilCast<INT>(pt[idx[n-1]].Y)); /// the bottom scanline
  127. /// step through scanlines
  128. /// scanline y is at y+.5 in continuous coordinates
  129. for(INT y = y0; y < y1; y ++)
  130. {
  131. /// check vertices between previous scanline and current one, if any
  132. for (; (k < n) && (pt[idx[k]].Y <= y); k++)
  133. {
  134. /// to simplify, if pt.y = y+.5, pretend it's above
  135. /// invariant: y-.5 < pt[i].y <= y+.5
  136. INT i = idx[k];
  137. /// insert or delete edges before and after vertex i
  138. /// (i-1 to i, and i to i+1) from active list if they cross scanline y
  139. INT j = i > 0 ? i-1 : n-1; /// vertex previous to i
  140. if (pt[j].Y <= y) /// old edge, remove from active list
  141. {
  142. DeleteActiveEdge(j);
  143. }
  144. else if (pt[j].Y > y) /// new edge, add to active list
  145. {
  146. InsertActiveEdge(j, y);
  147. }
  148. j = i < (n - 1) ? i+1 : 0; /// vertex next after i
  149. if (pt[j].Y <= y) /// old edge, remove from active list
  150. {
  151. DeleteActiveEdge(i);
  152. }
  153. else if (pt[j].Y > y) /// new edge, add to active list
  154. {
  155. InsertActiveEdge(i, y);
  156. }
  157. }
  158. /// sort active edge list by active[j].x
  159. std::sort(&active[0], &active[0] + nact, CompareEdge());
  160. /// draw horizontal segments for scanline y
  161. for (INT j=0; j < nact; j += 2) /// draw horizontal segments
  162. {
  163. /// span 'tween j & j+1 is inside, span tween j+1 & j+2 is outside
  164. INT xl = xtl::CeilCast<INT>(active[j].x); /// left end of span
  165. if (xl < clip.GetLeft()) xl = clip.GetLeft();
  166. if (xl > clip.GetRight()) xl = clip.GetRight();
  167. INT xr = xtl::CeilCast<INT>(active[j+1].x); /// right end of span
  168. if (xr < clip.GetLeft()) xr = clip.GetLeft();
  169. if (xr > clip.GetRight()) xr = clip.GetRight();
  170. if (xl < xr)
  171. {
  172. callback(y, xl, xr, hint); /// Invoke call back in span
  173. }
  174. active[j].x += active[j].dx; /// increment edge coords
  175. active[j+1].x += active[j+1].dx;
  176. }
  177. }
  178. return S_OK;
  179. }
  180. };
  181. } // namespace impl
  182. template<class P>
  183. HRESULT ScanConvertPolygon(
  184. const P* pts,
  185. size_t count,
  186. const Gdiplus::Rect& clip,
  187. ScanConvertCallback callback,
  188. INT_PTR hint)
  189. {
  190. impl::ScanConvertHelper<P> helper;
  191. return helper.Action(pts, count, clip, callback, hint);
  192. }
  193. #pragma endregion
  194. } // namespace img