174994d13SStephan Aßmus /*
274994d13SStephan Aßmus * Copyright 2005, Stephan Aßmus <superstippi@gmx.de>. All rights reserved.
374994d13SStephan Aßmus * Distributed under the terms of the MIT License.
474994d13SStephan Aßmus *
574994d13SStephan Aßmus * DrawingMode implementing B_OP_INVERT on B_RGBA32.
674994d13SStephan Aßmus *
774994d13SStephan Aßmus */
8c0fe8a07SStephan Aßmus
9c0fe8a07SStephan Aßmus #ifndef DRAWING_MODE_INVERT_H
10c0fe8a07SStephan Aßmus #define DRAWING_MODE_INVERT_H
11c0fe8a07SStephan Aßmus
12c0fe8a07SStephan Aßmus #include "DrawingMode.h"
13c0fe8a07SStephan Aßmus
14c0fe8a07SStephan Aßmus // BLEND_INVERT
158d7b8e8cSStephan Aßmus #define BLEND_INVERT(d, a) \
16c0fe8a07SStephan Aßmus { \
178d7b8e8cSStephan Aßmus pixel32 _p; \
188d7b8e8cSStephan Aßmus _p.data32 = *(uint32*)d; \
198d7b8e8cSStephan Aßmus BLEND(d, 255 - _p.data8[2], 255 - _p.data8[1], 255 - _p.data8[0], a); \
20*5c6b9eb0SJerome Duval (void)_p; \
21c0fe8a07SStephan Aßmus }
22c0fe8a07SStephan Aßmus
23c0fe8a07SStephan Aßmus // ASSIGN_INVERT
248d7b8e8cSStephan Aßmus #define ASSIGN_INVERT(d) \
25c0fe8a07SStephan Aßmus { \
268d7b8e8cSStephan Aßmus pixel32 _p; \
278d7b8e8cSStephan Aßmus _p.data32 = *(uint32*)d; \
288d7b8e8cSStephan Aßmus d[0] = 255 - _p.data8[0]; \
298d7b8e8cSStephan Aßmus d[1] = 255 - _p.data8[1]; \
308d7b8e8cSStephan Aßmus d[2] = 255 - _p.data8[2]; \
319d909e25SStephan Aßmus d[3] = 255; \
32c0fe8a07SStephan Aßmus }
33c0fe8a07SStephan Aßmus
348d7b8e8cSStephan Aßmus // blend_pixel_invert
358d7b8e8cSStephan Aßmus void
blend_pixel_invert(int x,int y,const color_type & c,uint8 cover,agg_buffer * buffer,const PatternHandler * pattern)368d7b8e8cSStephan Aßmus blend_pixel_invert(int x, int y, const color_type& c, uint8 cover,
378d7b8e8cSStephan Aßmus agg_buffer* buffer, const PatternHandler* pattern)
38c0fe8a07SStephan Aßmus {
398d7b8e8cSStephan Aßmus if (pattern->IsHighColor(x, y)) {
40e39da397SStephan Aßmus uint8* p = buffer->row_ptr(y) + (x << 2);
41c0fe8a07SStephan Aßmus if (cover == 255) {
428d7b8e8cSStephan Aßmus ASSIGN_INVERT(p);
43c0fe8a07SStephan Aßmus } else {
448d7b8e8cSStephan Aßmus BLEND_INVERT(p, cover);
45c0fe8a07SStephan Aßmus }
46c0fe8a07SStephan Aßmus }
47c0fe8a07SStephan Aßmus }
48c0fe8a07SStephan Aßmus
498d7b8e8cSStephan Aßmus // blend_hline_invert
508d7b8e8cSStephan Aßmus void
blend_hline_invert(int x,int y,unsigned len,const color_type & c,uint8 cover,agg_buffer * buffer,const PatternHandler * pattern)518d7b8e8cSStephan Aßmus blend_hline_invert(int x, int y, unsigned len,
528d7b8e8cSStephan Aßmus const color_type& c, uint8 cover,
538d7b8e8cSStephan Aßmus agg_buffer* buffer, const PatternHandler* pattern)
54c0fe8a07SStephan Aßmus {
55e39da397SStephan Aßmus uint8* p = buffer->row_ptr(y) + (x << 2);
56c0fe8a07SStephan Aßmus if(cover == 255) {
57c0fe8a07SStephan Aßmus do {
588d7b8e8cSStephan Aßmus if (pattern->IsHighColor(x, y)) {
598d7b8e8cSStephan Aßmus ASSIGN_INVERT(p);
60c0fe8a07SStephan Aßmus }
61c0fe8a07SStephan Aßmus x++;
62c0fe8a07SStephan Aßmus p += 4;
63c0fe8a07SStephan Aßmus } while(--len);
64c0fe8a07SStephan Aßmus } else {
65c0fe8a07SStephan Aßmus do {
668d7b8e8cSStephan Aßmus if (pattern->IsHighColor(x, y)) {
678d7b8e8cSStephan Aßmus BLEND_INVERT(p, cover);
68c0fe8a07SStephan Aßmus }
69c0fe8a07SStephan Aßmus x++;
70c0fe8a07SStephan Aßmus p += 4;
71c0fe8a07SStephan Aßmus } while(--len);
72c0fe8a07SStephan Aßmus }
73c0fe8a07SStephan Aßmus }
74c0fe8a07SStephan Aßmus
758d7b8e8cSStephan Aßmus // blend_solid_hspan_invert
768d7b8e8cSStephan Aßmus void
blend_solid_hspan_invert(int x,int y,unsigned len,const color_type & c,const uint8 * covers,agg_buffer * buffer,const PatternHandler * pattern)778d7b8e8cSStephan Aßmus blend_solid_hspan_invert(int x, int y, unsigned len,
788d7b8e8cSStephan Aßmus const color_type& c, const uint8* covers,
798d7b8e8cSStephan Aßmus agg_buffer* buffer, const PatternHandler* pattern)
80c0fe8a07SStephan Aßmus {
81e39da397SStephan Aßmus uint8* p = buffer->row_ptr(y) + (x << 2);
82c0fe8a07SStephan Aßmus do {
838d7b8e8cSStephan Aßmus if (pattern->IsHighColor(x, y)) {
84c0fe8a07SStephan Aßmus if (*covers) {
85c0fe8a07SStephan Aßmus if (*covers == 255) {
868d7b8e8cSStephan Aßmus ASSIGN_INVERT(p);
87c0fe8a07SStephan Aßmus } else {
888d7b8e8cSStephan Aßmus BLEND_INVERT(p, *covers);
89c0fe8a07SStephan Aßmus }
90c0fe8a07SStephan Aßmus }
91c0fe8a07SStephan Aßmus }
92c0fe8a07SStephan Aßmus covers++;
93c0fe8a07SStephan Aßmus p += 4;
94c0fe8a07SStephan Aßmus x++;
95c0fe8a07SStephan Aßmus } while(--len);
96c0fe8a07SStephan Aßmus }
97c0fe8a07SStephan Aßmus
98c0fe8a07SStephan Aßmus
99c0fe8a07SStephan Aßmus
1008d7b8e8cSStephan Aßmus // blend_solid_vspan_invert
1018d7b8e8cSStephan Aßmus void
blend_solid_vspan_invert(int x,int y,unsigned len,const color_type & c,const uint8 * covers,agg_buffer * buffer,const PatternHandler * pattern)1028d7b8e8cSStephan Aßmus blend_solid_vspan_invert(int x, int y, unsigned len,
1038d7b8e8cSStephan Aßmus const color_type& c, const uint8* covers,
1048d7b8e8cSStephan Aßmus agg_buffer* buffer, const PatternHandler* pattern)
105c0fe8a07SStephan Aßmus {
106e39da397SStephan Aßmus uint8* p = buffer->row_ptr(y) + (x << 2);
107c0fe8a07SStephan Aßmus do {
1088d7b8e8cSStephan Aßmus if (pattern->IsHighColor(x, y)) {
109c0fe8a07SStephan Aßmus if (*covers) {
110c0fe8a07SStephan Aßmus if (*covers == 255) {
1118d7b8e8cSStephan Aßmus ASSIGN_INVERT(p);
112c0fe8a07SStephan Aßmus } else {
1138d7b8e8cSStephan Aßmus BLEND_INVERT(p, *covers);
114c0fe8a07SStephan Aßmus }
115c0fe8a07SStephan Aßmus }
116c0fe8a07SStephan Aßmus }
117c0fe8a07SStephan Aßmus covers++;
1188d7b8e8cSStephan Aßmus p += buffer->stride();
119c0fe8a07SStephan Aßmus y++;
120c0fe8a07SStephan Aßmus } while(--len);
121c0fe8a07SStephan Aßmus }
122c0fe8a07SStephan Aßmus
123c0fe8a07SStephan Aßmus
1248d7b8e8cSStephan Aßmus // blend_color_hspan_invert
1258d7b8e8cSStephan Aßmus void
blend_color_hspan_invert(int x,int y,unsigned len,const color_type * colors,const uint8 * covers,uint8 cover,agg_buffer * buffer,const PatternHandler * pattern)1268d7b8e8cSStephan Aßmus blend_color_hspan_invert(int x, int y, unsigned len,
127c0fe8a07SStephan Aßmus const color_type* colors,
1288d7b8e8cSStephan Aßmus const uint8* covers, uint8 cover,
1298d7b8e8cSStephan Aßmus agg_buffer* buffer, const PatternHandler* pattern)
130c0fe8a07SStephan Aßmus {
131c0fe8a07SStephan Aßmus // TODO: does the R5 app_server check for the
132c0fe8a07SStephan Aßmus // appearance of the high color in the source bitmap?
133e39da397SStephan Aßmus uint8* p = buffer->row_ptr(y) + (x << 2);
134c0fe8a07SStephan Aßmus if (covers) {
135c0fe8a07SStephan Aßmus // non-solid opacity
136c0fe8a07SStephan Aßmus do {
137efec3399SMichael Lotz if (*covers && colors->a > 0) {
138c0fe8a07SStephan Aßmus if (*covers == 255) {
1398d7b8e8cSStephan Aßmus ASSIGN_INVERT(p);
140c0fe8a07SStephan Aßmus } else {
1418d7b8e8cSStephan Aßmus BLEND_INVERT(p, *covers);
142c0fe8a07SStephan Aßmus }
143c0fe8a07SStephan Aßmus }
144c0fe8a07SStephan Aßmus covers++;
145c0fe8a07SStephan Aßmus p += 4;
146efec3399SMichael Lotz ++colors;
147c0fe8a07SStephan Aßmus } while(--len);
148c0fe8a07SStephan Aßmus } else {
149c0fe8a07SStephan Aßmus // solid full opcacity
150c0fe8a07SStephan Aßmus if (cover == 255) {
151c0fe8a07SStephan Aßmus do {
152efec3399SMichael Lotz if (colors->a > 0) {
1538d7b8e8cSStephan Aßmus ASSIGN_INVERT(p);
154efec3399SMichael Lotz }
155c0fe8a07SStephan Aßmus p += 4;
156efec3399SMichael Lotz ++colors;
157c0fe8a07SStephan Aßmus } while(--len);
158c0fe8a07SStephan Aßmus // solid partial opacity
159c0fe8a07SStephan Aßmus } else if (cover) {
160c0fe8a07SStephan Aßmus do {
16161ed28eeSMichael Lotz if (colors->a > 0) {
1628d7b8e8cSStephan Aßmus BLEND_INVERT(p, cover);
16361ed28eeSMichael Lotz }
164c0fe8a07SStephan Aßmus p += 4;
16561ed28eeSMichael Lotz ++colors;
166c0fe8a07SStephan Aßmus } while(--len);
167c0fe8a07SStephan Aßmus }
168c0fe8a07SStephan Aßmus }
169c0fe8a07SStephan Aßmus }
170c0fe8a07SStephan Aßmus
171c0fe8a07SStephan Aßmus #endif // DRAWING_MODE_INVERT_H
172c0fe8a07SStephan Aßmus
173