1 /*------------------------------------
2 * VisualPng.C -- Shows a PNG image
3 *------------------------------------
5 * Copyright 2000, Willem van Schaik.
7 * This code is released under the libpng license.
8 * For conditions of distribution and use, see the disclaimer
16 #define PROGNAME "VisualPng"
17 #define LONGNAME "Win32 Viewer for PNG-files"
18 #define VERSION "1.0 of 2000 June 07"
24 /* standard includes */
32 /* application includes */
40 /* function prototypes */
42 LRESULT CALLBACK
WndProc (HWND
, UINT
, WPARAM
, LPARAM
);
43 BOOL CALLBACK
AboutDlgProc (HWND
, UINT
, WPARAM
, LPARAM
) ;
45 BOOL
CenterAbout (HWND hwndChild
, HWND hwndParent
);
47 BOOL
BuildPngList (PTSTR pstrPathName
, TCHAR
**ppFileList
, int *pFileCount
,
50 BOOL
SearchPngList (TCHAR
*pFileList
, int FileCount
, int *pFileIndex
,
51 PTSTR pstrPrevName
, PTSTR pstrNextName
);
53 BOOL
LoadImageFile(HWND hwnd
, PTSTR pstrPathName
,
54 png_byte
**ppbImage
, int *pxImgSize
, int *pyImgSize
, int *piChannels
,
55 png_color
*pBkgColor
);
57 BOOL
DisplayImage (HWND hwnd
, BYTE
**ppDib
,
58 BYTE
**ppDiData
, int cxWinSize
, int cyWinSize
,
59 BYTE
*pbImage
, int cxImgSize
, int cyImgSize
, int cImgChannels
,
63 BYTE
*pDiData
, int cxWinSize
, int cyWinSize
);
66 BYTE
*pDiData
, int cxWinSize
, int cyWinSize
,
67 BYTE
*pbImage
, int cxImgSize
, int cyImgSize
, int cImgChannels
,
70 /* a few global variables */
72 static char *szProgName
= PROGNAME
;
73 static char *szAppName
= LONGNAME
;
74 static char *szIconName
= PROGNAME
;
75 static char szCmdFileName
[MAX_PATH
];
79 int WINAPI
WinMain (HINSTANCE hInstance
, HINSTANCE hPrevInstance
,
80 PSTR szCmdLine
, int iCmdShow
)
86 int ixBorders
, iyBorders
;
88 wndclass
.style
= CS_HREDRAW
| CS_VREDRAW
;
89 wndclass
.lpfnWndProc
= WndProc
;
90 wndclass
.cbClsExtra
= 0;
91 wndclass
.cbWndExtra
= 0;
92 wndclass
.hInstance
= hInstance
;
93 wndclass
.hIcon
= LoadIcon (hInstance
, szIconName
) ;
94 wndclass
.hCursor
= LoadCursor (NULL
, IDC_ARROW
);
95 wndclass
.hbrBackground
= NULL
; /* (HBRUSH) GetStockObject (GRAY_BRUSH); */
96 wndclass
.lpszMenuName
= szProgName
;
97 wndclass
.lpszClassName
= szProgName
;
99 if (!RegisterClass (&wndclass
))
101 MessageBox (NULL
, TEXT ("Error: this program requires Windows NT!"),
102 szProgName
, MB_ICONERROR
);
106 /* if filename given on commandline, store it */
107 if ((szCmdLine
!= NULL
) && (*szCmdLine
!= '\0'))
108 if (szCmdLine
[0] == '"')
109 strncpy (szCmdFileName
, szCmdLine
+ 1, strlen(szCmdLine
) - 2);
111 strcpy (szCmdFileName
, szCmdLine
);
113 strcpy (szCmdFileName
, "");
115 /* calculate size of window-borders */
116 ixBorders
= 2 * (GetSystemMetrics (SM_CXBORDER
) +
117 GetSystemMetrics (SM_CXDLGFRAME
));
118 iyBorders
= 2 * (GetSystemMetrics (SM_CYBORDER
) +
119 GetSystemMetrics (SM_CYDLGFRAME
)) +
120 GetSystemMetrics (SM_CYCAPTION
) +
121 GetSystemMetrics (SM_CYMENUSIZE
) +
122 1; /* WvS: don't ask me why? */
124 hwnd
= CreateWindow (szProgName
, szAppName
,
126 CW_USEDEFAULT
, CW_USEDEFAULT
,
127 512 + 2 * MARGIN
+ ixBorders
, 384 + 2 * MARGIN
+ iyBorders
,
128 /* CW_USEDEFAULT, CW_USEDEFAULT, */
129 NULL
, NULL
, hInstance
, NULL
);
131 ShowWindow (hwnd
, iCmdShow
);
134 hAccel
= LoadAccelerators (hInstance
, szProgName
);
136 while (GetMessage (&msg
, NULL
, 0, 0))
138 if (!TranslateAccelerator (hwnd
, hAccel
, &msg
))
140 TranslateMessage (&msg
);
141 DispatchMessage (&msg
);
147 LRESULT CALLBACK
WndProc (HWND hwnd
, UINT message
, WPARAM wParam
,
150 static HINSTANCE hInstance
;
152 static PAINTSTRUCT ps
;
155 static BITMAPFILEHEADER
*pbmfh
;
156 static BITMAPINFOHEADER
*pbmih
;
157 static BYTE
*pbImage
;
158 static int cxWinSize
, cyWinSize
;
159 static int cxImgSize
, cyImgSize
;
160 static int cImgChannels
;
161 static png_color bkgColor
= {127, 127, 127};
163 static BOOL bStretched
= TRUE
;
165 static BYTE
*pDib
= NULL
;
166 static BYTE
*pDiData
= NULL
;
168 static TCHAR szImgPathName
[MAX_PATH
];
169 static TCHAR szTitleName
[MAX_PATH
];
171 static TCHAR
*pPngFileList
= NULL
;
172 static int iPngFileCount
;
173 static int iPngFileIndex
;
180 hInstance
= ((LPCREATESTRUCT
) lParam
)->hInstance
;
181 PngFileInitialize (hwnd
);
183 strcpy (szImgPathName
, "");
185 /* in case we process file given on command-line */
187 if (szCmdFileName
[0] != '\0')
189 strcpy (szImgPathName
, szCmdFileName
);
191 /* read the other png-files in the directory for later */
192 /* next/previous commands */
194 BuildPngList (szImgPathName
, &pPngFileList
, &iPngFileCount
,
197 /* load the image from file */
199 if (!LoadImageFile (hwnd
, szImgPathName
,
200 &pbImage
, &cxImgSize
, &cyImgSize
, &cImgChannels
, &bkgColor
))
203 /* invalidate the client area for later update */
205 InvalidateRect (hwnd
, NULL
, TRUE
);
207 /* display the PNG into the DIBitmap */
209 DisplayImage (hwnd
, &pDib
, &pDiData
, cxWinSize
, cyWinSize
,
210 pbImage
, cxImgSize
, cyImgSize
, cImgChannels
, bStretched
);
216 cxWinSize
= LOWORD (lParam
);
217 cyWinSize
= HIWORD (lParam
);
219 /* invalidate the client area for later update */
221 InvalidateRect (hwnd
, NULL
, TRUE
);
223 /* display the PNG into the DIBitmap */
225 DisplayImage (hwnd
, &pDib
, &pDiData
, cxWinSize
, cyWinSize
,
226 pbImage
, cxImgSize
, cyImgSize
, cImgChannels
, bStretched
);
230 case WM_INITMENUPOPUP
:
231 hMenu
= GetMenu (hwnd
);
234 EnableMenuItem (hMenu
, IDM_FILE_SAVE
, MF_ENABLED
);
236 EnableMenuItem (hMenu
, IDM_FILE_SAVE
, MF_GRAYED
);
241 hMenu
= GetMenu (hwnd
);
243 switch (LOWORD (wParam
))
247 /* show the File Open dialog box */
249 if (!PngFileOpenDlg (hwnd
, szImgPathName
, szTitleName
))
252 /* read the other png-files in the directory for later */
253 /* next/previous commands */
255 BuildPngList (szImgPathName
, &pPngFileList
, &iPngFileCount
,
258 /* load the image from file */
260 if (!LoadImageFile (hwnd
, szImgPathName
,
261 &pbImage
, &cxImgSize
, &cyImgSize
, &cImgChannels
, &bkgColor
))
264 /* invalidate the client area for later update */
266 InvalidateRect (hwnd
, NULL
, TRUE
);
268 /* display the PNG into the DIBitmap */
270 DisplayImage (hwnd
, &pDib
, &pDiData
, cxWinSize
, cyWinSize
,
271 pbImage
, cxImgSize
, cyImgSize
, cImgChannels
, bStretched
);
277 /* show the File Save dialog box */
279 if (!PngFileSaveDlg (hwnd
, szImgPathName
, szTitleName
))
282 /* save the PNG to a disk file */
284 SetCursor (LoadCursor (NULL
, IDC_WAIT
));
287 bOk
= PngSaveImage (szImgPathName
, pDiData
, cxWinSize
, cyWinSize
,
291 SetCursor (LoadCursor (NULL
, IDC_ARROW
));
294 MessageBox (hwnd
, TEXT ("Error in saving the PNG image"),
295 szProgName
, MB_ICONEXCLAMATION
| MB_OK
);
300 /* read next entry in the directory */
302 if (SearchPngList (pPngFileList
, iPngFileCount
, &iPngFileIndex
,
303 NULL
, szImgPathName
))
305 if (strcmp (szImgPathName
, "") == 0)
308 /* load the image from file */
310 if (!LoadImageFile (hwnd
, szImgPathName
, &pbImage
,
311 &cxImgSize
, &cyImgSize
, &cImgChannels
, &bkgColor
))
314 /* invalidate the client area for later update */
316 InvalidateRect (hwnd
, NULL
, TRUE
);
318 /* display the PNG into the DIBitmap */
320 DisplayImage (hwnd
, &pDib
, &pDiData
, cxWinSize
, cyWinSize
,
321 pbImage
, cxImgSize
, cyImgSize
, cImgChannels
, bStretched
);
326 case IDM_FILE_PREVIOUS
:
328 /* read previous entry in the directory */
330 if (SearchPngList (pPngFileList
, iPngFileCount
, &iPngFileIndex
,
331 szImgPathName
, NULL
))
334 if (strcmp (szImgPathName
, "") == 0)
337 /* load the image from file */
339 if (!LoadImageFile (hwnd
, szImgPathName
, &pbImage
, &cxImgSize
,
340 &cyImgSize
, &cImgChannels
, &bkgColor
))
343 /* invalidate the client area for later update */
345 InvalidateRect (hwnd
, NULL
, TRUE
);
347 /* display the PNG into the DIBitmap */
349 DisplayImage (hwnd
, &pDib
, &pDiData
, cxWinSize
, cyWinSize
,
350 pbImage
, cxImgSize
, cyImgSize
, cImgChannels
, bStretched
);
357 /* more cleanup needed... */
359 /* free image buffer */
369 if (pPngFileList
!= NULL
)
381 case IDM_OPTIONS_STRETCH
:
382 bStretched
= !bStretched
;
384 CheckMenuItem (hMenu
, IDM_OPTIONS_STRETCH
, MF_CHECKED
);
386 CheckMenuItem (hMenu
, IDM_OPTIONS_STRETCH
, MF_UNCHECKED
);
388 /* invalidate the client area for later update */
390 InvalidateRect (hwnd
, NULL
, TRUE
);
392 /* display the PNG into the DIBitmap */
394 DisplayImage (hwnd
, &pDib
, &pDiData
, cxWinSize
, cyWinSize
,
395 pbImage
, cxImgSize
, cyImgSize
, cImgChannels
, bStretched
);
400 DialogBox (hInstance
, TEXT ("AboutBox"), hwnd
, AboutDlgProc
) ;
408 hdc
= BeginPaint (hwnd
, &ps
);
411 SetDIBitsToDevice (hdc
, 0, 0, cxWinSize
, cyWinSize
, 0, 0,
412 0, cyWinSize
, pDiData
, (BITMAPINFO
*) pDib
, DIB_RGB_COLORS
);
414 EndPaint (hwnd
, &ps
);
428 return DefWindowProc (hwnd
, message
, wParam
, lParam
);
431 BOOL CALLBACK
AboutDlgProc (HWND hDlg
, UINT message
,
432 WPARAM wParam
, LPARAM lParam
)
437 ShowWindow (hDlg
, SW_HIDE
);
438 CenterAbout (hDlg
, GetWindow (hDlg
, GW_OWNER
));
439 ShowWindow (hDlg
, SW_SHOW
);
443 switch (LOWORD (wParam
))
447 EndDialog (hDlg
, 0) ;
459 BOOL
CenterAbout (HWND hwndChild
, HWND hwndParent
)
461 RECT rChild
, rParent
, rWorkArea
;
462 int wChild
, hChild
, wParent
, hParent
;
466 /* Get the Height and Width of the child window */
467 GetWindowRect (hwndChild
, &rChild
);
468 wChild
= rChild
.right
- rChild
.left
;
469 hChild
= rChild
.bottom
- rChild
.top
;
471 /* Get the Height and Width of the parent window */
472 GetWindowRect (hwndParent
, &rParent
);
473 wParent
= rParent
.right
- rParent
.left
;
474 hParent
= rParent
.bottom
- rParent
.top
;
476 /* Get the limits of the 'workarea' */
477 bResult
= SystemParametersInfo(
478 SPI_GETWORKAREA
, /* system parameter to query or set */
483 rWorkArea
.left
= rWorkArea
.top
= 0;
484 rWorkArea
.right
= GetSystemMetrics(SM_CXSCREEN
);
485 rWorkArea
.bottom
= GetSystemMetrics(SM_CYSCREEN
);
488 /* Calculate new X position, then adjust for workarea */
489 xNew
= rParent
.left
+ ((wParent
- wChild
) /2);
490 if (xNew
< rWorkArea
.left
) {
491 xNew
= rWorkArea
.left
;
492 } else if ((xNew
+wChild
) > rWorkArea
.right
) {
493 xNew
= rWorkArea
.right
- wChild
;
496 /* Calculate new Y position, then adjust for workarea */
497 yNew
= rParent
.top
+ ((hParent
- hChild
) /2);
498 if (yNew
< rWorkArea
.top
) {
499 yNew
= rWorkArea
.top
;
500 } else if ((yNew
+hChild
) > rWorkArea
.bottom
) {
501 yNew
= rWorkArea
.bottom
- hChild
;
504 /* Set it, and return */
505 return SetWindowPos (hwndChild
, NULL
, xNew
, yNew
, 0, 0, SWP_NOSIZE
|
513 BOOL
BuildPngList (PTSTR pstrPathName
, TCHAR
**ppFileList
, int *pFileCount
,
516 static TCHAR szImgPathName
[MAX_PATH
];
517 static TCHAR szImgFileName
[MAX_PATH
];
518 static TCHAR szImgFindName
[MAX_PATH
];
520 WIN32_FIND_DATA finddata
;
523 static TCHAR szTmp
[MAX_PATH
];
528 /* free previous file-list */
530 if (*ppFileList
!= NULL
)
536 /* extract foldername, filename and search-name */
538 strcpy (szImgPathName
, pstrPathName
);
539 strcpy (szImgFileName
, strrchr (pstrPathName
, '\\') + 1);
541 strcpy (szImgFindName
, szImgPathName
);
542 *(strrchr (szImgFindName
, '\\') + 1) = '\0';
543 strcat (szImgFindName
, "*.png");
545 /* first cycle: count number of files in directory for memory allocation */
549 hFind
= FindFirstFile(szImgFindName
, &finddata
);
550 bOk
= (hFind
!= (HANDLE
) -1);
555 bOk
= FindNextFile(hFind
, &finddata
);
559 /* allocation memory for file-list */
561 *ppFileList
= (TCHAR
*) malloc (*pFileCount
* MAX_PATH
);
563 /* second cycle: read directory and store filenames in file-list */
565 hFind
= FindFirstFile(szImgFindName
, &finddata
);
566 bOk
= (hFind
!= (HANDLE
) -1);
572 strcpy (*ppFileList
+ ii
, szImgPathName
);
573 strcpy (strrchr(*ppFileList
+ ii
, '\\') + 1, finddata
.cFileName
);
575 if (strcmp(pstrPathName
, *ppFileList
+ ii
) == 0)
581 bOk
= FindNextFile(hFind
, &finddata
);
585 /* finally we must sort the file-list */
587 for (i
= 0; i
< *pFileCount
- 1; i
++)
590 for (j
= i
+1; j
< *pFileCount
; j
++)
593 if (strcmp (*ppFileList
+ ii
, *ppFileList
+ jj
) > 0)
595 strcpy (szTmp
, *ppFileList
+ jj
);
596 strcpy (*ppFileList
+ jj
, *ppFileList
+ ii
);
597 strcpy (*ppFileList
+ ii
, szTmp
);
599 /* check if this was the current image that we moved */
601 if (*pFileIndex
== i
)
604 if (*pFileIndex
== j
)
619 TCHAR
*pFileList
, int FileCount
, int *pFileIndex
,
620 PTSTR pstrPrevName
, PTSTR pstrNextName
)
624 /* get previous entry */
626 if (pstrPrevName
!= NULL
)
631 *pFileIndex
= FileCount
- 1;
633 strcpy (pstrPrevName
, pFileList
+ (*pFileIndex
* MAX_PATH
));
638 if (pstrNextName
!= NULL
)
640 if (*pFileIndex
< FileCount
- 1)
645 strcpy (pstrNextName
, pFileList
+ (*pFileIndex
* MAX_PATH
));
661 BOOL
LoadImageFile (HWND hwnd
, PTSTR pstrPathName
,
662 png_byte
**ppbImage
, int *pxImgSize
, int *pyImgSize
,
663 int *piChannels
, png_color
*pBkgColor
)
665 static TCHAR szTmp
[MAX_PATH
];
667 /* if there's an existing PNG, free the memory */
675 /* Load the entire PNG into memory */
677 SetCursor (LoadCursor (NULL
, IDC_WAIT
));
680 PngLoadImage (pstrPathName
, ppbImage
, pxImgSize
, pyImgSize
, piChannels
,
684 SetCursor (LoadCursor (NULL
, IDC_ARROW
));
686 if (*ppbImage
!= NULL
)
688 sprintf (szTmp
, "VisualPng - %s", strrchr(pstrPathName
, '\\') + 1);
689 SetWindowText (hwnd
, szTmp
);
693 MessageBox (hwnd
, TEXT ("Error in loading the PNG image"),
694 szProgName
, MB_ICONEXCLAMATION
| MB_OK
);
705 BOOL
DisplayImage (HWND hwnd
, BYTE
**ppDib
,
706 BYTE
**ppDiData
, int cxWinSize
, int cyWinSize
,
707 BYTE
*pbImage
, int cxImgSize
, int cyImgSize
, int cImgChannels
,
711 BYTE
*pDiData
= *ppDiData
;
712 /* BITMAPFILEHEADER *pbmfh; */
713 BITMAPINFOHEADER
*pbmih
;
715 png_color bkgBlack
= {0, 0, 0};
716 png_color bkgGray
= {127, 127, 127};
717 png_color bkgWhite
= {255, 255, 255};
719 /* allocate memory for the Device Independant bitmap */
721 wDIRowBytes
= (WORD
) ((3 * cxWinSize
+ 3L) >> 2) << 2;
729 if (!(pDib
= (BYTE
*) malloc (sizeof(BITMAPINFOHEADER
) +
730 wDIRowBytes
* cyWinSize
)))
732 MessageBox (hwnd
, TEXT ("Error in displaying the PNG image"),
733 szProgName
, MB_ICONEXCLAMATION
| MB_OK
);
734 *ppDib
= pDib
= NULL
;
738 memset (pDib
, 0, sizeof(BITMAPINFOHEADER
));
740 /* initialize the dib-structure */
742 pbmih
= (BITMAPINFOHEADER
*) pDib
;
743 pbmih
->biSize
= sizeof(BITMAPINFOHEADER
);
744 pbmih
->biWidth
= cxWinSize
;
745 pbmih
->biHeight
= -((long) cyWinSize
);
747 pbmih
->biBitCount
= 24;
748 pbmih
->biCompression
= 0;
749 pDiData
= pDib
+ sizeof(BITMAPINFOHEADER
);
752 /* first fill bitmap with gray and image border */
754 InitBitmap (pDiData
, cxWinSize
, cyWinSize
);
756 /* then fill bitmap with image */
761 pDiData
, cxWinSize
, cyWinSize
,
762 pbImage
, cxImgSize
, cyImgSize
, cImgChannels
,
773 BOOL
InitBitmap (BYTE
*pDiData
, int cxWinSize
, int cyWinSize
)
778 /* initialize the background with gray */
781 for (y
= 0; y
< cyWinSize
; y
++)
784 for (x
= 0; x
< cxWinSize
; x
++)
792 /* rows start on 4 byte boundaries */
793 while ((col
% 4) != 0)
808 BYTE
*pDiData
, int cxWinSize
, int cyWinSize
,
809 BYTE
*pbImage
, int cxImgSize
, int cyImgSize
, int cImgChannels
,
812 BYTE
*pStretchedImage
;
816 const int cDIChannels
= 3;
819 int cxNewSize
, cyNewSize
;
820 int cxImgPos
, cyImgPos
;
828 cxNewSize
= cxWinSize
- 2 * MARGIN
;
829 cyNewSize
= cyWinSize
- 2 * MARGIN
;
831 /* stretch the image to it's window determined size */
833 /* the following two are mathematically the same, but the first
834 * has side-effects because of rounding
836 /* if ((cyNewSize / cxNewSize) > (cyImgSize / cxImgSize)) */
837 if ((cyNewSize
* cxImgSize
) > (cyImgSize
* cxNewSize
))
839 cyNewSize
= cxNewSize
* cyImgSize
/ cxImgSize
;
841 cyImgPos
= (cyWinSize
- cyNewSize
) / 2;
845 cxNewSize
= cyNewSize
* cxImgSize
/ cyImgSize
;
847 cxImgPos
= (cxWinSize
- cxNewSize
) / 2;
850 pStretchedImage
= malloc (cImgChannels
* cxNewSize
* cyNewSize
);
851 pImg
= pStretchedImage
;
853 for (yNew
= 0; yNew
< cyNewSize
; yNew
++)
855 yOld
= yNew
* cyImgSize
/ cyNewSize
;
856 for (xNew
= 0; xNew
< cxNewSize
; xNew
++)
858 xOld
= xNew
* cxImgSize
/ cxNewSize
;
860 r
= *(pbImage
+ cImgChannels
* ((yOld
* cxImgSize
) + xOld
) + 0);
861 g
= *(pbImage
+ cImgChannels
* ((yOld
* cxImgSize
) + xOld
) + 1);
862 b
= *(pbImage
+ cImgChannels
* ((yOld
* cxImgSize
) + xOld
) + 2);
866 if (cImgChannels
== 4)
868 a
= *(pbImage
+ cImgChannels
* ((yOld
* cxImgSize
) + xOld
)
875 /* calculate row-bytes */
877 wImgRowBytes
= cImgChannels
* cxNewSize
;
878 wDIRowBytes
= (WORD
) ((cDIChannels
* cxWinSize
+ 3L) >> 2) << 2;
880 /* copy image to screen */
882 for (yImg
= 0, yWin
= cyImgPos
; yImg
< cyNewSize
; yImg
++, yWin
++)
884 if (yWin
>= cyWinSize
- cyImgPos
)
886 src
= pStretchedImage
+ yImg
* wImgRowBytes
;
887 dst
= pDiData
+ yWin
* wDIRowBytes
+ cxImgPos
* cDIChannels
;
889 for (xImg
= 0, xWin
= cxImgPos
; xImg
< cxNewSize
; xImg
++, xWin
++)
891 if (xWin
>= cxWinSize
- cxImgPos
)
896 *dst
++ = b
; /* note the reverse order */
899 if (cImgChannels
== 4)
908 if (pStretchedImage
!= NULL
)
910 free (pStretchedImage
);
911 pStretchedImage
= NULL
;
916 /* process the image not-stretched */
920 /* calculate the central position */
922 cxImgPos
= (cxWinSize
- cxImgSize
) / 2;
923 cyImgPos
= (cyWinSize
- cyImgSize
) / 2;
925 /* check for image larger than window */
927 if (cxImgPos
< MARGIN
)
929 if (cyImgPos
< MARGIN
)
932 /* calculate both row-bytes */
934 wImgRowBytes
= cImgChannels
* cxImgSize
;
935 wDIRowBytes
= (WORD
) ((cDIChannels
* cxWinSize
+ 3L) >> 2) << 2;
937 /* copy image to screen */
939 for (yImg
= 0, yWin
= cyImgPos
; yImg
< cyImgSize
; yImg
++, yWin
++)
941 if (yWin
>= cyWinSize
- MARGIN
)
943 src
= pbImage
+ yImg
* wImgRowBytes
;
944 dst
= pDiData
+ yWin
* wDIRowBytes
+ cxImgPos
* cDIChannels
;
946 for (xImg
= 0, xWin
= cxImgPos
; xImg
< cxImgSize
; xImg
++, xWin
++)
948 if (xWin
>= cxWinSize
- MARGIN
)
953 *dst
++ = b
; /* note the reverse order */
956 if (cImgChannels
== 4)