Convert all line endings to LF.
[pdfium.git] / core / src / fxge / dib / fx_dib_main.cpp
1 // Copyright 2014 PDFium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4  
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6
7 #include "../../../include/fxge/fx_dib.h"
8 #include "../../../include/fxge/fx_ge.h"
9 #include "../../../include/fxcodec/fx_codec.h"
10 #include "dib_int.h"
11 #include <limits.h>
12 FX_BOOL ConvertBuffer(FXDIB_Format dest_format, FX_LPBYTE dest_buf, int dest_pitch, int width, int height,
13                       const CFX_DIBSource* pSrcBitmap, int src_left, int src_top, FX_DWORD*& pal, void* pIccTransform);
14 void CmykDecode(FX_DWORD cmyk, int& c, int& m, int& y, int& k)
15 {
16     c = FXSYS_GetCValue(cmyk);
17     m = FXSYS_GetMValue(cmyk);
18     y = FXSYS_GetYValue(cmyk);
19     k = FXSYS_GetKValue(cmyk);
20 }
21 void ArgbDecode(FX_DWORD argb, int& a, int& r, int& g, int& b)
22 {
23     a = FXARGB_A(argb);
24     r = FXARGB_R(argb);
25     g = FXARGB_G(argb);
26     b = FXARGB_B(argb);
27 }
28 void ArgbDecode(FX_DWORD argb, int& a, FX_COLORREF& rgb)
29 {
30     a = FXARGB_A(argb);
31     rgb = FXSYS_RGB(FXARGB_R(argb), FXARGB_G(argb), FXARGB_B(argb));
32 }
33 FX_DWORD ArgbEncode(int a, FX_COLORREF rgb)
34 {
35     return FXARGB_MAKE(a, FXSYS_GetRValue(rgb), FXSYS_GetGValue(rgb), FXSYS_GetBValue(rgb));
36 }
37 CFX_DIBSource::CFX_DIBSource()
38 {
39     m_bpp = 0;
40     m_AlphaFlag = 0;
41     m_Width = m_Height = 0;
42     m_Pitch = 0;
43     m_pPalette = NULL;
44     m_pAlphaMask = NULL;
45 }
46 CFX_DIBSource::~CFX_DIBSource()
47 {
48     if (m_pPalette) {
49         FX_Free(m_pPalette);
50     }
51     if (m_pAlphaMask) {
52         delete m_pAlphaMask;
53     }
54 }
55 CFX_DIBitmap::CFX_DIBitmap()
56 {
57     m_bExtBuf = FALSE;
58     m_pBuffer = NULL;
59     m_pPalette = NULL;
60 }
61 #define _MAX_OOM_LIMIT_ 12000000
62 FX_BOOL CFX_DIBitmap::Create(int width, int height, FXDIB_Format format, FX_LPBYTE pBuffer, int pitch)
63 {
64     m_pBuffer = NULL;
65     m_bpp = (FX_BYTE)format;
66     m_AlphaFlag = (FX_BYTE)(format >> 8);
67     m_Width = m_Height = m_Pitch = 0;
68     if (width <= 0 || height <= 0 || pitch < 0) {
69         return FALSE;
70     }
71     if ((INT_MAX - 31) / width < (format & 0xff)) {
72         return FALSE;
73     }
74     if (!pitch) {
75         pitch = (width * (format & 0xff) + 31) / 32 * 4;
76     }
77     if ((1 << 30) / pitch < height) {
78         return FALSE;
79     }
80     if (pBuffer) {
81         m_pBuffer = pBuffer;
82         m_bExtBuf = TRUE;
83     } else {
84         int size = pitch * height + 4;
85         int oomlimit = _MAX_OOM_LIMIT_;
86         if (oomlimit >= 0 && size >= oomlimit) {
87             m_pBuffer = FX_AllocNL(FX_BYTE, size);
88         } else {
89             m_pBuffer = FX_Alloc(FX_BYTE, size);
90         }
91         if (m_pBuffer == NULL) {
92             return FALSE;
93         }
94         FXSYS_memset32(m_pBuffer, 0, sizeof (FX_BYTE) * size);
95     }
96     m_Width = width;
97     m_Height = height;
98     m_Pitch = pitch;
99     if (HasAlpha() && format != FXDIB_Argb) {
100         FX_BOOL ret = TRUE;
101         ret = BuildAlphaMask();
102         if (!ret) {
103             if (!m_bExtBuf && m_pBuffer) {
104                 FX_Free(m_pBuffer);
105                 m_pBuffer = NULL;
106                 m_Width = m_Height = m_Pitch = 0;
107                 return FALSE;
108             }
109         }
110     }
111     return TRUE;
112 }
113 FX_BOOL CFX_DIBitmap::Copy(const CFX_DIBSource* pSrc)
114 {
115     if (m_pBuffer) {
116         return FALSE;
117     }
118     if (!Create(pSrc->GetWidth(), pSrc->GetHeight(), pSrc->GetFormat())) {
119         return FALSE;
120     }
121     CopyPalette(pSrc->GetPalette());
122     CopyAlphaMask(pSrc->m_pAlphaMask);
123     for (int row = 0; row < pSrc->GetHeight(); row ++) {
124         FXSYS_memcpy32(m_pBuffer + row * m_Pitch, pSrc->GetScanline(row), m_Pitch);
125     }
126     return TRUE;
127 }
128 CFX_DIBitmap::~CFX_DIBitmap()
129 {
130     if (m_pBuffer && !m_bExtBuf) {
131         FX_Free(m_pBuffer);
132     }
133     m_pBuffer = NULL;
134 }
135 void CFX_DIBitmap::TakeOver(CFX_DIBitmap* pSrcBitmap)
136 {
137     if (m_pBuffer && !m_bExtBuf) {
138         FX_Free(m_pBuffer);
139     }
140     if (m_pPalette) {
141         FX_Free(m_pPalette);
142     }
143     if (m_pAlphaMask) {
144         delete m_pAlphaMask;
145     }
146     m_pBuffer = pSrcBitmap->m_pBuffer;
147     m_pPalette = pSrcBitmap->m_pPalette;
148     m_pAlphaMask = pSrcBitmap->m_pAlphaMask;
149     pSrcBitmap->m_pBuffer = NULL;
150     pSrcBitmap->m_pPalette = NULL;
151     pSrcBitmap->m_pAlphaMask = NULL;
152     m_bpp = pSrcBitmap->m_bpp;
153     m_bExtBuf = pSrcBitmap->m_bExtBuf;
154     m_AlphaFlag = pSrcBitmap->m_AlphaFlag;
155     m_Width = pSrcBitmap->m_Width;
156     m_Height = pSrcBitmap->m_Height;
157     m_Pitch = pSrcBitmap->m_Pitch;
158 }
159 CFX_DIBitmap* CFX_DIBSource::Clone(const FX_RECT* pClip) const
160 {
161     FX_RECT rect(0, 0, m_Width, m_Height);
162     if (pClip) {
163         rect.Intersect(*pClip);
164         if (rect.IsEmpty()) {
165             return NULL;
166         }
167     }
168     CFX_DIBitmap* pNewBitmap = FX_NEW CFX_DIBitmap;
169     if (!pNewBitmap) {
170         return NULL;
171     }
172     if (!pNewBitmap->Create(rect.Width(), rect.Height(), GetFormat())) {
173         delete pNewBitmap;
174         return NULL;
175     }
176     pNewBitmap->CopyPalette(m_pPalette);
177     pNewBitmap->CopyAlphaMask(m_pAlphaMask, pClip);
178     if (GetBPP() == 1 && rect.left % 8 != 0) {
179         int left_shift = rect.left % 32;
180         int right_shift = 32 - left_shift;
181         int dword_count = pNewBitmap->m_Pitch / 4;
182         for (int row = rect.top; row < rect.bottom; row ++) {
183             FX_DWORD* src_scan = (FX_DWORD*)GetScanline(row) + rect.left / 32;
184             FX_DWORD* dest_scan = (FX_DWORD*)pNewBitmap->GetScanline(row - rect.top);
185             for (int i = 0; i < dword_count; i ++) {
186                 dest_scan[i] = (src_scan[i] << left_shift) | (src_scan[i + 1] >> right_shift);
187             }
188         }
189     } else {
190         int copy_len = (pNewBitmap->GetWidth() * pNewBitmap->GetBPP() + 7) / 8;
191         if (m_Pitch < (FX_DWORD)copy_len) {
192             copy_len = m_Pitch;
193         }
194         for (int row = rect.top; row < rect.bottom; row ++) {
195             FX_LPCBYTE src_scan = GetScanline(row) + rect.left * m_bpp / 8;
196             FX_LPBYTE dest_scan = (FX_LPBYTE)pNewBitmap->GetScanline(row - rect.top);
197             FXSYS_memcpy32(dest_scan, src_scan, copy_len);
198         }
199     }
200     return pNewBitmap;
201 }
202 void CFX_DIBSource::BuildPalette()
203 {
204     if (m_pPalette) {
205         return;
206     }
207     if (GetBPP() == 1) {
208         m_pPalette = FX_Alloc(FX_DWORD, 2);
209         if (!m_pPalette) {
210             return;
211         }
212         if(IsCmykImage()) {
213             m_pPalette[0] = 0xff;
214             m_pPalette[1] = 0;
215         } else {
216             m_pPalette[0] = 0xff000000;
217             m_pPalette[1] = 0xffffffff;
218         }
219     } else if (GetBPP() == 8) {
220         m_pPalette = FX_Alloc(FX_DWORD, 256);
221         if (!m_pPalette) {
222             return;
223         }
224         if(IsCmykImage()) {
225             for (int i = 0; i < 256; i ++) {
226                 m_pPalette[i] = 0xff - i;
227             }
228         } else {
229             for (int i = 0; i < 256; i ++) {
230                 m_pPalette[i] = 0xff000000 | (i * 0x10101);
231             }
232         }
233     }
234 }
235 FX_BOOL CFX_DIBSource::BuildAlphaMask()
236 {
237     if (m_pAlphaMask) {
238         return TRUE;
239     }
240     m_pAlphaMask = FX_NEW CFX_DIBitmap;
241     if (!m_pAlphaMask) {
242         return FALSE;
243     }
244     if (!m_pAlphaMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
245         delete m_pAlphaMask;
246         m_pAlphaMask = NULL;
247         return FALSE;
248     }
249     FXSYS_memset8(m_pAlphaMask->GetBuffer(), 0xff, m_pAlphaMask->GetHeight()*m_pAlphaMask->GetPitch());
250     return TRUE;
251 }
252 FX_DWORD CFX_DIBSource::GetPaletteEntry(int index) const
253 {
254     ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());
255     if (m_pPalette) {
256         return m_pPalette[index];
257     }
258     if (IsCmykImage()) {
259         if (GetBPP() == 1) {
260             return index ? 0 : 0xff;
261         }
262         return 0xff - index;
263     }
264     if (GetBPP() == 1) {
265         return index ? 0xffffffff : 0xff000000;
266     }
267     return index * 0x10101 | 0xff000000;
268 }
269 void CFX_DIBSource::SetPaletteEntry(int index, FX_DWORD color)
270 {
271     ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());
272     if (m_pPalette == NULL) {
273         BuildPalette();
274     }
275     m_pPalette[index] = color;
276 }
277 int CFX_DIBSource::FindPalette(FX_DWORD color) const
278 {
279     ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());
280     if (m_pPalette == NULL) {
281         if (IsCmykImage()) {
282             if (GetBPP() == 1) {
283                 return ((FX_BYTE)color == 0xff) ? 0 : 1;
284             }
285             return 0xff - (FX_BYTE)color;
286         }
287         if (GetBPP() == 1) {
288             return ((FX_BYTE)color == 0xff) ? 1 : 0;
289         }
290         return (FX_BYTE)color;
291     }
292     int palsize = (1 << GetBPP());
293     for (int i = 0; i < palsize; i ++)
294         if (m_pPalette[i] == color) {
295             return i;
296         }
297     return -1;
298 }
299 void CFX_DIBitmap::Clear(FX_DWORD color)
300 {
301     if (m_pBuffer == NULL) {
302         return;
303     }
304     switch (GetFormat()) {
305         case FXDIB_1bppMask:
306             FXSYS_memset8(m_pBuffer, (color & 0xff000000) ? 0xff : 0, m_Pitch * m_Height);
307             break;
308         case FXDIB_1bppRgb: {
309                 int index = FindPalette(color);
310                 FXSYS_memset8(m_pBuffer, index ? 0xff : 0, m_Pitch * m_Height);
311                 break;
312             }
313         case FXDIB_8bppMask:
314             FXSYS_memset8(m_pBuffer, color >> 24, m_Pitch * m_Height);
315             break;
316         case FXDIB_8bppRgb: {
317                 int index = FindPalette(color);
318                 FXSYS_memset8(m_pBuffer, index, m_Pitch * m_Height);
319                 break;
320             }
321         case FXDIB_Rgb:
322         case FXDIB_Rgba: {
323                 int a, r, g, b;
324                 ArgbDecode(color, a, r, g, b);
325                 if (r == g && g == b) {
326                     FXSYS_memset8(m_pBuffer, r, m_Pitch * m_Height);
327                 } else {
328                     int byte_pos = 0;
329                     for (int col = 0; col < m_Width; col ++) {
330                         m_pBuffer[byte_pos++] = b;
331                         m_pBuffer[byte_pos++] = g;
332                         m_pBuffer[byte_pos++] = r;
333                     }
334                     for (int row = 1; row < m_Height; row ++) {
335                         FXSYS_memcpy32(m_pBuffer + row * m_Pitch, m_pBuffer, m_Pitch);
336                     }
337                 }
338                 break;
339             }
340         case FXDIB_Rgb32:
341         case FXDIB_Argb: {
342                 color = IsCmykImage() ? FXCMYK_TODIB(color) : FXARGB_TODIB(color);
343                 for (int i = 0; i < m_Width; i ++) {
344                     ((FX_DWORD*)m_pBuffer)[i] = color;
345                 }
346                 for (int row = 1; row < m_Height; row ++) {
347                     FXSYS_memcpy32(m_pBuffer + row * m_Pitch, m_pBuffer, m_Pitch);
348                 }
349                 break;
350             }
351         default:
352             break;
353     }
354 }
355 void CFX_DIBSource::GetOverlapRect(int& dest_left, int& dest_top, int& width, int& height,
356                                    int src_width, int src_height, int& src_left, int& src_top,
357                                    const CFX_ClipRgn* pClipRgn)
358 {
359     if (width == 0 || height == 0) {
360         return;
361     }
362     ASSERT(width > 0 && height > 0);
363     if (dest_left > m_Width || dest_top > m_Height) {
364         width = 0;
365         height = 0;
366         return;
367     }
368     int x_offset = dest_left - src_left;
369     int y_offset = dest_top - src_top;
370     FX_RECT src_rect(src_left, src_top, src_left + width, src_top + height);
371     FX_RECT src_bound(0, 0, src_width, src_height);
372     src_rect.Intersect(src_bound);
373     FX_RECT dest_rect(src_rect.left + x_offset, src_rect.top + y_offset,
374                       src_rect.right + x_offset, src_rect.bottom + y_offset);
375     FX_RECT dest_bound(0, 0, m_Width, m_Height);
376     dest_rect.Intersect(dest_bound);
377     if (pClipRgn) {
378         dest_rect.Intersect(pClipRgn->GetBox());
379     }
380     dest_left = dest_rect.left;
381     dest_top = dest_rect.top;
382     src_left = dest_left - x_offset;
383     src_top = dest_top - y_offset;
384     width = dest_rect.right - dest_rect.left;
385     height = dest_rect.bottom - dest_rect.top;
386 }
387 FX_BOOL CFX_DIBitmap::TransferBitmap(int dest_left, int dest_top, int width, int height,
388                                      const CFX_DIBSource* pSrcBitmap, int src_left, int src_top, void* pIccTransform)
389 {
390     if (m_pBuffer == NULL) {
391         return FALSE;
392     }
393     GetOverlapRect(dest_left, dest_top, width, height, pSrcBitmap->GetWidth(), pSrcBitmap->GetHeight(), src_left, src_top, NULL);
394     if (width == 0 || height == 0) {
395         return TRUE;
396     }
397     FXDIB_Format dest_format = GetFormat();
398     FXDIB_Format src_format = pSrcBitmap->GetFormat();
399     if (dest_format == src_format && pIccTransform == NULL) {
400         if (GetBPP() == 1) {
401             for (int row = 0; row < height; row ++) {
402                 FX_LPBYTE dest_scan = m_pBuffer + (dest_top + row) * m_Pitch;
403                 FX_LPCBYTE src_scan = pSrcBitmap->GetScanline(src_top + row);
404                 for (int col = 0; col < width; col ++) {
405                     if (src_scan[(src_left + col) / 8] & (1 << (7 - (src_left + col) % 8))) {
406                         dest_scan[(dest_left + col) / 8] |= 1 << (7 - (dest_left + col) % 8);
407                     } else {
408                         dest_scan[(dest_left + col) / 8] &= ~(1 << (7 - (dest_left + col) % 8));
409                     }
410                 }
411             }
412         } else {
413             int Bpp = GetBPP() / 8;
414             for (int row = 0; row < height; row ++) {
415                 FX_LPBYTE dest_scan = m_pBuffer + (dest_top + row) * m_Pitch + dest_left * Bpp;
416                 FX_LPCBYTE src_scan = pSrcBitmap->GetScanline(src_top + row) + src_left * Bpp;
417                 FXSYS_memcpy32(dest_scan, src_scan, width * Bpp);
418             }
419         }
420     } else {
421         if (m_pPalette) {
422             return FALSE;
423         }
424         if (m_bpp == 8) {
425             dest_format = FXDIB_8bppMask;
426         }
427         FX_LPBYTE dest_buf = m_pBuffer + dest_top * m_Pitch + dest_left * GetBPP() / 8;
428         FX_DWORD* d_plt = NULL;
429         if(!ConvertBuffer(dest_format, dest_buf, m_Pitch, width, height, pSrcBitmap, src_left, src_top, d_plt, pIccTransform)) {
430             return FALSE;
431         }
432     }
433     return TRUE;
434 }
435 #ifndef _FPDFAPI_MINI_
436 FX_BOOL CFX_DIBitmap::TransferMask(int dest_left, int dest_top, int width, int height,
437                                    const CFX_DIBSource* pMask, FX_DWORD color, int src_left, int src_top, int alpha_flag, void* pIccTransform)
438 {
439     if (m_pBuffer == NULL) {
440         return FALSE;
441     }
442     ASSERT(HasAlpha() && (m_bpp >= 24));
443     ASSERT(pMask->IsAlphaMask());
444     if (!HasAlpha() || !pMask->IsAlphaMask() || m_bpp < 24) {
445         return FALSE;
446     }
447     GetOverlapRect(dest_left, dest_top, width, height, pMask->GetWidth(), pMask->GetHeight(), src_left, src_top, NULL);
448     if (width == 0 || height == 0) {
449         return TRUE;
450     }
451     int src_pitch = pMask->GetPitch();
452     int src_bpp = pMask->GetBPP();
453     int alpha;
454     FX_DWORD dst_color;
455     if (alpha_flag >> 8) {
456         alpha = alpha_flag & 0xff;
457         dst_color = FXCMYK_TODIB(color);
458     } else {
459         alpha = FXARGB_A(color);
460         dst_color = FXARGB_TODIB(color);
461     }
462     FX_LPBYTE color_p = (FX_LPBYTE)&dst_color;
463     if (pIccTransform && CFX_GEModule::Get()->GetCodecModule() && CFX_GEModule::Get()->GetCodecModule()->GetIccModule()) {
464         ICodec_IccModule* pIccModule = CFX_GEModule::Get()->GetCodecModule()->GetIccModule();
465         pIccModule->TranslateScanline(pIccTransform, color_p, color_p, 1);
466     } else {
467         if (alpha_flag >> 8 && !IsCmykImage())
468             AdobeCMYK_to_sRGB1(FXSYS_GetCValue(color), FXSYS_GetMValue(color), FXSYS_GetYValue(color), FXSYS_GetKValue(color),
469                                color_p[2], color_p[1], color_p[0]);
470         else if (!(alpha_flag >> 8) && IsCmykImage()) {
471             return FALSE;
472         }
473     }
474     if(!IsCmykImage()) {
475         color_p[3] = (FX_BYTE)alpha;
476     }
477     if (GetFormat() == FXDIB_Argb) {
478         for (int row = 0; row < height; row ++) {
479             FX_DWORD* dest_pos = (FX_DWORD*)(m_pBuffer + (dest_top + row) * m_Pitch + dest_left * 4);
480             FX_LPCBYTE src_scan = pMask->GetScanline(src_top + row);
481             if (src_bpp == 1) {
482                 for (int col = 0; col < width; col ++) {
483                     int src_bitpos = src_left + col;
484                     if (src_scan[src_bitpos / 8] & (1 << (7 - src_bitpos % 8))) {
485                         *dest_pos = dst_color;
486                     } else {
487                         *dest_pos = 0;
488                     }
489                     dest_pos ++;
490                 }
491             } else {
492                 src_scan += src_left;
493                 dst_color = FXARGB_TODIB(dst_color);
494                 dst_color &= 0xffffff;
495                 for (int col = 0; col < width; col ++) {
496                     FXARGB_SETDIB(dest_pos++, dst_color | ((alpha * (*src_scan++) / 255) << 24));
497                 }
498             }
499         }
500     } else {
501         int comps = m_bpp / 8;
502         for (int row = 0; row < height; row ++) {
503             FX_LPBYTE dest_color_pos = m_pBuffer + (dest_top + row) * m_Pitch + dest_left * comps;
504             FX_LPBYTE dest_alpha_pos = (FX_LPBYTE)m_pAlphaMask->GetScanline(dest_top + row) + dest_left;
505             FX_LPCBYTE src_scan = pMask->GetScanline(src_top + row);
506             if (src_bpp == 1) {
507                 for (int col = 0; col < width; col ++) {
508                     int src_bitpos = src_left + col;
509                     if (src_scan[src_bitpos / 8] & (1 << (7 - src_bitpos % 8))) {
510                         FXSYS_memcpy32(dest_color_pos, color_p, comps);
511                         *dest_alpha_pos = 0xff;
512                     } else {
513                         FXSYS_memset32(dest_color_pos, 0, comps);
514                         *dest_alpha_pos = 0;
515                     }
516                     dest_color_pos += comps;
517                     dest_alpha_pos ++;
518                 }
519             } else {
520                 src_scan += src_left;
521                 for (int col = 0; col < width; col ++) {
522                     FXSYS_memcpy32(dest_color_pos, color_p, comps);
523                     dest_color_pos += comps;
524                     *dest_alpha_pos++ = (alpha * (*src_scan++) / 255);
525                 }
526             }
527         }
528     }
529     return TRUE;
530 }
531 #endif
532 void CFX_DIBSource::CopyPalette(const FX_DWORD* pSrc, FX_DWORD size)
533 {
534     if (pSrc == NULL || GetBPP() > 8) {
535         if (m_pPalette) {
536             FX_Free(m_pPalette);
537         }
538         m_pPalette = NULL;
539     } else {
540         FX_DWORD pal_size = 1 << GetBPP();
541         if (m_pPalette == NULL) {
542             m_pPalette = FX_Alloc(FX_DWORD, pal_size);
543         }
544         if (!m_pPalette) {
545             return;
546         }
547         if (pal_size > size) {
548             pal_size = size;
549         }
550         FXSYS_memcpy32(m_pPalette, pSrc, pal_size * sizeof(FX_DWORD));
551     }
552 }
553 void CFX_DIBSource::GetPalette(FX_DWORD* pal, int alpha) const
554 {
555     ASSERT(GetBPP() <= 8 && !IsCmykImage());
556     if (GetBPP() == 1) {
557         pal[0] = ((m_pPalette ? m_pPalette[0] : 0xff000000) & 0xffffff) | (alpha << 24);
558         pal[1] = ((m_pPalette ? m_pPalette[1] : 0xffffffff) & 0xffffff) | (alpha << 24);
559         return;
560     }
561     if (m_pPalette) {
562         for (int i = 0; i < 256; i ++) {
563             pal[i] = (m_pPalette[i] & 0x00ffffff) | (alpha << 24);
564         }
565     } else {
566         for (int i = 0; i < 256; i ++) {
567             pal[i] = (i * 0x10101) | (alpha << 24);
568         }
569     }
570 }
571 CFX_DIBitmap* CFX_DIBSource::GetAlphaMask(const FX_RECT* pClip) const
572 {
573     ASSERT(GetFormat() == FXDIB_Argb);
574     FX_RECT rect(0, 0, m_Width, m_Height);
575     if (pClip) {
576         rect.Intersect(*pClip);
577         if (rect.IsEmpty()) {
578             return NULL;
579         }
580     }
581     CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;
582     if (!pMask) {
583         return NULL;
584     }
585     if (!pMask->Create(rect.Width(), rect.Height(), FXDIB_8bppMask)) {
586         delete pMask;
587         return NULL;
588     }
589     for (int row = rect.top; row < rect.bottom; row ++) {
590         FX_LPCBYTE src_scan = GetScanline(row) + rect.left * 4 + 3;
591         FX_LPBYTE dest_scan = (FX_LPBYTE)pMask->GetScanline(row - rect.top);
592         for (int col = rect.left; col < rect.right; col ++) {
593             *dest_scan ++ = *src_scan;
594             src_scan += 4;
595         }
596     }
597     return pMask;
598 }
599 FX_BOOL CFX_DIBSource::CopyAlphaMask(const CFX_DIBSource* pAlphaMask, const FX_RECT* pClip)
600 {
601     if (!HasAlpha() || GetFormat() == FXDIB_Argb) {
602         return FALSE;
603     }
604     if (pAlphaMask) {
605         FX_RECT rect(0, 0, pAlphaMask->m_Width, pAlphaMask->m_Height);
606         if (pClip) {
607             rect.Intersect(*pClip);
608             if (rect.IsEmpty() || rect.Width() != m_Width || rect.Height() != m_Height) {
609                 return FALSE;
610             }
611         } else {
612             if (pAlphaMask->m_Width != m_Width || pAlphaMask->m_Height != m_Height) {
613                 return FALSE;
614             }
615         }
616         for (int row = 0; row < m_Height; row ++)
617             FXSYS_memcpy32((void*)m_pAlphaMask->GetScanline(row),
618                            pAlphaMask->GetScanline(row + rect.top) + rect.left, m_pAlphaMask->m_Pitch);
619     } else {
620         m_pAlphaMask->Clear(0xff000000);
621     }
622     return TRUE;
623 }
624 const int g_ChannelOffset[] = {0, 2, 1, 0, 0, 1, 2, 3, 3};
625 FX_BOOL CFX_DIBitmap::LoadChannel(FXDIB_Channel destChannel, const CFX_DIBSource* pSrcBitmap, FXDIB_Channel srcChannel)
626 {
627     if (m_pBuffer == NULL) {
628         return FALSE;
629     }
630     CFX_DIBSource* pSrcClone = (CFX_DIBSource*)pSrcBitmap;
631     CFX_DIBitmap* pDst = this;
632     int destOffset, srcOffset;
633     if (srcChannel == FXDIB_Alpha) {
634         if (!pSrcBitmap->HasAlpha() && !pSrcBitmap->IsAlphaMask()) {
635             return FALSE;
636         }
637         if (pSrcBitmap->GetBPP() == 1) {
638             pSrcClone = pSrcBitmap->CloneConvert(FXDIB_8bppMask);
639             if (pSrcClone == NULL) {
640                 return FALSE;
641             }
642         }
643         if(pSrcBitmap->GetFormat() == FXDIB_Argb) {
644             srcOffset = 3;
645         } else {
646             srcOffset = 0;
647         }
648     } else {
649         if (pSrcBitmap->IsAlphaMask()) {
650             return FALSE;
651         }
652         if (pSrcBitmap->GetBPP() < 24) {
653             if (pSrcBitmap->IsCmykImage()) {
654                 pSrcClone = pSrcBitmap->CloneConvert((FXDIB_Format)((pSrcBitmap->GetFormat() & 0xff00) | 0x20));
655             } else {
656                 pSrcClone = pSrcBitmap->CloneConvert((FXDIB_Format)((pSrcBitmap->GetFormat() & 0xff00) | 0x18));
657             }
658             if (pSrcClone == NULL) {
659                 return FALSE;
660             }
661         }
662         srcOffset = g_ChannelOffset[srcChannel];
663     }
664     if (destChannel == FXDIB_Alpha) {
665         if (IsAlphaMask()) {
666             if(!ConvertFormat(FXDIB_8bppMask)) {
667                 if (pSrcClone != pSrcBitmap) {
668                     delete pSrcClone;
669                 }
670                 return FALSE;
671             }
672             destOffset = 0;
673         } else {
674             destOffset = 0;
675             if(!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
676                 if (pSrcClone != pSrcBitmap) {
677                     delete pSrcClone;
678                 }
679                 return FALSE;
680             }
681             if (GetFormat() == FXDIB_Argb) {
682                 destOffset = 3;
683             }
684         }
685     } else {
686         if (IsAlphaMask()) {
687             if (pSrcClone != pSrcBitmap) {
688                 delete pSrcClone;
689             }
690             return FALSE;
691         }
692         if (GetBPP() < 24) {
693             if (HasAlpha()) {
694                 if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
695                     if (pSrcClone != pSrcBitmap) {
696                         delete pSrcClone;
697                     }
698                     return FALSE;
699                 }
700             } else
701 #if _FXM_PLATFORM_  == _FXM_PLATFORM_APPLE_
702                 if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb32)) {
703 #else
704                 if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb)) {
705 #endif
706                     if (pSrcClone != pSrcBitmap) {
707                         delete pSrcClone;
708                     }
709                     return FALSE;
710                 }
711         }
712         destOffset = g_ChannelOffset[destChannel];
713     }
714     if (srcChannel == FXDIB_Alpha && pSrcClone->m_pAlphaMask) {
715         CFX_DIBitmap* pAlphaMask = pSrcClone->m_pAlphaMask;
716         if (pSrcClone->GetWidth() != m_Width || pSrcClone->GetHeight() != m_Height) {
717             if (pAlphaMask) {
718                 pAlphaMask = pAlphaMask->StretchTo(m_Width, m_Height);
719                 if (pAlphaMask == NULL) {
720                     if (pSrcClone != pSrcBitmap) {
721                         delete pSrcClone;
722                     }
723                     return FALSE;
724                 }
725             }
726         }
727         if (pSrcClone != pSrcBitmap) {
728             pSrcClone->m_pAlphaMask = NULL;
729             delete pSrcClone;
730         }
731         pSrcClone = pAlphaMask;
732         srcOffset = 0;
733     } else if (pSrcClone->GetWidth() != m_Width || pSrcClone->GetHeight() != m_Height) {
734         CFX_DIBitmap* pSrcMatched = pSrcClone->StretchTo(m_Width, m_Height);
735         if (pSrcClone != pSrcBitmap) {
736             delete pSrcClone;
737         }
738         if (pSrcMatched == NULL) {
739             return FALSE;
740         }
741         pSrcClone = pSrcMatched;
742     }
743     if (destChannel == FXDIB_Alpha && m_pAlphaMask) {
744         pDst = m_pAlphaMask;
745         destOffset = 0;
746     }
747     int srcBytes = pSrcClone->GetBPP() / 8;
748     int destBytes = pDst->GetBPP() / 8;
749     for (int row = 0; row < m_Height; row ++) {
750         FX_LPBYTE dest_pos = (FX_LPBYTE)pDst->GetScanline(row) + destOffset;
751         FX_LPCBYTE src_pos = pSrcClone->GetScanline(row) + srcOffset;
752         for (int col = 0; col < m_Width; col ++) {
753             *dest_pos = *src_pos;
754             dest_pos += destBytes;
755             src_pos += srcBytes;
756         }
757     }
758     if (pSrcClone != pSrcBitmap && pSrcClone != pSrcBitmap->m_pAlphaMask) {
759         delete pSrcClone;
760     }
761     return TRUE;
762 }
763 FX_BOOL CFX_DIBitmap::LoadChannel(FXDIB_Channel destChannel, int value)
764 {
765     if (m_pBuffer == NULL) {
766         return FALSE;
767     }
768     int destOffset;
769     if (destChannel == FXDIB_Alpha) {
770         if (IsAlphaMask()) {
771             if(!ConvertFormat(FXDIB_8bppMask)) {
772                 return FALSE;
773             }
774             destOffset = 0;
775         } else {
776             destOffset = 0;
777             if(!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
778                 return FALSE;
779             }
780             if (GetFormat() == FXDIB_Argb) {
781                 destOffset = 3;
782             }
783         }
784     } else {
785         if (IsAlphaMask()) {
786             return FALSE;
787         }
788         if (GetBPP() < 24) {
789             if (HasAlpha()) {
790                 if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
791                     return FALSE;
792                 }
793             } else
794 #if _FXM_PLATFORM_  == _FXM_PLATFORM_APPLE_
795                 if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb)) {
796                     return FALSE;
797                 }
798 #else
799                 if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb32)) {
800                     return FALSE;
801                 }
802 #endif
803         }
804         destOffset = g_ChannelOffset[destChannel];
805     }
806     int Bpp = GetBPP() / 8;
807     if (Bpp == 1) {
808         FXSYS_memset8(m_pBuffer, value, m_Height * m_Pitch);
809         return TRUE;
810     }
811     if (destChannel == FXDIB_Alpha && m_pAlphaMask) {
812         FXSYS_memset8(m_pAlphaMask->GetBuffer(), value, m_pAlphaMask->GetHeight()*m_pAlphaMask->GetPitch());
813         return TRUE;
814     }
815     for (int row = 0; row < m_Height; row ++) {
816         FX_LPBYTE scan_line = m_pBuffer + row * m_Pitch + destOffset;
817         for (int col = 0; col < m_Width; col ++) {
818             *scan_line = value;
819             scan_line += Bpp;
820         }
821     }
822     return TRUE;
823 }
824 FX_BOOL CFX_DIBitmap::MultiplyAlpha(const CFX_DIBSource* pSrcBitmap)
825 {
826     if (m_pBuffer == NULL) {
827         return FALSE;
828     }
829     ASSERT(pSrcBitmap->IsAlphaMask());
830     if (!pSrcBitmap->IsAlphaMask()) {
831         return FALSE;
832     }
833     if (!IsAlphaMask() && !HasAlpha()) {
834         return LoadChannel(FXDIB_Alpha, pSrcBitmap, FXDIB_Alpha);
835     }
836     CFX_DIBitmap* pSrcClone = (CFX_DIBitmap*)pSrcBitmap;
837     if (pSrcBitmap->GetWidth() != m_Width || pSrcBitmap->GetHeight() != m_Height) {
838         pSrcClone = pSrcBitmap->StretchTo(m_Width, m_Height);
839         ASSERT(pSrcClone != NULL);
840         if (pSrcClone == NULL) {
841             return FALSE;
842         }
843     }
844     if (IsAlphaMask()) {
845         if(!ConvertFormat(FXDIB_8bppMask)) {
846             if (pSrcClone != pSrcBitmap) {
847                 delete pSrcClone;
848             }
849             return FALSE;
850         }
851         for (int row = 0; row < m_Height; row ++) {
852             FX_LPBYTE dest_scan = m_pBuffer + m_Pitch * row;
853             FX_LPBYTE src_scan = pSrcClone->m_pBuffer + pSrcClone->m_Pitch * row;
854             if (pSrcClone->GetBPP() == 1) {
855                 for (int col = 0; col < m_Width; col ++) {
856                     if (!((1 << (7 - col % 8)) & src_scan[col / 8])) {
857                         dest_scan[col] = 0;
858                     }
859                 }
860             } else {
861                 for (int col = 0; col < m_Width; col ++) {
862                     *dest_scan = (*dest_scan) * src_scan[col] / 255;
863                     dest_scan ++;
864                 }
865             }
866         }
867     } else {
868         if(GetFormat() == FXDIB_Argb) {
869             if (pSrcClone->GetBPP() == 1) {
870                 if (pSrcClone != pSrcBitmap) {
871                     delete pSrcClone;
872                 }
873                 return FALSE;
874             }
875             for (int row = 0; row < m_Height; row ++) {
876                 FX_LPBYTE dest_scan = m_pBuffer + m_Pitch * row + 3;
877                 FX_LPBYTE src_scan = pSrcClone->m_pBuffer + pSrcClone->m_Pitch * row;
878                 for (int col = 0; col < m_Width; col ++) {
879                     *dest_scan = (*dest_scan) * src_scan[col] / 255;
880                     dest_scan += 4;
881                 }
882             }
883         } else {
884             m_pAlphaMask->MultiplyAlpha(pSrcClone);
885         }
886     }
887     if (pSrcClone != pSrcBitmap) {
888         delete pSrcClone;
889     }
890     return TRUE;
891 }
892 FX_BOOL CFX_DIBitmap::GetGrayData(void* pIccTransform)
893 {
894     if (m_pBuffer == NULL) {
895         return FALSE;
896     }
897     switch (GetFormat()) {
898         case FXDIB_1bppRgb: {
899                 if (m_pPalette == NULL) {
900                     return FALSE;
901                 }
902                 FX_BYTE gray[2];
903                 for (int i = 0; i < 2; i ++) {
904                     int r = (FX_BYTE)(m_pPalette[i] >> 16);
905                     int g = (FX_BYTE)(m_pPalette[i] >> 8);
906                     int b = (FX_BYTE)m_pPalette[i];
907                     gray[i] = (FX_BYTE)FXRGB2GRAY(r, g, b);
908                 }
909                 CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;
910                 if (!pMask) {
911                     return FALSE;
912                 }
913                 if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
914                     delete pMask;
915                     return FALSE;
916                 }
917                 FXSYS_memset8(pMask->GetBuffer(), gray[0], pMask->GetPitch() * m_Height);
918                 for (int row = 0; row < m_Height; row ++) {
919                     FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;
920                     FX_LPBYTE dest_pos = (FX_LPBYTE)pMask->GetScanline(row);
921                     for (int col = 0; col < m_Width; col ++) {
922                         if (src_pos[col / 8] & (1 << (7 - col % 8))) {
923                             *dest_pos = gray[1];
924                         }
925                         dest_pos ++;
926                     }
927                 }
928                 TakeOver(pMask);
929                 delete pMask;
930                 break;
931             }
932         case FXDIB_8bppRgb: {
933                 if (m_pPalette == NULL) {
934                     return FALSE;
935                 }
936                 FX_BYTE gray[256];
937                 for (int i = 0; i < 256; i ++) {
938                     int r = (FX_BYTE)(m_pPalette[i] >> 16);
939                     int g = (FX_BYTE)(m_pPalette[i] >> 8);
940                     int b = (FX_BYTE)m_pPalette[i];
941                     gray[i] = (FX_BYTE)FXRGB2GRAY(r, g, b);
942                 }
943                 CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;
944                 if (!pMask) {
945                     return FALSE;
946                 }
947                 if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
948                     delete pMask;
949                     return FALSE;
950                 }
951                 for (int row = 0; row < m_Height; row ++) {
952                     FX_LPBYTE dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();
953                     FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;
954                     for (int col = 0; col < m_Width; col ++) {
955                         *dest_pos ++ = gray[*src_pos ++];
956                     }
957                 }
958                 TakeOver(pMask);
959                 delete pMask;
960                 break;
961             }
962         case FXDIB_Rgb: {
963                 CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;
964                 if (!pMask) {
965                     return FALSE;
966                 }
967                 if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
968                     delete pMask;
969                     return FALSE;
970                 }
971                 for (int row = 0; row < m_Height; row ++) {
972                     FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;
973                     FX_LPBYTE dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();
974                     for (int col = 0; col < m_Width; col ++) {
975                         *dest_pos ++ = FXRGB2GRAY(src_pos[2], src_pos[1], *src_pos);
976                         src_pos += 3;
977                     }
978                 }
979                 TakeOver(pMask);
980                 delete pMask;
981                 break;
982             }
983         case FXDIB_Rgb32: {
984                 CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;
985                 if (!pMask) {
986                     return FALSE;
987                 }
988                 if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
989                     delete pMask;
990                     return FALSE;
991                 }
992                 for (int row = 0; row < m_Height; row ++) {
993                     FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;
994                     FX_LPBYTE dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();
995                     for (int col = 0; col < m_Width; col ++) {
996                         *dest_pos ++ = FXRGB2GRAY(src_pos[2], src_pos[1], *src_pos);
997                         src_pos += 4;
998                     }
999                 }
1000                 TakeOver(pMask);
1001                 delete pMask;
1002                 break;
1003             }
1004         default:
1005             return FALSE;
1006     }
1007     return TRUE;
1008 }
1009 FX_BOOL CFX_DIBitmap::MultiplyAlpha(int alpha)
1010 {
1011     if (m_pBuffer == NULL) {
1012         return FALSE;
1013     }
1014     switch (GetFormat()) {
1015         case FXDIB_1bppMask:
1016             if (!ConvertFormat(FXDIB_8bppMask)) {
1017                 return FALSE;
1018             }
1019             MultiplyAlpha(alpha);
1020             break;
1021         case FXDIB_8bppMask: {
1022                 for (int row = 0; row < m_Height; row ++) {
1023                     FX_LPBYTE scan_line = m_pBuffer + row * m_Pitch;
1024                     for (int col = 0; col < m_Width; col ++) {
1025                         scan_line[col] = scan_line[col] * alpha / 255;
1026                     }
1027                 }
1028                 break;
1029             }
1030         case FXDIB_Argb: {
1031                 for (int row = 0; row < m_Height; row ++) {
1032                     FX_LPBYTE scan_line = m_pBuffer + row * m_Pitch + 3;
1033                     for (int col = 0; col < m_Width; col ++) {
1034                         *scan_line = (*scan_line) * alpha / 255;
1035                         scan_line += 4;
1036                     }
1037                 }
1038                 break;
1039             }
1040         default:
1041             if (HasAlpha()) {
1042                 m_pAlphaMask->MultiplyAlpha(alpha);
1043             } else if (IsCmykImage()) {
1044                 if (!ConvertFormat((FXDIB_Format)(GetFormat() | 0x0200))) {
1045                     return FALSE;
1046                 }
1047                 m_pAlphaMask->MultiplyAlpha(alpha);
1048             } else {
1049                 if (!ConvertFormat(FXDIB_Argb)) {
1050                     return FALSE;
1051                 }
1052                 MultiplyAlpha(alpha);
1053             }
1054             break;
1055     }
1056     return TRUE;
1057 }
1058 #if !defined(_FPDFAPI_MINI_) || defined(_FXCORE_FEATURE_ALL_)
1059 FX_DWORD CFX_DIBitmap::GetPixel(int x, int y) const
1060 {
1061     if (m_pBuffer == NULL) {
1062         return 0;
1063     }
1064     FX_LPBYTE pos = m_pBuffer + y * m_Pitch + x * GetBPP() / 8;
1065     switch (GetFormat()) {
1066         case FXDIB_1bppMask: {
1067                 if ((*pos) & (1 << (7 - x % 8))) {
1068                     return 0xff000000;
1069                 }
1070                 return 0;
1071             }
1072         case FXDIB_1bppRgb: {
1073                 if ((*pos) & (1 << (7 - x % 8))) {
1074                     return m_pPalette ? m_pPalette[1] : 0xffffffff;
1075                 } else {
1076                     return m_pPalette ? m_pPalette[0] : 0xff000000;
1077                 }
1078                 break;
1079             }
1080         case FXDIB_8bppMask:
1081             return (*pos) << 24;
1082         case FXDIB_8bppRgb:
1083             return m_pPalette ? m_pPalette[*pos] : (0xff000000 | ((*pos) * 0x10101));
1084         case FXDIB_Rgb:
1085         case FXDIB_Rgba:
1086         case FXDIB_Rgb32:
1087             return FXARGB_GETDIB(pos) | 0xff000000;
1088         case FXDIB_Argb:
1089             return FXARGB_GETDIB(pos);
1090         default:
1091             break;
1092     }
1093     return 0;
1094 }
1095 #endif
1096 void CFX_DIBitmap::SetPixel(int x, int y, FX_DWORD color)
1097 {
1098     if (m_pBuffer == NULL) {
1099         return;
1100     }
1101     if (x < 0 || x >= m_Width || y < 0 || y >= m_Height) {
1102         return;
1103     }
1104     FX_LPBYTE pos = m_pBuffer + y * m_Pitch + x * GetBPP() / 8;
1105     switch (GetFormat()) {
1106         case FXDIB_1bppMask:
1107             if (color >> 24) {
1108                 *pos |= 1 << (7 - x % 8);
1109             } else {
1110                 *pos &= ~(1 << (7 - x % 8));
1111             }
1112             break;
1113         case FXDIB_1bppRgb:
1114             if (m_pPalette) {
1115                 if (color == m_pPalette[1]) {
1116                     *pos |= 1 << (7 - x % 8);
1117                 } else {
1118                     *pos &= ~(1 << (7 - x % 8));
1119                 }
1120             } else {
1121                 if (color == 0xffffffff) {
1122                     *pos |= 1 << (7 - x % 8);
1123                 } else {
1124                     *pos &= ~(1 << (7 - x % 8));
1125                 }
1126             }
1127             break;
1128         case FXDIB_8bppMask:
1129             *pos = (FX_BYTE)(color >> 24);
1130             break;
1131         case FXDIB_8bppRgb: {
1132                 if (m_pPalette) {
1133                     for (int i = 0; i < 256; i ++) {
1134                         if (m_pPalette[i] == color) {
1135                             *pos = (FX_BYTE)i;
1136                             return;
1137                         }
1138                     }
1139                     *pos = 0;
1140                 } else {
1141                     *pos = FXRGB2GRAY(FXARGB_R(color), FXARGB_G(color), FXARGB_B(color));
1142                 }
1143                 break;
1144             }
1145         case FXDIB_Rgb:
1146         case FXDIB_Rgb32: {
1147                 int alpha = FXARGB_A(color);
1148                 pos[0] = (FXARGB_B(color) * alpha + pos[0] * (255 - alpha)) / 255;
1149                 pos[1] = (FXARGB_G(color) * alpha + pos[1] * (255 - alpha)) / 255;
1150                 pos[2] = (FXARGB_R(color) * alpha + pos[2] * (255 - alpha)) / 255;
1151                 break;
1152             }
1153         case FXDIB_Rgba: {
1154                 pos[0] = FXARGB_B(color);
1155                 pos[1] = FXARGB_G(color);
1156                 pos[2] = FXARGB_R(color);
1157                 break;
1158             }
1159         case FXDIB_Argb:
1160             FXARGB_SETDIB(pos, color);
1161             break;
1162         default:
1163             break;
1164     }
1165 }
1166 void CFX_DIBitmap::DownSampleScanline(int line, FX_LPBYTE dest_scan, int dest_bpp,
1167                                       int dest_width, FX_BOOL bFlipX, int clip_left, int clip_width) const
1168 {
1169     if (m_pBuffer == NULL) {
1170         return;
1171     }
1172     int src_Bpp = m_bpp / 8;
1173     FX_LPBYTE scanline = m_pBuffer + line * m_Pitch;
1174     if (src_Bpp == 0) {
1175         for (int i = 0; i < clip_width; i ++) {
1176             FX_DWORD dest_x = clip_left + i;
1177             FX_DWORD src_x = dest_x * m_Width / dest_width;
1178             if (bFlipX) {
1179                 src_x = m_Width - src_x - 1;
1180             }
1181 #ifdef FOXIT_CHROME_BUILD
1182             src_x %= m_Width;
1183 #endif
1184             dest_scan[i] = (scanline[src_x / 8] & (1 << (7 - src_x % 8))) ? 255 : 0;
1185         }
1186     } else if (src_Bpp == 1) {
1187         for (int i = 0; i < clip_width; i ++) {
1188             FX_DWORD dest_x = clip_left + i;
1189             FX_DWORD src_x = dest_x * m_Width / dest_width;
1190             if (bFlipX) {
1191                 src_x = m_Width - src_x - 1;
1192             }
1193 #ifdef FOXIT_CHROME_BUILD
1194             src_x %= m_Width;
1195 #endif
1196             int dest_pos = i;
1197             if (m_pPalette) {
1198                 if (!IsCmykImage()) {
1199                     dest_pos *= 3;
1200                     FX_ARGB argb = m_pPalette[scanline[src_x]];
1201                     dest_scan[dest_pos] = FXARGB_B(argb);
1202                     dest_scan[dest_pos + 1] = FXARGB_G(argb);
1203                     dest_scan[dest_pos + 2] = FXARGB_R(argb);
1204                 } else {
1205                     dest_pos *= 4;
1206                     FX_CMYK cmyk = m_pPalette[scanline[src_x]];
1207                     dest_scan[dest_pos] = FXSYS_GetCValue(cmyk);
1208                     dest_scan[dest_pos + 1] = FXSYS_GetMValue(cmyk);
1209                     dest_scan[dest_pos + 2] = FXSYS_GetYValue(cmyk);
1210                     dest_scan[dest_pos + 3] = FXSYS_GetKValue(cmyk);
1211                 }
1212             } else {
1213                 dest_scan[dest_pos] = scanline[src_x];
1214             }
1215         }
1216     } else {
1217         for (int i = 0; i < clip_width; i ++) {
1218             FX_DWORD dest_x = clip_left + i;
1219             FX_DWORD src_x = bFlipX ? (m_Width - dest_x * m_Width / dest_width - 1) * src_Bpp : (dest_x * m_Width / dest_width) * src_Bpp;
1220 #ifdef FOXIT_CHROME_BUILD
1221             src_x %= m_Width * src_Bpp;
1222 #endif
1223             int dest_pos = i * src_Bpp;
1224             for (int b = 0; b < src_Bpp; b ++) {
1225                 dest_scan[dest_pos + b] = scanline[src_x + b];
1226             }
1227         }
1228     }
1229 }
1230 FX_BOOL CFX_DIBitmap::ConvertColorScale(FX_DWORD forecolor, FX_DWORD backcolor)
1231 {
1232     ASSERT(!IsAlphaMask());
1233     if (m_pBuffer == NULL || IsAlphaMask()) {
1234         return FALSE;
1235     }
1236     int fc, fm, fy, fk, bc, bm, by, bk;
1237     int fr, fg, fb, br, bg, bb;
1238     FX_BOOL isCmykImage = IsCmykImage();
1239     if (isCmykImage) {
1240         fc = FXSYS_GetCValue(forecolor);
1241         fm = FXSYS_GetMValue(forecolor);
1242         fy = FXSYS_GetYValue(forecolor);
1243         fk = FXSYS_GetKValue(forecolor);
1244         bc = FXSYS_GetCValue(backcolor);
1245         bm = FXSYS_GetMValue(backcolor);
1246         by = FXSYS_GetYValue(backcolor);
1247         bk = FXSYS_GetKValue(backcolor);
1248     } else {
1249         fr = FXSYS_GetRValue(forecolor);
1250         fg = FXSYS_GetGValue(forecolor);
1251         fb = FXSYS_GetBValue(forecolor);
1252         br = FXSYS_GetRValue(backcolor);
1253         bg = FXSYS_GetGValue(backcolor);
1254         bb = FXSYS_GetBValue(backcolor);
1255     }
1256     if (m_bpp <= 8) {
1257         if (isCmykImage) {
1258             if (forecolor == 0xff && backcolor == 0 && m_pPalette == NULL) {
1259                 return TRUE;
1260             }
1261         } else if (forecolor == 0 && backcolor == 0xffffff && m_pPalette == NULL) {
1262             return TRUE;
1263         }
1264         if (m_pPalette == NULL) {
1265             BuildPalette();
1266         }
1267         int size = 1 << m_bpp;
1268         if (isCmykImage) {
1269             for (int i = 0; i < size; i ++) {
1270                 FX_BYTE b, g, r;
1271                 AdobeCMYK_to_sRGB1(FXSYS_GetCValue(m_pPalette[i]), FXSYS_GetMValue(m_pPalette[i]), FXSYS_GetYValue(m_pPalette[i]), FXSYS_GetKValue(m_pPalette[i]),
1272                                    r, g, b);
1273                 int gray = 255 - FXRGB2GRAY(r, g, b);
1274                 m_pPalette[i] = CmykEncode(bc + (fc - bc) * gray / 255, bm + (fm - bm) * gray / 255,
1275                                            by + (fy - by) * gray / 255, bk + (fk - bk) * gray / 255);
1276             }
1277         } else
1278             for (int i = 0; i < size; i ++) {
1279                 int gray = FXRGB2GRAY(FXARGB_R(m_pPalette[i]), FXARGB_G(m_pPalette[i]), FXARGB_B(m_pPalette[i]));
1280                 m_pPalette[i] = FXARGB_MAKE(0xff, br + (fr - br) * gray / 255, bg + (fg - bg) * gray / 255,
1281                                             bb + (fb - bb) * gray / 255);
1282             }
1283         return TRUE;
1284     }
1285     if (isCmykImage) {
1286         if (forecolor == 0xff && backcolor == 0x00) {
1287             for (int row = 0; row < m_Height; row ++) {
1288                 FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;
1289                 for (int col = 0; col < m_Width; col ++) {
1290                     FX_BYTE b, g, r;
1291                     AdobeCMYK_to_sRGB1(scanline[0], scanline[1], scanline[2], scanline[3],
1292                                        r, g, b);
1293                     *scanline ++ = 0;
1294                     *scanline ++ = 0;
1295                     *scanline ++ = 0;
1296                     *scanline ++ = 255 - FXRGB2GRAY(r, g, b);
1297                 }
1298             }
1299             return TRUE;
1300         }
1301     } else if (forecolor == 0 && backcolor == 0xffffff) {
1302         for (int row = 0; row < m_Height; row ++) {
1303             FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;
1304             int gap = m_bpp / 8 - 2;
1305             for (int col = 0; col < m_Width; col ++) {
1306                 int gray = FXRGB2GRAY(scanline[2], scanline[1], scanline[0]);
1307                 *scanline ++ = gray;
1308                 *scanline ++ = gray;
1309                 *scanline    = gray;
1310                 scanline += gap;
1311             }
1312         }
1313         return TRUE;
1314     }
1315     if (isCmykImage) {
1316         for (int row = 0; row < m_Height; row ++) {
1317             FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;
1318             for (int col = 0; col < m_Width; col ++) {
1319                 FX_BYTE b, g, r;
1320                 AdobeCMYK_to_sRGB1(scanline[0], scanline[1], scanline[2], scanline[3],
1321                                    r, g, b);
1322                 int gray = 255 - FXRGB2GRAY(r, g, b);
1323                 *scanline ++ = bc + (fc - bc) * gray / 255;
1324                 *scanline ++ = bm + (fm - bm) * gray / 255;
1325                 *scanline ++ = by + (fy - by) * gray / 255;
1326                 *scanline ++ = bk + (fk - bk) * gray / 255;
1327             }
1328         }
1329     } else
1330         for (int row = 0; row < m_Height; row ++) {
1331             FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;
1332             int gap = m_bpp / 8 - 2;
1333             for (int col = 0; col < m_Width; col ++) {
1334                 int gray = FXRGB2GRAY(scanline[2], scanline[1], scanline[0]);
1335                 *scanline ++ = bb + (fb - bb) * gray / 255;
1336                 *scanline ++ = bg + (fg - bg) * gray / 255;
1337                 *scanline    = br + (fr - br) * gray / 255;
1338                 scanline += gap;
1339             }
1340         }
1341     return TRUE;
1342 }
1343 FX_BOOL CFX_DIBitmap::DitherFS(const FX_DWORD* pPalette, int pal_size, const FX_RECT* pRect)
1344 {
1345     if (m_pBuffer == NULL) {
1346         return FALSE;
1347     }
1348     if (m_bpp != 8 && m_pPalette != NULL && m_AlphaFlag != 0) {
1349         return FALSE;
1350     }
1351     if (m_Width < 4 && m_Height < 4) {
1352         return FALSE;
1353     }
1354     FX_RECT rect(0, 0, m_Width, m_Height);
1355     if (pRect) {
1356         rect.Intersect(*pRect);
1357     }
1358     FX_BYTE translate[256];
1359     for (int i = 0; i < 256; i ++) {
1360         int err2 = 65536;
1361         for (int j = 0; j < pal_size; j ++) {
1362             FX_BYTE entry = (FX_BYTE)pPalette[j];
1363             int err = (int)entry - i;
1364             if (err * err < err2) {
1365                 err2 = err * err;
1366                 translate[i] = entry;
1367             }
1368         }
1369     }
1370     for (int row = rect.top; row < rect.bottom; row ++) {
1371         FX_LPBYTE scan = m_pBuffer + row * m_Pitch;
1372         FX_LPBYTE next_scan = m_pBuffer + (row + 1) * m_Pitch;
1373         for (int col = rect.left; col < rect.right; col ++) {
1374             int src_pixel = scan[col];
1375             int dest_pixel = translate[src_pixel];
1376             scan[col] = (FX_BYTE)dest_pixel;
1377             int error = -dest_pixel + src_pixel;
1378             if (col < rect.right - 1) {
1379                 int src = scan[col + 1];
1380                 src += error * 7 / 16;
1381                 if (src > 255) {
1382                     scan[col + 1] = 255;
1383                 } else if (src < 0) {
1384                     scan[col + 1] = 0;
1385                 } else {
1386                     scan[col + 1] = src;
1387                 }
1388             }
1389             if (col < rect.right - 1 && row < rect.bottom - 1) {
1390                 int src = next_scan[col + 1];
1391                 src += error * 1 / 16;
1392                 if (src > 255) {
1393                     next_scan[col + 1] = 255;
1394                 } else if (src < 0) {
1395                     next_scan[col + 1] = 0;
1396                 } else {
1397                     next_scan[col + 1] = src;
1398                 }
1399             }
1400             if (row < rect.bottom - 1) {
1401                 int src = next_scan[col];
1402                 src += error * 5 / 16;
1403                 if (src > 255) {
1404                     next_scan[col] = 255;
1405                 } else if (src < 0) {
1406                     next_scan[col] = 0;
1407                 } else {
1408                     next_scan[col] = src;
1409                 }
1410             }
1411             if (col > rect.left && row < rect.bottom - 1) {
1412                 int src = next_scan[col - 1];
1413                 src += error * 3 / 16;
1414                 if (src > 255) {
1415                     next_scan[col - 1] = 255;
1416                 } else if (src < 0) {
1417                     next_scan[col - 1] = 0;
1418                 } else {
1419                     next_scan[col - 1] = src;
1420                 }
1421             }
1422         }
1423     }
1424     return TRUE;
1425 }
1426 CFX_DIBitmap* CFX_DIBSource::FlipImage(FX_BOOL bXFlip, FX_BOOL bYFlip) const
1427 {
1428     CFX_DIBitmap* pFlipped = FX_NEW CFX_DIBitmap;
1429     if (!pFlipped) {
1430         return NULL;
1431     }
1432     if (!pFlipped->Create(m_Width, m_Height, GetFormat())) {
1433         delete pFlipped;
1434         return NULL;
1435     }
1436     pFlipped->CopyPalette(m_pPalette);
1437     FX_LPBYTE pDestBuffer = pFlipped->GetBuffer();
1438     int Bpp = m_bpp / 8;
1439     for (int row = 0; row < m_Height; row ++) {
1440         FX_LPCBYTE src_scan = GetScanline(row);
1441         FX_LPBYTE dest_scan = pDestBuffer + m_Pitch * (bYFlip ? (m_Height - row - 1) : row);
1442         if (!bXFlip) {
1443             FXSYS_memcpy32(dest_scan, src_scan, m_Pitch);
1444             continue;
1445         }
1446         if (m_bpp == 1) {
1447             FXSYS_memset32(dest_scan, 0, m_Pitch);
1448             for (int col = 0; col < m_Width; col ++)
1449                 if (src_scan[col / 8] & (1 << (7 - col % 8))) {
1450                     int dest_col = m_Width - col - 1;
1451                     dest_scan[dest_col / 8] |= (1 << (7 - dest_col % 8));
1452                 }
1453         } else {
1454             dest_scan += (m_Width - 1) * Bpp;
1455             if (Bpp == 1) {
1456                 for (int col = 0; col < m_Width; col ++) {
1457                     *dest_scan = *src_scan;
1458                     dest_scan --;
1459                     src_scan ++;
1460                 }
1461             } else if (Bpp == 3) {
1462                 for (int col = 0; col < m_Width; col ++) {
1463                     dest_scan[0] = src_scan[0];
1464                     dest_scan[1] = src_scan[1];
1465                     dest_scan[2] = src_scan[2];
1466                     dest_scan -= 3;
1467                     src_scan += 3;
1468                 }
1469             } else {
1470                 ASSERT(Bpp == 4);
1471                 for (int col = 0; col < m_Width; col ++) {
1472                     *(FX_DWORD*)dest_scan = *(FX_DWORD*)src_scan;
1473                     dest_scan -= 4;
1474                     src_scan += 4;
1475                 }
1476             }
1477         }
1478     }
1479     if (m_pAlphaMask) {
1480         pDestBuffer = pFlipped->m_pAlphaMask->GetBuffer();
1481         FX_DWORD dest_pitch = pFlipped->m_pAlphaMask->GetPitch();
1482         for (int row = 0; row < m_Height; row ++) {
1483             FX_LPCBYTE src_scan = m_pAlphaMask->GetScanline(row);
1484             FX_LPBYTE dest_scan = pDestBuffer + dest_pitch * (bYFlip ? (m_Height - row - 1) : row);
1485             if (!bXFlip) {
1486                 FXSYS_memcpy32(dest_scan, src_scan, dest_pitch);
1487                 continue;
1488             }
1489             dest_scan += (m_Width - 1);
1490             for (int col = 0; col < m_Width; col ++) {
1491                 *dest_scan = *src_scan;
1492                 dest_scan --;
1493                 src_scan ++;
1494             }
1495         }
1496     }
1497     return pFlipped;
1498 }
1499 CFX_DIBExtractor::CFX_DIBExtractor(const CFX_DIBSource* pSrc)
1500 {
1501     m_pBitmap = NULL;
1502     if (pSrc->GetBuffer() == NULL) {
1503         m_pBitmap = pSrc->Clone();
1504     } else {
1505         m_pBitmap = FX_NEW CFX_DIBitmap;
1506         if (!m_pBitmap) {
1507             return;
1508         }
1509         if (!m_pBitmap->Create(pSrc->GetWidth(), pSrc->GetHeight(), pSrc->GetFormat(), pSrc->GetBuffer())) {
1510             delete m_pBitmap;
1511             m_pBitmap = NULL;
1512             return;
1513         }
1514         m_pBitmap->CopyPalette(pSrc->GetPalette());
1515         m_pBitmap->CopyAlphaMask(pSrc->m_pAlphaMask);
1516     }
1517 }
1518 CFX_DIBExtractor::~CFX_DIBExtractor()
1519 {
1520     if (m_pBitmap) {
1521         delete m_pBitmap;
1522     }
1523 }
1524 CFX_FilteredDIB::CFX_FilteredDIB()
1525 {
1526     m_pScanline = NULL;
1527     m_pSrc = NULL;
1528 }
1529 CFX_FilteredDIB::~CFX_FilteredDIB()
1530 {
1531     if (m_pSrc && m_bAutoDropSrc) {
1532         delete m_pSrc;
1533     }
1534     if (m_pScanline) {
1535         FX_Free(m_pScanline);
1536     }
1537 }
1538 void CFX_FilteredDIB::LoadSrc(const CFX_DIBSource* pSrc, FX_BOOL bAutoDropSrc)
1539 {
1540     m_pSrc = pSrc;
1541     m_bAutoDropSrc = bAutoDropSrc;
1542     m_Width = pSrc->GetWidth();
1543     m_Height = pSrc->GetHeight();
1544     FXDIB_Format format = GetDestFormat();
1545     m_bpp = (FX_BYTE)format;
1546     m_AlphaFlag = (FX_BYTE)(format >> 8);
1547     m_Pitch = (m_Width * (format & 0xff) + 31) / 32 * 4;
1548     m_pPalette = GetDestPalette();
1549     m_pScanline = FX_Alloc(FX_BYTE, m_Pitch);
1550 }
1551 FX_LPCBYTE CFX_FilteredDIB::GetScanline(int line) const
1552 {
1553     TranslateScanline(m_pScanline, m_pSrc->GetScanline(line));
1554     return m_pScanline;
1555 }
1556 void CFX_FilteredDIB::DownSampleScanline(int line, FX_LPBYTE dest_scan, int dest_bpp,
1557         int dest_width, FX_BOOL bFlipX, int clip_left, int clip_width) const
1558 {
1559     m_pSrc->DownSampleScanline(line, dest_scan, dest_bpp, dest_width, bFlipX, clip_left, clip_width);
1560     TranslateDownSamples(dest_scan, dest_scan, clip_width, dest_bpp);
1561 }
1562 CFX_ImageRenderer::CFX_ImageRenderer()
1563 {
1564     m_Status = 0;
1565     m_pTransformer = NULL;
1566     m_bRgbByteOrder = FALSE;
1567     m_BlendType = FXDIB_BLEND_NORMAL;
1568 }
1569 CFX_ImageRenderer::~CFX_ImageRenderer()
1570 {
1571     if (m_pTransformer) {
1572         delete m_pTransformer;
1573     }
1574 }
1575 extern FX_RECT _FXDIB_SwapClipBox(FX_RECT& clip, int width, int height, FX_BOOL bFlipX, FX_BOOL bFlipY);
1576 FX_BOOL CFX_ImageRenderer::Start(CFX_DIBitmap* pDevice, const CFX_ClipRgn* pClipRgn,
1577                                  const CFX_DIBSource* pSource, int bitmap_alpha,
1578                                  FX_DWORD mask_color, const CFX_AffineMatrix* pMatrix,
1579                                  FX_DWORD dib_flags, FX_BOOL bRgbByteOrder,
1580                                  int alpha_flag, void* pIccTransform, int blend_type)
1581 {
1582     m_Matrix = *pMatrix;
1583     CFX_FloatRect image_rect_f = m_Matrix.GetUnitRect();
1584     FX_RECT image_rect = image_rect_f.GetOutterRect();
1585     m_ClipBox = pClipRgn ? pClipRgn->GetBox() : FX_RECT(0, 0, pDevice->GetWidth(), pDevice->GetHeight());
1586     m_ClipBox.Intersect(image_rect);
1587     if (m_ClipBox.IsEmpty()) {
1588         return FALSE;
1589     }
1590     m_pDevice = pDevice;
1591     m_pClipRgn = pClipRgn;
1592     m_MaskColor = mask_color;
1593     m_BitmapAlpha = bitmap_alpha;
1594     m_Matrix = *pMatrix;
1595     m_Flags = dib_flags;
1596     m_AlphaFlag = alpha_flag;
1597     m_pIccTransform = pIccTransform;
1598     m_bRgbByteOrder = bRgbByteOrder;
1599     m_BlendType = blend_type;
1600     FX_BOOL ret = TRUE;
1601     if ((FXSYS_fabs(m_Matrix.b) >= 0.5f || m_Matrix.a == 0) ||
1602             (FXSYS_fabs(m_Matrix.c) >= 0.5f || m_Matrix.d == 0) ) {
1603         if (FXSYS_fabs(m_Matrix.a) < FXSYS_fabs(m_Matrix.b) / 20 && FXSYS_fabs(m_Matrix.d) < FXSYS_fabs(m_Matrix.c) / 20 &&
1604                 FXSYS_fabs(m_Matrix.a) < 0.5f && FXSYS_fabs(m_Matrix.d) < 0.5f) {
1605             int dest_width = image_rect.Width();
1606             int dest_height = image_rect.Height();
1607             FX_RECT bitmap_clip = m_ClipBox;
1608             bitmap_clip.Offset(-image_rect.left, -image_rect.top);
1609             bitmap_clip = _FXDIB_SwapClipBox(bitmap_clip, dest_width, dest_height, m_Matrix.c > 0, m_Matrix.b < 0);
1610             m_Composer.Compose(pDevice, pClipRgn, bitmap_alpha, mask_color, m_ClipBox, TRUE,
1611                                m_Matrix.c > 0, m_Matrix.b < 0, m_bRgbByteOrder, alpha_flag, pIccTransform, m_BlendType);
1612             if (!m_Stretcher.Start(&m_Composer, pSource, dest_height, dest_width, bitmap_clip, dib_flags)) {
1613                 return FALSE;
1614             }
1615             m_Status = 1;
1616             return TRUE;
1617         }
1618         m_Status = 2;
1619         m_pTransformer = FX_NEW CFX_ImageTransformer;
1620         if (!m_pTransformer) {
1621             return FALSE;
1622         }
1623         m_pTransformer->Start(pSource, &m_Matrix, dib_flags, &m_ClipBox);
1624         return TRUE;
1625     }
1626     int dest_width = image_rect.Width();
1627     if (m_Matrix.a < 0) {
1628         dest_width = -dest_width;
1629     }
1630     int dest_height = image_rect.Height();
1631     if (m_Matrix.d > 0) {
1632         dest_height = -dest_height;
1633     }
1634     if (dest_width == 0 || dest_height == 0) {
1635         return FALSE;
1636     }
1637     FX_RECT bitmap_clip = m_ClipBox;
1638     bitmap_clip.Offset(-image_rect.left, -image_rect.top);
1639     m_Composer.Compose(pDevice, pClipRgn, bitmap_alpha, mask_color,
1640                        m_ClipBox, FALSE, FALSE, FALSE, m_bRgbByteOrder, alpha_flag, pIccTransform, m_BlendType);
1641     m_Status = 1;
1642     ret = m_Stretcher.Start(&m_Composer, pSource, dest_width, dest_height, bitmap_clip, dib_flags);
1643     return ret;
1644 }
1645 FX_BOOL CFX_ImageRenderer::Continue(IFX_Pause* pPause)
1646 {
1647     if (m_Status == 1) {
1648         return m_Stretcher.Continue(pPause);
1649     } else if (m_Status == 2) {
1650         if (m_pTransformer->Continue(pPause)) {
1651             return TRUE;
1652         }
1653         CFX_DIBitmap* pBitmap = m_pTransformer->m_Storer.Detach();
1654         if (pBitmap == NULL) {
1655             return FALSE;
1656         }
1657         if (pBitmap->GetBuffer() == NULL) {
1658             delete pBitmap;
1659             return FALSE;
1660         }
1661         if (pBitmap->IsAlphaMask()) {
1662             if (m_BitmapAlpha != 255) {
1663                 if (m_AlphaFlag >> 8) {
1664                     m_AlphaFlag = (((FX_BYTE)((m_AlphaFlag & 0xff) * m_BitmapAlpha / 255)) | ((m_AlphaFlag >> 8) << 8));
1665                 } else {
1666                     m_MaskColor = FXARGB_MUL_ALPHA(m_MaskColor, m_BitmapAlpha);
1667                 }
1668             }
1669             m_pDevice->CompositeMask(m_pTransformer->m_ResultLeft, m_pTransformer->m_ResultTop,
1670                                      pBitmap->GetWidth(), pBitmap->GetHeight(), pBitmap, m_MaskColor,
1671                                      0, 0, m_BlendType, m_pClipRgn, m_bRgbByteOrder, m_AlphaFlag, m_pIccTransform);
1672         } else {
1673             if (m_BitmapAlpha != 255) {
1674                 pBitmap->MultiplyAlpha(m_BitmapAlpha);
1675             }
1676             m_pDevice->CompositeBitmap(m_pTransformer->m_ResultLeft, m_pTransformer->m_ResultTop,
1677                                        pBitmap->GetWidth(), pBitmap->GetHeight(), pBitmap, 0, 0, m_BlendType, m_pClipRgn, m_bRgbByteOrder, m_pIccTransform);
1678         }
1679         delete pBitmap;
1680         return FALSE;
1681     }
1682     return FALSE;
1683 }
1684 CFX_BitmapStorer::CFX_BitmapStorer()
1685 {
1686     m_pBitmap = NULL;
1687 }
1688 CFX_BitmapStorer::~CFX_BitmapStorer()
1689 {
1690     if (m_pBitmap) {
1691         delete m_pBitmap;
1692     }
1693 }
1694 CFX_DIBitmap* CFX_BitmapStorer::Detach()
1695 {
1696     CFX_DIBitmap* pBitmap = m_pBitmap;
1697     m_pBitmap = NULL;
1698     return pBitmap;
1699 }
1700 void CFX_BitmapStorer::Replace(CFX_DIBitmap* pBitmap)
1701 {
1702     if (m_pBitmap) {
1703         delete m_pBitmap;
1704     }
1705     m_pBitmap = pBitmap;
1706 }
1707 void CFX_BitmapStorer::ComposeScanline(int line, FX_LPCBYTE scanline, FX_LPCBYTE scan_extra_alpha)
1708 {
1709     FX_LPBYTE dest_buf = (FX_LPBYTE)m_pBitmap->GetScanline(line);
1710     FX_LPBYTE dest_alpha_buf = m_pBitmap->m_pAlphaMask ?
1711                                (FX_LPBYTE)m_pBitmap->m_pAlphaMask->GetScanline(line) : NULL;
1712     if (dest_buf) {
1713         FXSYS_memcpy32(dest_buf, scanline, m_pBitmap->GetPitch());
1714     }
1715     if (dest_alpha_buf) {
1716         FXSYS_memcpy32(dest_alpha_buf, scan_extra_alpha, m_pBitmap->m_pAlphaMask->GetPitch());
1717     }
1718 }
1719 FX_BOOL CFX_BitmapStorer::SetInfo(int width, int height, FXDIB_Format src_format, FX_DWORD* pSrcPalette)
1720 {
1721     m_pBitmap = FX_NEW CFX_DIBitmap;
1722     if (!m_pBitmap) {
1723         return FALSE;
1724     }
1725     if (!m_pBitmap->Create(width, height, src_format)) {
1726         delete m_pBitmap;
1727         m_pBitmap = NULL;
1728         return FALSE;
1729     }
1730     if (pSrcPalette) {
1731         m_pBitmap->CopyPalette(pSrcPalette);
1732     }
1733     return TRUE;
1734 }