Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

663 lines
22 KiB

  1. //-------------------------------------------------------------------------//
  2. // Rgn.cpp - Bitmap-to-Region transforms
  3. //
  4. // History:
  5. // 01/31/2000 scotthan created
  6. //-------------------------------------------------------------------------//
  7. #include "stdafx.h"
  8. #include "rgn.h"
  9. #include "tmutils.h"
  10. //------------//
  11. // Helpers:
  12. //-------------------------------------------------------------------------//
  13. #define CX_USEDEFAULT -1
  14. #define CY_USEDEFAULT -1
  15. #define _ABS( val1, val2 ) ((val1)>(val2) ? (val1)-(val2) : (val2)-(val1))
  16. //-------------------------------------------------------------------------//
  17. inline BOOL IsColorMatch( COLORREF rgb1, COLORREF rgb2, int nTolerance = 0 )
  18. {
  19. if( nTolerance == 0 )
  20. return (rgb1 << 8) == (rgb2 << 8);
  21. return _ABS(GetRValue(rgb1),GetRValue(rgb2)) <= nTolerance &&
  22. _ABS(GetGValue(rgb1),GetGValue(rgb2)) <= nTolerance &&
  23. _ABS(GetBValue(rgb1),GetBValue(rgb2)) <= nTolerance;
  24. }
  25. //-------------------------------------------------------------------------//
  26. inline BOOL _IsNormalRect( IN LPCRECT prc )
  27. {
  28. return (prc->right >= prc->left) &&
  29. (prc->bottom >= prc->top);
  30. }
  31. //-------------------------------------------------------------------------//
  32. inline BOOL _IsOnScreenRect( IN LPCRECT prc )
  33. {
  34. return prc->left >= 0 && prc->top >= 0 &&
  35. prc->right >= 0 && prc->bottom >= 0;
  36. }
  37. //-------------------------------------------------------------------------//
  38. inline void _InPlaceUnionRect( IN OUT LPRECT prcDest, IN LPCRECT prcSrc )
  39. {
  40. ASSERT(prcDest);
  41. ASSERT(prcSrc);
  42. ASSERT(_IsNormalRect(prcSrc));
  43. if( prcDest->left == -1 || prcDest->left > prcSrc ->left )
  44. prcDest->left = prcSrc ->left;
  45. if( prcDest->right == -1 || prcDest->right < prcSrc ->right )
  46. prcDest->right = prcSrc ->right;
  47. if( prcDest->top == -1 || prcDest->top > prcSrc ->top )
  48. prcDest->top = prcSrc ->top;
  49. if( prcDest->bottom == -1 || prcDest->bottom < prcSrc ->bottom )
  50. prcDest->bottom = prcSrc ->bottom;
  51. }
  52. //-------------------------------------------------------------------------//
  53. // Walks the pixels and computes the region
  54. HRGN WINAPI _PixelsToRgn(
  55. DWORD *pdwBits,
  56. int cxImageOffset, // image cell horz offset
  57. int cyImageOffset, // image cell vert offset
  58. int cxImage, // image cell width
  59. int cyImage, // image cell height
  60. int cxSrc, // src bitmap width
  61. int cySrc, // src bitmap height
  62. BOOL fAlphaChannel,
  63. int iAlphaThreshold,
  64. COLORREF rgbMask,
  65. int nMaskTolerance )
  66. {
  67. // Establish a series of rectangles, each corresponding to a scan line (row)
  68. // in the bitmap, that will comprise the region.
  69. const UINT RECTBLOCK = 512;
  70. UINT nAllocRects = 0;
  71. HRGN hrgnRet = NULL;
  72. HGLOBAL hrgnData = GlobalAlloc( GMEM_MOVEABLE,
  73. sizeof(RGNDATAHEADER) + (sizeof(RECT) * (nAllocRects + RECTBLOCK)) );
  74. if( hrgnData )
  75. {
  76. nAllocRects += RECTBLOCK;
  77. RGNDATA* prgnData = (RGNDATA*)GlobalLock( hrgnData );
  78. LPRECT prgrc = (LPRECT)prgnData->Buffer;
  79. ZeroMemory( prgnData, sizeof(prgnData->rdh) );
  80. prgnData->rdh.dwSize = sizeof(prgnData->rdh);
  81. prgnData->rdh.iType = RDH_RECTANGLES;
  82. SetRect( &prgnData->rdh.rcBound, -1, -1, -1, -1 );
  83. // invert offset in y dimension since bits are arrayed bottom to top
  84. int cyRow0 = cySrc - (cyImage + cyImageOffset);
  85. int cyRowN = (cyRow0 + cyImage) - 1 ; // index of the last row
  86. // Compute a transparency mask if not specified.
  87. if( -1 == rgbMask )
  88. rgbMask = pdwBits[cxImageOffset + (cyRowN * cxSrc)];
  89. //---- pixels in pdwBits[] have RBG's reversed ----
  90. //---- reverse our mask to match ----
  91. rgbMask = REVERSE3(rgbMask);
  92. //---- rows in pdwBits[] are reversed (bottom to top) ----
  93. for( int y = cyRow0; y <= cyRowN; y++ ) // working bottom-to-top
  94. {
  95. //---- Scanning pixels left to right ----
  96. DWORD *pdwFirst = &pdwBits[cxImageOffset + (y * cxSrc)];
  97. DWORD *pdwLast = pdwFirst + cxImage - 1;
  98. DWORD *pdwPixel = pdwFirst;
  99. while (pdwPixel <= pdwLast)
  100. {
  101. //---- skip TRANSPARENT pixels to find next OPAQUE (on this row) ----
  102. if (fAlphaChannel)
  103. {
  104. while ((pdwPixel <= pdwLast) && (ALPHACHANNEL(*pdwPixel) < iAlphaThreshold))
  105. pdwPixel++;
  106. }
  107. else
  108. {
  109. while ((pdwPixel <= pdwLast) && (IsColorMatch(*pdwPixel, rgbMask, nMaskTolerance)))
  110. pdwPixel++;
  111. }
  112. if (pdwPixel > pdwLast) // too far; try next row
  113. break;
  114. DWORD *pdw0 = pdwPixel;
  115. pdwPixel++; // skip over current opaque pixel
  116. //---- skip OPAQUE pixels to find next TRANSPARENT (on this row) ----
  117. if (fAlphaChannel)
  118. {
  119. while ((pdwPixel <= pdwLast) && (ALPHACHANNEL(*pdwPixel) >= iAlphaThreshold))
  120. pdwPixel++;
  121. }
  122. else
  123. {
  124. while ((pdwPixel <= pdwLast) && (! IsColorMatch(*pdwPixel, rgbMask, nMaskTolerance)))
  125. pdwPixel++;
  126. }
  127. //---- got a stream of 1 or more opaque pixels on this row ----
  128. // allocate more region rects if necessary (a particularly complex line)
  129. if( prgnData->rdh.nCount >= nAllocRects )
  130. {
  131. GlobalUnlock( hrgnData );
  132. prgnData = NULL;
  133. HGLOBAL hNew = GlobalReAlloc( hrgnData,
  134. sizeof(RGNDATAHEADER) + (sizeof(RECT) * (nAllocRects + RECTBLOCK)),
  135. GMEM_MOVEABLE );
  136. if( hNew )
  137. {
  138. hrgnData = hNew;
  139. nAllocRects += RECTBLOCK;
  140. prgnData = (RGNDATA*)GlobalLock( hrgnData );
  141. prgrc = (LPRECT)prgnData->Buffer;
  142. ASSERT(prgnData);
  143. }
  144. else
  145. goto exit; // out of memory
  146. }
  147. // assign region rectangle
  148. int x0 = (int)(pdw0 - pdwFirst);
  149. int x = (int)(pdwPixel - pdwFirst);
  150. int y0 = cyRowN - y;
  151. SetRect( prgrc + prgnData->rdh.nCount,
  152. x0, y0, x, y0+1 /* each rectangle is always 1 pixel high */ );
  153. // merge into bounding box
  154. _InPlaceUnionRect( &prgnData->rdh.rcBound,
  155. prgrc + prgnData->rdh.nCount );
  156. prgnData->rdh.nCount++;
  157. } // while ()
  158. } // for(y)
  159. if( prgnData->rdh.nCount && _IsOnScreenRect(&prgnData->rdh.rcBound) )
  160. {
  161. // Create the region representing the scan line.
  162. hrgnRet = ExtCreateRegion( NULL, sizeof(RGNDATAHEADER) + (sizeof(RECT) * nAllocRects),
  163. prgnData );
  164. }
  165. exit:
  166. // Free region def block.
  167. GlobalUnlock( hrgnData );
  168. GlobalFree( hrgnData );
  169. }
  170. return hrgnRet;
  171. }
  172. //-------------------------------------------------------------------------//
  173. // Creates a region based on a text string in the indicated font.
  174. HRGN WINAPI CreateTextRgn( HFONT hf, LPCTSTR pszText )
  175. {
  176. HRGN hrgnRet = NULL;
  177. if( pszText && *pszText )
  178. {
  179. int cchText = lstrlen( pszText );
  180. // Create a composite DC for assembling the region.
  181. HDC hdcMem = CreateCompatibleDC( NULL );
  182. SetBkMode( hdcMem, TRANSPARENT );
  183. SetTextAlign( hdcMem, TA_TOP|TA_LEFT );
  184. HFONT hfOld = (HFONT)SelectObject( hdcMem, hf );
  185. // Derive a region from a path.
  186. BeginPath( hdcMem );
  187. TextOut( hdcMem, 0, 0, pszText, cchText );
  188. EndPath( hdcMem );
  189. hrgnRet = PathToRegion( hdcMem );
  190. // Clean up composite DC
  191. SelectObject( hdcMem, hfOld );
  192. DeleteDC( hdcMem );
  193. }
  194. return hrgnRet;
  195. }
  196. //-------------------------------------------------------------------------//
  197. // Creates a region based on an arbitrary bitmap, transparency-keyed on a
  198. // RGB value within a specified tolerance. The key value is optional
  199. // (-1 == use the value of the first pixel as the key).
  200. //
  201. HRESULT WINAPI CreateBitmapRgn(
  202. HBITMAP hbm,
  203. int cxOffset,
  204. int cyOffset,
  205. int cx,
  206. int cy,
  207. BOOL fAlphaChannel,
  208. int iAlphaThreshold,
  209. COLORREF rgbMask,
  210. int nMaskTolerance,
  211. OUT HRGN *phrgn)
  212. {
  213. CBitmapPixels BitmapPixels;
  214. DWORD *prgdwPixels;
  215. int cwidth, cheight;
  216. HRESULT hr = BitmapPixels.OpenBitmap(NULL, hbm, TRUE, &prgdwPixels, &cwidth, &cheight);
  217. if (FAILED(hr))
  218. return hr;
  219. if (cx <= 0)
  220. cx = cwidth;
  221. if (cy <= 0)
  222. cy = cheight;
  223. HRGN hrgn = _PixelsToRgn(prgdwPixels, cxOffset, cyOffset, cx, cy, cwidth, cheight, fAlphaChannel,
  224. iAlphaThreshold, rgbMask, nMaskTolerance);
  225. if (! hrgn)
  226. return MakeError32(E_FAIL); // unknown reason for failure
  227. *phrgn = hrgn;
  228. return S_OK;
  229. }
  230. //-------------------------------------------------------------------------//
  231. // Creates a region based on an arbitrary bitmap, transparency-keyed on a
  232. // RGB value within a specified tolerance. The key value is optional (-1 ==
  233. // use the value of the first pixel as the key).
  234. //
  235. HRGN WINAPI CreateScaledBitmapRgn(
  236. HBITMAP hbm,
  237. int cx,
  238. int cy,
  239. COLORREF rgbMask,
  240. int nMaskTolerance )
  241. {
  242. HRGN hrgnRet = NULL;
  243. BITMAP bm;
  244. if( hbm && GetObject( hbm, sizeof(bm), &bm ) )
  245. {
  246. // Create a memory DC to do the pixel walk
  247. HDC hdcMem = NULL;
  248. if( (hdcMem = CreateCompatibleDC(NULL)) != NULL )
  249. {
  250. if( CX_USEDEFAULT == cx )
  251. cx = bm.bmWidth;
  252. if( CY_USEDEFAULT == cy )
  253. cy = bm.bmHeight;
  254. // Create a 32-bit empty bitmap for the walk
  255. BITMAPINFO bmi;
  256. ZeroMemory( &bmi, sizeof(bmi) );
  257. bmi.bmiHeader.biSize = sizeof(bmi.bmiHeader);
  258. bmi.bmiHeader.biWidth = cx;
  259. bmi.bmiHeader.biHeight = cy;
  260. bmi.bmiHeader.biPlanes = 1;
  261. bmi.bmiHeader.biBitCount = 32;
  262. bmi.bmiHeader.biCompression = BI_RGB; // uncompressed.
  263. VOID* pvBits = NULL;
  264. HBITMAP hbmMem = CreateDIBSection( hdcMem, &bmi, DIB_RGB_COLORS, &pvBits, NULL, NULL );
  265. BITMAP bmMem;
  266. if( hbmMem )
  267. {
  268. // Transfer the image to our 32-bit format for the pixel walk.
  269. HBITMAP hbmMemOld = (HBITMAP)SelectObject( hdcMem, hbmMem );
  270. HDC hdc = CreateCompatibleDC( hdcMem );
  271. HBITMAP hbmOld = (HBITMAP)SelectObject( hdc, hbm );
  272. StretchBlt( hdcMem, 0, 0, cx, cy,
  273. hdc, 0, 0, bm.bmWidth, bm.bmHeight, SRCCOPY );
  274. SelectObject( hdc, hbmOld );
  275. DeleteDC( hdc );
  276. GetObject( hbmMem, sizeof(bmMem), &bmMem );
  277. ASSERT(bmMem.bmBitsPixel == 32);
  278. ASSERT(bmMem.bmWidthBytes/bmMem.bmWidth == sizeof(DWORD));
  279. LPDWORD pdwBits = (LPDWORD)bmMem.bmBits;
  280. ASSERT(pdwBits != NULL);
  281. hrgnRet = _PixelsToRgn(pdwBits, 0, 0, cx, cy, cx, cy, FALSE, 0, rgbMask, nMaskTolerance);
  282. // Delete 32-bit memory bitmap
  283. SelectObject( hdcMem, hbmMemOld );
  284. DeleteObject( hbmMem );
  285. }
  286. // Delete memory DC
  287. DeleteDC(hdcMem);
  288. }
  289. }
  290. return hrgnRet;
  291. }
  292. //-------------------------------------------------------------------------//
  293. int WINAPI AddToCompositeRgn(
  294. IN OUT HRGN* phrgnComposite,
  295. IN OUT HRGN hrgnSrc,
  296. IN int cxOffset,
  297. IN int cyOffset )
  298. {
  299. int nRet = ERROR;
  300. if( NULL != phrgnComposite && NULL != hrgnSrc )
  301. {
  302. nRet = OffsetRgn( hrgnSrc, cxOffset, cyOffset );
  303. if( nRet != ERROR )
  304. {
  305. int nMode = RGN_OR;
  306. if( NULL == *phrgnComposite )
  307. {
  308. *phrgnComposite = CreateRectRgn(0,0,1,1);
  309. if( NULL == *phrgnComposite )
  310. return ERROR;
  311. nMode = RGN_COPY;
  312. }
  313. nRet = CombineRgn( *phrgnComposite, hrgnSrc, *phrgnComposite, nMode );
  314. }
  315. }
  316. return nRet;
  317. }
  318. //-------------------------------------------------------------------------//
  319. int WINAPI RemoveFromCompositeRgn(
  320. HRGN hrgnDest,
  321. LPCRECT prcRemove )
  322. {
  323. ASSERT(hrgnDest);
  324. ASSERT(prcRemove);
  325. ASSERT(!IsRectEmpty(prcRemove));
  326. int nRet = ERROR;
  327. RECT rc = *prcRemove;
  328. HRGN hrgn;
  329. if( (hrgn = CreateRectRgnIndirect( &rc )) != NULL )
  330. {
  331. nRet = CombineRgn( hrgnDest, hrgnDest, hrgn, RGN_DIFF );
  332. DeleteObject( hrgn );
  333. }
  334. return nRet;
  335. }
  336. //-------------------------------------------------------------------------//
  337. HRGN WINAPI CreateTiledRectRgn(
  338. IN HRGN hrgnSrc,
  339. IN int cxSrc,
  340. IN int cySrc,
  341. IN int cxDest,
  342. IN int cyDest )
  343. {
  344. HRGN hrgnBound = NULL; // return value
  345. HRGN hrgnTile = _DupRgn( hrgnSrc );
  346. if( hrgnTile )
  347. {
  348. // Build up an unplaced, unclipped composite
  349. HRGN hrgnTmp = NULL;
  350. for( int y = 0; y < cyDest; y += cySrc )
  351. {
  352. for( int x = 0; x < cxDest; x += cxSrc )
  353. {
  354. AddToCompositeRgn( &hrgnTmp, hrgnTile,
  355. (x ? cxSrc : 0), (y ? cySrc : 0) );
  356. }
  357. }
  358. if( NULL != hrgnTmp )
  359. {
  360. // Clip the composite to the specified rectangle
  361. hrgnBound = CreateRectRgn( 0, 0, cxDest, cyDest );
  362. if( hrgnBound )
  363. {
  364. if( ERROR == CombineRgn( hrgnBound, hrgnTmp, hrgnBound, RGN_AND ) )
  365. {
  366. DeleteObject( hrgnBound );
  367. hrgnBound = NULL;
  368. }
  369. }
  370. DeleteObject( hrgnTmp );
  371. }
  372. DeleteObject( hrgnTile );
  373. }
  374. return hrgnBound;
  375. }
  376. //-------------------------------------------------------------------------//
  377. HRGN WINAPI _DupRgn( HRGN hrgnSrc )
  378. {
  379. if( hrgnSrc )
  380. {
  381. HRGN hrgnDest = CreateRectRgn(0,0,1,1);
  382. if (hrgnDest)
  383. {
  384. if (CombineRgn( hrgnDest, hrgnSrc, NULL, RGN_COPY ) )
  385. return hrgnDest;
  386. DeleteObject(hrgnDest);
  387. }
  388. }
  389. return NULL;
  390. }
  391. //-------------------------------------------------------------------------//
  392. void FixMarginOverlaps(int szDest, int *pm1, int *pm2)
  393. {
  394. int szSrc = (*pm1 + *pm2);
  395. if ((szSrc > szDest) && (szSrc > 0))
  396. {
  397. //---- reduce each but maintain ratio ----
  398. *pm1 = int(.5 + float(*pm1 * szDest)/float(szSrc));
  399. *pm2 = szDest - *pm1;
  400. }
  401. }
  402. //-------------------------------------------------------------------------//
  403. HRESULT _ScaleRectsAndCreateRegion(
  404. RGNDATA *prd,
  405. const RECT *prc,
  406. MARGINS *pMargins,
  407. SIZE *pszSrcImage,
  408. HRGN *phrgn)
  409. {
  410. //---- note: "prd" is region data with the 2 points in each ----
  411. //---- rectangle made relative to its grid. Also, after the points, ----
  412. //---- there is a BYTE for each point signifying the grid id (0-8) ----
  413. //---- that each point lies within. the grid is determined using ----
  414. //---- the original region with the background "margins". This is ----
  415. //---- done to make scaling the points as fast as possible. ----
  416. if (! prd) // required
  417. return MakeError32(E_POINTER);
  418. //---- easy access variables ----
  419. int lw = pMargins->cxLeftWidth;
  420. int rw = pMargins->cxRightWidth;
  421. int th = pMargins->cyTopHeight;
  422. int bh = pMargins->cyBottomHeight;
  423. int iDestW = WIDTH(*prc);
  424. int iDestH = HEIGHT(*prc);
  425. //---- prevent left/right dest margins from overlapping ----
  426. FixMarginOverlaps(iDestW, &lw, &rw);
  427. //---- prevent top/bottom dest margins from overlapping ----
  428. FixMarginOverlaps(iDestH, &th, &bh);
  429. int lwFrom = lw;
  430. int rwFrom = pszSrcImage->cx - rw;
  431. int thFrom = th;
  432. int bhFrom = pszSrcImage->cy - bh;
  433. int lwTo = prc->left + lw;
  434. int rwTo = prc->right - rw;
  435. int thTo = prc->top + th;
  436. int bhTo = prc->bottom - bh;
  437. //---- compute offsets & factors ----
  438. int iLeftXOffset = prc->left;
  439. int iMiddleXOffset = lwTo;
  440. int iRightXOffset = rwTo;
  441. int iTopYOffset = prc->top;
  442. int iMiddleYOffset = thTo;
  443. int iBottomYOffset = bhTo;
  444. int iToMiddleWidth = rwTo - lwTo;
  445. int iFromMiddleWidth = rwFrom - lwFrom;
  446. int iToMiddleHeight = bhTo - thTo;
  447. int iFromMiddleHeight = bhFrom - thFrom;
  448. if (! iFromMiddleWidth) // avoid divide by zero
  449. {
  450. //--- map point to x=0 ----
  451. iToMiddleWidth = 0;
  452. iFromMiddleWidth = 1;
  453. }
  454. if (! iFromMiddleHeight) // avoid divide by zero
  455. {
  456. //--- map point to y=0 ----
  457. iToMiddleHeight = 0;
  458. iFromMiddleHeight = 1;
  459. }
  460. //---- clipping values for adjusted lw/rw/th/bh ----
  461. int lwMaxVal = __max(lw - 1, 0);
  462. int rwMinVal = __min(pMargins->cxRightWidth - rw, __max(pMargins->cxRightWidth-1, 0));
  463. int thMaxVal = __max(th - 1, 0);
  464. int bhMinVal = __min(pMargins->cyBottomHeight - bh, __max(pMargins->cyBottomHeight-1, 0));
  465. //---- allocte a buffer for the new points (rects) ----
  466. int newlen = sizeof(RGNDATAHEADER) + prd->rdh.nRgnSize; // same # of rects
  467. BYTE *newData = (BYTE *)new BYTE[newlen];
  468. RGNDATA *prdNew = (RGNDATA *)newData;
  469. if (! prdNew)
  470. return MakeError32(E_OUTOFMEMORY);
  471. ZeroMemory(prdNew, sizeof(prd->rdh));
  472. prdNew->rdh.dwSize = sizeof(prdNew->rdh);
  473. prdNew->rdh.iType = RDH_RECTANGLES;
  474. int cRects = prd->rdh.nCount;
  475. prdNew->rdh.nCount = cRects;
  476. SetRect(&prdNew->rdh.rcBound, -1, -1, -1, -1);
  477. //---- step thru our custom data (POINT + BYTE combos) ----
  478. POINT *pt = (POINT *)prd->Buffer;
  479. BYTE *pByte = (BYTE *)prd->Buffer + prd->rdh.nRgnSize;
  480. int cPoints = 2 * cRects;
  481. POINT *ptNew = (POINT *)prdNew->Buffer;
  482. for (int i=0; i < cPoints; i++, pt++, pByte++, ptNew++) // transform each "point"
  483. {
  484. switch (*pByte)
  485. {
  486. //---- in the "don't scale" areas, we clip the translated values ----
  487. //---- for the case where the destination areas are too small ----
  488. //---- using the below "__min()" and "__max()" calls ----
  489. //---- remember: each point has been made 0-relative to its grid ----
  490. case GN_LEFTTOP: // left top
  491. ptNew->x = __min(pt->x, lwMaxVal) + iLeftXOffset;
  492. ptNew->y = __min(pt->y, thMaxVal) + iTopYOffset;
  493. break;
  494. case GN_MIDDLETOP: // middle top
  495. ptNew->x = (pt->x*iToMiddleWidth)/iFromMiddleWidth + iMiddleXOffset;
  496. ptNew->y = __min(pt->y, thMaxVal) + iTopYOffset;
  497. break;
  498. case GN_RIGHTTOP: // right top
  499. ptNew->x = __max(pt->x, rwMinVal) + iRightXOffset;
  500. ptNew->y = __min(pt->y, thMaxVal) + iTopYOffset;
  501. break;
  502. case GN_LEFTMIDDLE: // left middle
  503. ptNew->x = __min(pt->x, lwMaxVal) + iLeftXOffset;
  504. ptNew->y = (pt->y*iToMiddleHeight)/iFromMiddleHeight + iMiddleYOffset;
  505. break;
  506. case GN_MIDDLEMIDDLE: // middle middle
  507. ptNew->x = (pt->x*iToMiddleWidth)/iFromMiddleWidth + iMiddleXOffset;
  508. ptNew->y = (pt->y*iToMiddleHeight)/iFromMiddleHeight + iMiddleYOffset;
  509. break;
  510. case GN_RIGHTMIDDLE: // right middle
  511. ptNew->x = __max(pt->x, rwMinVal) + iRightXOffset;
  512. ptNew->y = (pt->y*iToMiddleHeight)/iFromMiddleHeight + iMiddleYOffset;
  513. break;
  514. case GN_LEFTBOTTOM: // left bottom
  515. ptNew->x = __min(pt->x, lwMaxVal) + iLeftXOffset;
  516. ptNew->y = __max(pt->y, bhMinVal) + iBottomYOffset;
  517. break;
  518. case GN_MIDDLEBOTTOM: // middle bottom
  519. ptNew->x = (pt->x*iToMiddleWidth)/iFromMiddleWidth + iMiddleXOffset;
  520. ptNew->y = __max(pt->y, bhMinVal) + iBottomYOffset;
  521. break;
  522. case GN_RIGHTBOTTOM: // right bottom
  523. ptNew->x = __max(pt->x, rwMinVal) + iRightXOffset;
  524. ptNew->y = __max(pt->y, bhMinVal) + iBottomYOffset;
  525. break;
  526. }
  527. }
  528. //---- compute bounding box of new region ----
  529. RECT *pRect = (RECT *)prdNew->Buffer;
  530. RECT newBox = {-1, -1, -1, -1};
  531. for (i=0; i < cRects; i++, pRect++)
  532. _InPlaceUnionRect(&newBox, pRect);
  533. //---- create the new region ----
  534. prdNew->rdh.rcBound = newBox;
  535. HRGN hrgn = ExtCreateRegion(NULL, newlen, prdNew);
  536. delete [] newData; // free prdNew (aka newdata)
  537. if (! hrgn)
  538. return MakeErrorLast();
  539. *phrgn = hrgn;
  540. return S_OK;
  541. }
  542. //---------------------------------------------------------------------------------//
  543. #ifdef _DEBUG
  544. void RegionDebug(
  545. HRGN hrgn)
  546. {
  547. DWORD len = GetRegionData(hrgn, 0, NULL); // get required length
  548. ATLASSERT(len);
  549. RGNDATA *pRgnData = (RGNDATA *) new BYTE[len + sizeof(RGNDATAHEADER)];
  550. DWORD len2 = GetRegionData(hrgn, len, pRgnData);
  551. ATLASSERT(len == len2);
  552. }
  553. #endif