1 //---------------------------------------------------------------------------- 2 // Anti-Grain Geometry - Version 2.4 3 // Copyright (C) 2002-2005 Maxim Shemanarev (http://www.antigrain.com) 4 // 5 // Permission to copy, use, modify, sell and distribute this software 6 // is granted provided this copyright notice appears in all copies. 7 // This software is provided "as is" without express or implied 8 // warranty, and with no claim as to its suitability for any purpose. 9 // 10 //---------------------------------------------------------------------------- 11 // Contact: mcseem@antigrain.com 12 // mcseemagg@yahoo.com 13 // http://www.antigrain.com 14 //---------------------------------------------------------------------------- 15 // 16 // Adaptation for 32-bit screen coordinates (scanline32_u) has been sponsored by 17 // Liberty Technology Systems, Inc., visit http://lib-sys.com 18 // 19 // Liberty Technology Systems, Inc. is the provider of 20 // PostScript and PDF technology for software developers. 21 // 22 //---------------------------------------------------------------------------- 23 24 #ifndef AGG_SCANLINE_U_INCLUDED 25 #define AGG_SCANLINE_U_INCLUDED 26 27 #include "agg_array.h" 28 29 namespace agg 30 { 31 //=============================================================scanline_u8 32 // 33 // Unpacked scanline container class 34 // 35 // This class is used to transfer data from a scanline rasterizer 36 // to the rendering buffer. It's organized very simple. The class stores 37 // information of horizontal spans to render it into a pixel-map buffer. 38 // Each span has staring X, length, and an array of bytes that determine the 39 // cover-values for each pixel. 40 // Before using this class you should know the minimal and maximal pixel 41 // coordinates of your scanline. The protocol of using is: 42 // 1. reset(min_x, max_x) 43 // 2. add_cell() / add_span() - accumulate scanline. 44 // When forming one scanline the next X coordinate must be always greater 45 // than the last stored one, i.e. it works only with ordered coordinates. 46 // 3. Call finalize(y) and render the scanline. 47 // 3. Call reset_spans() to prepare for the new scanline. 48 // 49 // 4. Rendering: 50 // 51 // Scanline provides an iterator class that allows you to extract 52 // the spans and the cover values for each pixel. Be aware that clipping 53 // has not been done yet, so you should perform it yourself. 54 // Use scanline_u8::iterator to render spans: 55 //------------------------------------------------------------------------- 56 // 57 // int y = sl.y(); // Y-coordinate of the scanline 58 // 59 // ************************************ 60 // ...Perform vertical clipping here... 61 // ************************************ 62 // 63 // scanline_u8::const_iterator span = sl.begin(); 64 // 65 // unsigned char* row = m_rbuf->row(y); // The the address of the beginning 66 // // of the current row 67 // 68 // unsigned num_spans = sl.num_spans(); // Number of spans. It's guaranteed that 69 // // num_spans is always greater than 0. 70 // 71 // do 72 // { 73 // const scanline_u8::cover_type* covers = 74 // span->covers; // The array of the cover values 75 // 76 // int num_pix = span->len; // Number of pixels of the span. 77 // // Always greater than 0, still it's 78 // // better to use "int" instead of 79 // // "unsigned" because it's more 80 // // convenient for clipping 81 // int x = span->x; 82 // 83 // ************************************** 84 // ...Perform horizontal clipping here... 85 // ...you have x, covers, and pix_count.. 86 // ************************************** 87 // 88 // unsigned char* dst = row + x; // Calculate the start address of the row. 89 // // In this case we assume a simple 90 // // grayscale image 1-byte per pixel. 91 // do 92 // { 93 // *dst++ = *covers++; // Hypotetical rendering. 94 // } 95 // while(--num_pix); 96 // 97 // ++span; 98 // } 99 // while(--num_spans); // num_spans cannot be 0, so this loop is quite safe 100 //------------------------------------------------------------------------ 101 // 102 // The question is: why should we accumulate the whole scanline when we 103 // could render just separate spans when they're ready? 104 // That's because using the scanline is generally faster. When is consists 105 // of more than one span the conditions for the processor cash system 106 // are better, because switching between two different areas of memory 107 // (that can be very large) occurs less frequently. 108 //------------------------------------------------------------------------ 109 class scanline_u8 110 { 111 public: 112 typedef scanline_u8 self_type; 113 typedef int8u cover_type; 114 typedef int16 coord_type; 115 116 //-------------------------------------------------------------------- 117 struct span 118 { 119 coord_type x; 120 coord_type len; 121 cover_type* covers; 122 }; 123 124 typedef span* iterator; 125 typedef const span* const_iterator; 126 127 //-------------------------------------------------------------------- 128 scanline_u8() : 129 m_min_x(0), 130 m_last_x(0x7FFFFFF0), 131 m_cur_span(0) 132 {} 133 134 //-------------------------------------------------------------------- 135 void reset(int min_x, int max_x) 136 { 137 unsigned max_len = max_x - min_x + 2; 138 if(max_len > m_spans.size()) 139 { 140 m_spans.resize(max_len); 141 m_covers.resize(max_len); 142 } 143 m_last_x = 0x7FFFFFF0; 144 m_min_x = min_x; 145 m_cur_span = &m_spans[0]; 146 } 147 148 //-------------------------------------------------------------------- 149 void add_cell(int x, unsigned cover) 150 { 151 x -= m_min_x; 152 m_covers[x] = (cover_type)cover; 153 if(x == m_last_x+1) 154 { 155 m_cur_span->len++; 156 } 157 else 158 { 159 m_cur_span++; 160 m_cur_span->x = (coord_type)(x + m_min_x); 161 m_cur_span->len = 1; 162 m_cur_span->covers = &m_covers[x]; 163 } 164 m_last_x = x; 165 } 166 167 //-------------------------------------------------------------------- 168 void add_cells(int x, unsigned len, const cover_type* covers) 169 { 170 x -= m_min_x; 171 memcpy(&m_covers[x], covers, len * sizeof(cover_type)); 172 if(x == m_last_x+1) 173 { 174 m_cur_span->len += (coord_type)len; 175 } 176 else 177 { 178 m_cur_span++; 179 m_cur_span->x = (coord_type)(x + m_min_x); 180 m_cur_span->len = (coord_type)len; 181 m_cur_span->covers = &m_covers[x]; 182 } 183 m_last_x = x + len - 1; 184 } 185 186 //-------------------------------------------------------------------- 187 void add_span(int x, unsigned len, unsigned cover) 188 { 189 x -= m_min_x; 190 memset(&m_covers[x], cover, len); 191 if(x == m_last_x+1) 192 { 193 m_cur_span->len += (coord_type)len; 194 } 195 else 196 { 197 m_cur_span++; 198 m_cur_span->x = (coord_type)(x + m_min_x); 199 m_cur_span->len = (coord_type)len; 200 m_cur_span->covers = &m_covers[x]; 201 } 202 m_last_x = x + len - 1; 203 } 204 205 //-------------------------------------------------------------------- 206 void finalize(int y) 207 { 208 m_y = y; 209 } 210 211 //-------------------------------------------------------------------- 212 void reset_spans() 213 { 214 m_last_x = 0x7FFFFFF0; 215 m_cur_span = &m_spans[0]; 216 } 217 218 //-------------------------------------------------------------------- 219 int y() const { return m_y; } 220 unsigned num_spans() const { return unsigned(m_cur_span - &m_spans[0]); } 221 const_iterator begin() const { return &m_spans[1]; } 222 iterator begin() { return &m_spans[1]; } 223 224 private: 225 scanline_u8(const self_type&); 226 const self_type& operator = (const self_type&); 227 228 private: 229 int m_min_x; 230 int m_last_x; 231 int m_y; 232 pod_array<cover_type> m_covers; 233 pod_array<span> m_spans; 234 span* m_cur_span; 235 }; 236 237 238 239 240 //==========================================================scanline_u8_am 241 // 242 // The scanline container with alpha-masking 243 // 244 //------------------------------------------------------------------------ 245 template<class AlphaMask> 246 class scanline_u8_am : public scanline_u8 247 { 248 public: 249 typedef scanline_u8 base_type; 250 typedef AlphaMask alpha_mask_type; 251 typedef base_type::cover_type cover_type; 252 typedef base_type::coord_type coord_type; 253 254 scanline_u8_am() : base_type(), m_alpha_mask(0) {} 255 scanline_u8_am(const AlphaMask& am) : base_type(), m_alpha_mask(&am) {} 256 257 //-------------------------------------------------------------------- 258 void finalize(int span_y) 259 { 260 base_type::finalize(span_y); 261 if(m_alpha_mask) 262 { 263 typename base_type::iterator span = base_type::begin(); 264 unsigned count = base_type::num_spans(); 265 do 266 { 267 m_alpha_mask->combine_hspan(span->x, 268 base_type::y(), 269 span->covers, 270 span->len); 271 ++span; 272 } 273 while(--count); 274 } 275 } 276 277 private: 278 const AlphaMask* m_alpha_mask; 279 }; 280 281 282 283 284 //===========================================================scanline32_u8 285 class scanline32_u8 286 { 287 public: 288 typedef scanline32_u8 self_type; 289 typedef int8u cover_type; 290 typedef int32 coord_type; 291 292 //-------------------------------------------------------------------- 293 struct span 294 { 295 span() {} 296 span(coord_type x_, coord_type len_, cover_type* covers_) : 297 x(x_), len(len_), covers(covers_) {} 298 299 coord_type x; 300 coord_type len; 301 cover_type* covers; 302 }; 303 304 typedef pod_bvector<span, 4> span_array_type; 305 306 //-------------------------------------------------------------------- 307 class const_iterator 308 { 309 public: 310 const_iterator(const span_array_type& spans) : 311 m_spans(spans), 312 m_span_idx(0) 313 {} 314 315 const span& operator*() const { return m_spans[m_span_idx]; } 316 const span* operator->() const { return &m_spans[m_span_idx]; } 317 318 void operator ++ () { ++m_span_idx; } 319 320 private: 321 const span_array_type& m_spans; 322 unsigned m_span_idx; 323 }; 324 325 //-------------------------------------------------------------------- 326 class iterator 327 { 328 public: 329 iterator(span_array_type& spans) : 330 m_spans(spans), 331 m_span_idx(0) 332 {} 333 334 span& operator*() { return m_spans[m_span_idx]; } 335 span* operator->() { return &m_spans[m_span_idx]; } 336 337 void operator ++ () { ++m_span_idx; } 338 339 private: 340 span_array_type& m_spans; 341 unsigned m_span_idx; 342 }; 343 344 345 346 //-------------------------------------------------------------------- 347 scanline32_u8() : 348 m_min_x(0), 349 m_last_x(0x7FFFFFF0), 350 m_covers() 351 {} 352 353 //-------------------------------------------------------------------- 354 void reset(int min_x, int max_x) 355 { 356 unsigned max_len = max_x - min_x + 2; 357 if(max_len > m_covers.size()) 358 { 359 m_covers.resize(max_len); 360 } 361 m_last_x = 0x7FFFFFF0; 362 m_min_x = min_x; 363 m_spans.remove_all(); 364 } 365 366 //-------------------------------------------------------------------- 367 void add_cell(int x, unsigned cover) 368 { 369 x -= m_min_x; 370 m_covers[x] = cover_type(cover); 371 if(x == m_last_x+1) 372 { 373 m_spans.last().len++; 374 } 375 else 376 { 377 m_spans.add(span(coord_type(x + m_min_x), 1, &m_covers[x])); 378 } 379 m_last_x = x; 380 } 381 382 //-------------------------------------------------------------------- 383 void add_cells(int x, unsigned len, const cover_type* covers) 384 { 385 x -= m_min_x; 386 memcpy(&m_covers[x], covers, len * sizeof(cover_type)); 387 if(x == m_last_x+1) 388 { 389 m_spans.last().len += coord_type(len); 390 } 391 else 392 { 393 m_spans.add(span(coord_type(x + m_min_x), 394 coord_type(len), 395 &m_covers[x])); 396 } 397 m_last_x = x + len - 1; 398 } 399 400 //-------------------------------------------------------------------- 401 void add_span(int x, unsigned len, unsigned cover) 402 { 403 x -= m_min_x; 404 memset(&m_covers[x], cover, len); 405 if(x == m_last_x+1) 406 { 407 m_spans.last().len += coord_type(len); 408 } 409 else 410 { 411 m_spans.add(span(coord_type(x + m_min_x), 412 coord_type(len), 413 &m_covers[x])); 414 } 415 m_last_x = x + len - 1; 416 } 417 418 //-------------------------------------------------------------------- 419 void finalize(int y) 420 { 421 m_y = y; 422 } 423 424 //-------------------------------------------------------------------- 425 void reset_spans() 426 { 427 m_last_x = 0x7FFFFFF0; 428 m_spans.remove_all(); 429 } 430 431 //-------------------------------------------------------------------- 432 int y() const { return m_y; } 433 unsigned num_spans() const { return m_spans.size(); } 434 const_iterator begin() const { return const_iterator(m_spans); } 435 iterator begin() { return iterator(m_spans); } 436 437 private: 438 scanline32_u8(const self_type&); 439 const self_type& operator = (const self_type&); 440 441 private: 442 int m_min_x; 443 int m_last_x; 444 int m_y; 445 pod_array<cover_type> m_covers; 446 span_array_type m_spans; 447 }; 448 449 450 451 452 //========================================================scanline32_u8_am 453 // 454 // The scanline container with alpha-masking 455 // 456 //------------------------------------------------------------------------ 457 template<class AlphaMask> 458 class scanline32_u8_am : public scanline32_u8 459 { 460 public: 461 typedef scanline_u8 base_type; 462 typedef AlphaMask alpha_mask_type; 463 typedef base_type::cover_type cover_type; 464 typedef base_type::coord_type coord_type; 465 466 467 scanline32_u8_am() : m_alpha_mask(0) 468 { 469 this->base_type(); 470 } 471 472 scanline32_u8_am(const AlphaMask& am) : m_alpha_mask(&am) 473 { 474 this->base_type(); 475 } 476 477 //-------------------------------------------------------------------- 478 void finalize(int span_y) 479 { 480 this->base_type::finalize(span_y); 481 if(m_alpha_mask) 482 { 483 typename base_type::iterator span = this->base_type::begin(); 484 unsigned count = this->base_type::num_spans(); 485 do 486 { 487 m_alpha_mask->combine_hspan(span->x, 488 this->base_type::y(), 489 span->covers, 490 span->len); 491 ++span; 492 } 493 while(--count); 494 } 495 } 496 497 private: 498 const AlphaMask* m_alpha_mask; 499 }; 500 501 502 503 } 504 505 #endif 506 507