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.

5251 lines
138 KiB

  1. #include "ctlspriv.h"
  2. #include "image.h"
  3. #include "math.h"
  4. #ifndef AC_MIRRORBITMAP
  5. #define AC_MIRRORBITMAP 0 // BUGBUG: Remove me
  6. #endif
  7. void ImageList_DeleteDragBitmaps();
  8. BOOL ImageList_SetDragImage(HIMAGELIST piml, int i, int dxHotspot, int dyHotspot);
  9. HDC g_hdcSrc = NULL;
  10. HBITMAP g_hbmSrc = NULL;
  11. HBITMAP g_hbmDcDeselect = NULL;
  12. HDC g_hdcDst = NULL;
  13. HBITMAP g_hbmDst = NULL;
  14. int g_iILRefCount = 0;
  15. HRESULT WINAPI HIMAGELIST_QueryInterface(HIMAGELIST himl, REFIID riid, void** ppv)
  16. {
  17. *ppv = NULL;
  18. if (himl)
  19. {
  20. // First Convert the HIMAGELIST to an IUnknown.
  21. IUnknown* punk = reinterpret_cast<IUnknown*>(himl);
  22. // Now, we need to validate the object. CImageListBase contains the goo needed to figure out if this
  23. // is a valid imagelist.
  24. CImageListBase* pval = FindImageListBase(punk);
  25. // Now we call some private member.
  26. if (pval->IsValid())
  27. {
  28. // If it's valid then we can QI safely.
  29. return punk->QueryInterface(riid, ppv);
  30. }
  31. }
  32. return E_POINTER;
  33. }
  34. HRESULT WimpyDrawEx(IImageList* pux, int i, HDC hdcDst, int x, int y, int cx, int cy, COLORREF rgbBk, COLORREF rgbFg, UINT fStyle)
  35. {
  36. IMAGELISTDRAWPARAMS imldp = {0};
  37. imldp.cbSize = sizeof(imldp);
  38. imldp.himl = reinterpret_cast<HIMAGELIST>(pux);
  39. imldp.i = i;
  40. imldp.hdcDst = hdcDst;
  41. imldp.x = x;
  42. imldp.y = y;
  43. imldp.cx = cx;
  44. imldp.cy = cy;
  45. imldp.rgbBk = rgbBk;
  46. imldp.rgbFg = rgbFg;
  47. imldp.fStyle = fStyle;
  48. imldp.dwRop = SRCCOPY;
  49. return pux->Draw(&imldp);
  50. }
  51. HRESULT WimpyDraw(IImageList* pux, int i, HDC hdcDst, int x, int y, UINT fStyle)
  52. {
  53. IMAGELISTDRAWPARAMS imldp = {0};
  54. imldp.cbSize = sizeof(imldp);
  55. imldp.himl = reinterpret_cast<HIMAGELIST>(pux);
  56. imldp.i = i;
  57. imldp.hdcDst = hdcDst;
  58. imldp.x = x;
  59. imldp.y = y;
  60. imldp.rgbBk = CLR_DEFAULT;
  61. imldp.rgbFg = CLR_DEFAULT;
  62. imldp.fStyle = fStyle;
  63. imldp.dwRop = SRCCOPY;
  64. return pux->Draw(&imldp);
  65. }
  66. CImageList::CImageList() : _cRef(1)
  67. {
  68. }
  69. CImageList::~CImageList()
  70. {
  71. if (_pimlMirror)
  72. {
  73. _pimlMirror->Release();
  74. }
  75. _Destroy();
  76. }
  77. DWORD CImageList::_GetItemFlags(int i)
  78. {
  79. DWORD dw = 0;
  80. // NOTE: Currently we only add the flags in 32bit mode. If needed, you have
  81. // to modify ::Load in order to add items during a load. I'm just lazy
  82. if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  83. DSA_GetItem(_dsaFlags, i, &dw);
  84. return dw;
  85. }
  86. void CImageList::SetItemFlags(int i, DWORD dwFlags)
  87. {
  88. if (_dsaFlags)
  89. DSA_SetItem(_dsaFlags, i, &dwFlags);
  90. }
  91. HRESULT CImageList::Initialize(int cxI, int cyI, UINT flagsI, int cInitialI, int cGrowI)
  92. {
  93. HRESULT hr = E_OUTOFMEMORY;
  94. if (cGrowI < 4)
  95. {
  96. cGrowI = 4;
  97. }
  98. else
  99. {
  100. // round up by 4's
  101. cGrowI = (cGrowI + 3) & ~3;
  102. }
  103. _cStrip = 1;
  104. _cGrow = cGrowI;
  105. _cx = cxI;
  106. _cy = cyI;
  107. _clrBlend = CLR_NONE;
  108. _clrBk = CLR_NONE;
  109. _hbrBk = (HBRUSH)GetStockObject(BLACK_BRUSH);
  110. _fSolidBk = TRUE;
  111. _flags = flagsI;
  112. _pimlMirror = NULL;
  113. //
  114. // Initialize the overlay indexes to -1 since 0 is a valid index.
  115. //
  116. for (int i = 0; i < NUM_OVERLAY_IMAGES; i++)
  117. {
  118. _aOverlayIndexes[i] = -1;
  119. }
  120. _hdcImage = CreateCompatibleDC(NULL);
  121. if (_hdcImage)
  122. {
  123. hr = S_OK;
  124. if (_flags & ILC_MASK)
  125. {
  126. _hdcMask = CreateCompatibleDC(NULL);
  127. if (!_hdcMask)
  128. hr = E_OUTOFMEMORY;
  129. }
  130. if (SUCCEEDED(hr))
  131. {
  132. hr = _ReAllocBitmaps(cInitialI + 1);
  133. if (FAILED(hr))
  134. {
  135. hr = _ReAllocBitmaps(2);
  136. }
  137. }
  138. }
  139. // Don't do this if we are already initialized, we just want to pass new information....
  140. if (!_fInitialized)
  141. g_iILRefCount++;
  142. _fInitialized = TRUE;
  143. return hr;
  144. }
  145. HRESULT CImageList::QueryInterface(REFIID riid, void **ppv)
  146. {
  147. static const QITAB qit[] =
  148. {
  149. QITABENT(CImageList, IImageListPriv),
  150. QITABENT(CImageList, IImageList),
  151. QITABENT(CImageList, IImageListPersistStream),
  152. QITABENT(CImageList, IPersistStream),
  153. QITABENTMULTI(CImageList, IPersist, IPersistStream),
  154. { 0 },
  155. };
  156. return QISearch(this, (LPCQITAB)qit, riid, ppv);
  157. }
  158. ULONG CImageList::AddRef()
  159. {
  160. return InterlockedIncrement(&_cRef);
  161. }
  162. ULONG CImageList::Release()
  163. {
  164. if (InterlockedDecrement(&_cRef))
  165. return _cRef;
  166. delete this;
  167. return 0;
  168. }
  169. HRESULT CImageList::GetPrivateGoo(HBITMAP* phbmp, HDC* phdc, HBITMAP* phbmpMask, HDC* phdcMask)
  170. {
  171. if (phbmp)
  172. *phbmp = _hbmImage;
  173. if (phdc)
  174. *phdc = _hdcImage;
  175. if (phbmpMask)
  176. *phbmpMask = _hbmMask;
  177. if (phdcMask)
  178. *phdcMask = _hdcMask;
  179. return S_OK;
  180. }
  181. HRESULT CImageList::GetMirror(REFIID riid, void** ppv)
  182. {
  183. if (_pimlMirror)
  184. return _pimlMirror->QueryInterface(riid, ppv);
  185. return E_NOINTERFACE;
  186. }
  187. //
  188. // global work buffer, this buffer is always a DDB never a DIBSection
  189. //
  190. HBITMAP g_hbmWork = NULL; // work buffer.
  191. BITMAP g_bmWork = {0}; // work buffer size
  192. HBRUSH g_hbrMonoDither = NULL; // gray dither brush for dragging
  193. HBRUSH g_hbrStripe = NULL;
  194. #define NOTSRCAND 0x00220326L
  195. #define ROP_PSo 0x00FC008A
  196. #define ROP_DPo 0x00FA0089
  197. #define ROP_DPna 0x000A0329
  198. #define ROP_DPSona 0x00020c89
  199. #define ROP_SDPSanax 0x00E61ce8
  200. #define ROP_DSna 0x00220326
  201. #define ROP_PSDPxax 0x00b8074a
  202. #define ROP_PatNotMask 0x00b8074a // D <- S==0 ? P : D
  203. #define ROP_PatMask 0x00E20746 // D <- S==1 ? P : D
  204. #define ROP_MaskPat 0x00AC0744 // D <- P==1 ? D : S
  205. #define ROP_DSo 0x00EE0086L
  206. #define ROP_DSno 0x00BB0226L
  207. #define ROP_DSa 0x008800C6L
  208. static int g_iDither = 0;
  209. void InitDitherBrush()
  210. {
  211. HBITMAP hbmTemp;
  212. static const WORD graybits[] = {0xAAAA, 0x5555, 0xAAAA, 0x5555,
  213. 0xAAAA, 0x5555, 0xAAAA, 0x5555};
  214. if (g_iDither)
  215. {
  216. g_iDither++;
  217. }
  218. else
  219. {
  220. // build the dither brush. this is a fixed 8x8 bitmap
  221. hbmTemp = CreateBitmap(8, 8, 1, 1, graybits);
  222. if (hbmTemp)
  223. {
  224. // now use the bitmap for what it was really intended...
  225. g_hbrMonoDither = CreatePatternBrush(hbmTemp);
  226. DeleteObject(hbmTemp);
  227. g_iDither++;
  228. }
  229. }
  230. }
  231. void TerminateDitherBrush()
  232. {
  233. g_iDither--;
  234. if (g_iDither == 0)
  235. {
  236. DeleteObject(g_hbrMonoDither);
  237. g_hbrMonoDither = NULL;
  238. }
  239. }
  240. /*
  241. ** GetScreenDepth()
  242. */
  243. int GetScreenDepth()
  244. {
  245. int i;
  246. HDC hdc = GetDC(NULL);
  247. i = GetDeviceCaps(hdc, BITSPIXEL) * GetDeviceCaps(hdc, PLANES);
  248. ReleaseDC(NULL, hdc);
  249. return i;
  250. }
  251. //
  252. // should we use a DIB section on the current device?
  253. //
  254. // the main goal of using DS is to save memory, but they draw slow
  255. // on some devices.
  256. //
  257. // 4bpp Device (ie 16 color VGA) dont use DS
  258. // 8bpp Device (ie 256 color SVGA) use DS if DIBENG based.
  259. // >8bpp Device (ie 16bpp 24bpp) always use DS, saves memory
  260. //
  261. #define CAPS1 94 /* other caps */
  262. #define C1_DIBENGINE 0x0010 /* DIB Engine compliant driver */
  263. //
  264. // create a bitmap compatible with the given ImageList
  265. //
  266. HBITMAP CImageList::_CreateBitmap(int cx, int cy, RGBQUAD** ppargb)
  267. {
  268. HDC hdc;
  269. HBITMAP hbm;
  270. struct
  271. {
  272. BITMAPINFOHEADER bi;
  273. DWORD ct[256];
  274. } dib;
  275. hdc = GetDC(NULL);
  276. // no color depth was specifed
  277. //
  278. // if we are on a DIBENG based DISPLAY, we use 4bit DIBSections to save
  279. // memory.
  280. //
  281. if ((_flags & ILC_COLORMASK) == 0)
  282. {
  283. _flags |= ILC_COLOR4;
  284. }
  285. if ((_flags & ILC_COLORMASK) != ILC_COLORDDB)
  286. {
  287. dib.bi.biSize = sizeof(BITMAPINFOHEADER);
  288. dib.bi.biWidth = cx;
  289. dib.bi.biHeight = cy;
  290. dib.bi.biPlanes = 1;
  291. dib.bi.biBitCount = (_flags & ILC_COLORMASK);
  292. dib.bi.biCompression = BI_RGB;
  293. dib.bi.biSizeImage = 0;
  294. dib.bi.biXPelsPerMeter = 0;
  295. dib.bi.biYPelsPerMeter = 0;
  296. dib.bi.biClrUsed = 16;
  297. dib.bi.biClrImportant = 0;
  298. dib.ct[0] = 0x00000000; // 0000 black
  299. dib.ct[1] = 0x00800000; // 0001 dark red
  300. dib.ct[2] = 0x00008000; // 0010 dark green
  301. dib.ct[3] = 0x00808000; // 0011 mustard
  302. dib.ct[4] = 0x00000080; // 0100 dark blue
  303. dib.ct[5] = 0x00800080; // 0101 purple
  304. dib.ct[6] = 0x00008080; // 0110 dark turquoise
  305. dib.ct[7] = 0x00C0C0C0; // 1000 gray
  306. dib.ct[8] = 0x00808080; // 0111 dark gray
  307. dib.ct[9] = 0x00FF0000; // 1001 red
  308. dib.ct[10] = 0x0000FF00; // 1010 green
  309. dib.ct[11] = 0x00FFFF00; // 1011 yellow
  310. dib.ct[12] = 0x000000FF; // 1100 blue
  311. dib.ct[13] = 0x00FF00FF; // 1101 pink (magenta)
  312. dib.ct[14] = 0x0000FFFF; // 1110 cyan
  313. dib.ct[15] = 0x00FFFFFF; // 1111 white
  314. if (dib.bi.biBitCount == 8)
  315. {
  316. HPALETTE hpal;
  317. int i;
  318. if (hpal = CreateHalftonePalette(NULL))
  319. {
  320. i = GetPaletteEntries(hpal, 0, 256, (LPPALETTEENTRY)&dib.ct[0]);
  321. DeleteObject(hpal);
  322. if (i > 64)
  323. {
  324. dib.bi.biClrUsed = i;
  325. for (i=0; i<(int)dib.bi.biClrUsed; i++)
  326. dib.ct[i] = RGB(GetBValue(dib.ct[i]),GetGValue(dib.ct[i]),GetRValue(dib.ct[i]));
  327. }
  328. }
  329. else
  330. {
  331. dib.bi.biBitCount = (_flags & ILC_COLORMASK);
  332. dib.bi.biClrUsed = 256;
  333. }
  334. if (dib.bi.biClrUsed <= 16)
  335. dib.bi.biBitCount = 4;
  336. }
  337. hbm = CreateDIBSection(hdc, (LPBITMAPINFO)&dib, DIB_RGB_COLORS, (PVOID*)ppargb, NULL, 0);
  338. }
  339. else
  340. {
  341. hbm = CreateCompatibleBitmap(hdc, cx, cy);
  342. }
  343. ReleaseDC(NULL, hdc);
  344. return hbm;
  345. }
  346. EXTERN_C HBITMAP CreateColorBitmap(int cx, int cy)
  347. {
  348. HBITMAP hbm;
  349. HDC hdc;
  350. hdc = GetDC(NULL);
  351. //
  352. // on a multimonitor system with mixed bitdepths
  353. // always use a 32bit bitmap for our work buffer
  354. // this will prevent us from losing colors when
  355. // blting to and from the screen. this is mainly
  356. // important for the drag & drop offscreen buffers.
  357. //
  358. if (!(GetDeviceCaps(hdc, RASTERCAPS) & RC_PALETTE) &&
  359. GetSystemMetrics(SM_CMONITORS) > 1 &&
  360. GetSystemMetrics(SM_SAMEDISPLAYFORMAT) == 0)
  361. {
  362. void* p;
  363. BITMAPINFO bi = {sizeof(BITMAPINFOHEADER), cx, cy, 1, 32};
  364. hbm = CreateDIBSection(hdc, &bi, DIB_RGB_COLORS, &p, NULL, 0);
  365. }
  366. else
  367. {
  368. hbm = CreateCompatibleBitmap(hdc, cx, cy);
  369. }
  370. ReleaseDC(NULL, hdc);
  371. return hbm;
  372. }
  373. HBITMAP CreateDIB(HDC h, int cx, int cy, RGBQUAD** pprgb)
  374. {
  375. BITMAPINFO bi = {0};
  376. bi.bmiHeader.biSize = sizeof(bi.bmiHeader);
  377. bi.bmiHeader.biWidth = cx;
  378. bi.bmiHeader.biHeight = cy;
  379. bi.bmiHeader.biPlanes = 1;
  380. bi.bmiHeader.biBitCount = 32;
  381. bi.bmiHeader.biCompression = BI_RGB;
  382. return CreateDIBSection(h, &bi, DIB_RGB_COLORS, (void**)pprgb, NULL, 0);
  383. }
  384. BOOL DIBHasAlpha(int cx, int cy, RGBQUAD* prgb)
  385. {
  386. int cTotal = cx * cy;
  387. for (int i = 0; i < cTotal; i++)
  388. {
  389. if (prgb[i].rgbReserved != 0)
  390. return TRUE;
  391. }
  392. return FALSE;
  393. }
  394. void PreProcessDIB(int cx, int cy, RGBQUAD* pargb)
  395. {
  396. int cTotal = cx * cy;
  397. for (int i = 0; i < cTotal; i++)
  398. {
  399. RGBQUAD* prgb = &pargb[i];
  400. if (prgb->rgbReserved != 0)
  401. {
  402. prgb->rgbRed = ((prgb->rgbRed * prgb->rgbReserved) + 128) / 255;
  403. prgb->rgbGreen = ((prgb->rgbGreen * prgb->rgbReserved) + 128) / 255;
  404. prgb->rgbBlue = ((prgb->rgbBlue * prgb->rgbReserved) + 128) / 255;
  405. }
  406. else
  407. {
  408. *((DWORD*)prgb) = 0;
  409. }
  410. }
  411. }
  412. EXTERN_C HBITMAP CreateMonoBitmap(int cx, int cy)
  413. {
  414. #ifdef MONO_DIB
  415. struct
  416. {
  417. BITMAPINFOHEADER bi;
  418. DWORD ct[2];
  419. } dib = {0};
  420. dib.bi.biSize = sizeof(dib.bi);
  421. dib.bi.biWidth = cx;
  422. dib.bi.biHeight = cy;
  423. dib.bi.biPlanes = 1;
  424. dib.bi.biBitCount = 1;
  425. dib.bi.biCompression = BI_RGB;
  426. dib.ct[0] = 0x00000000;
  427. dib.ct[1] = 0x00ffffff;
  428. HBITMAP hbmp = NULL;
  429. HDC hdc = CreateCompatibleDC(NULL);
  430. if (hdc)
  431. {
  432. hbmp = CreateDIBSection(hdc, (BITMAPINFO*)&dib, DIB_PAL_COLORS, NULL, NULL, 0);
  433. DeleteDC(hdc);
  434. }
  435. return hbmp;
  436. #else
  437. return CreateBitmap(cx, cy, 1, 1, NULL);
  438. #endif
  439. }
  440. //============================================================================
  441. BOOL CImageList::GlobalInit(void)
  442. {
  443. HDC hdcScreen;
  444. static const WORD stripebits[] = {0x7777, 0xdddd, 0x7777, 0xdddd,
  445. 0x7777, 0xdddd, 0x7777, 0xdddd};
  446. HBITMAP hbmTemp;
  447. TraceMsg(TF_IMAGELIST, "CImageList::GlobalInit");
  448. // if already initialized, there is nothing to do
  449. if (g_hdcDst)
  450. return TRUE;
  451. hdcScreen = GetDC(HWND_DESKTOP);
  452. g_hdcSrc = CreateCompatibleDC(hdcScreen);
  453. g_hdcDst = CreateCompatibleDC(hdcScreen);
  454. InitDitherBrush();
  455. hbmTemp = CreateBitmap(8, 8, 1, 1, stripebits);
  456. if (hbmTemp)
  457. {
  458. // initialize the deselect 1x1 bitmap
  459. g_hbmDcDeselect = SelectBitmap(g_hdcDst, hbmTemp);
  460. SelectBitmap(g_hdcDst, g_hbmDcDeselect);
  461. g_hbrStripe = CreatePatternBrush(hbmTemp);
  462. DeleteObject(hbmTemp);
  463. }
  464. ReleaseDC(HWND_DESKTOP, hdcScreen);
  465. if (!g_hdcSrc || !g_hdcDst || !g_hbrMonoDither)
  466. {
  467. CImageList::GlobalUninit();
  468. TraceMsg(TF_ERROR, "ImageList: Unable to initialize");
  469. return FALSE;
  470. }
  471. return TRUE;
  472. }
  473. void CImageList::GlobalUninit()
  474. {
  475. TerminateDitherBrush();
  476. if (g_hbrStripe)
  477. {
  478. DeleteObject(g_hbrStripe);
  479. g_hbrStripe = NULL;
  480. }
  481. ImageList_DeleteDragBitmaps();
  482. if (g_hdcDst)
  483. {
  484. CImageList::SelectDstBitmap(NULL);
  485. DeleteDC(g_hdcDst);
  486. g_hdcDst = NULL;
  487. }
  488. if (g_hdcSrc)
  489. {
  490. CImageList::SelectSrcBitmap(NULL);
  491. DeleteDC(g_hdcSrc);
  492. g_hdcSrc = NULL;
  493. }
  494. if (g_hbmWork)
  495. {
  496. DeleteBitmap(g_hbmWork);
  497. g_hbmWork = NULL;
  498. }
  499. }
  500. void CImageList::SelectDstBitmap(HBITMAP hbmDst)
  501. {
  502. ASSERTCRITICAL;
  503. if (hbmDst != g_hbmDst)
  504. {
  505. // If it's selected in the source DC, then deselect it first
  506. //
  507. if (hbmDst && hbmDst == g_hbmSrc)
  508. CImageList::SelectSrcBitmap(NULL);
  509. SelectBitmap(g_hdcDst, hbmDst ? hbmDst : g_hbmDcDeselect);
  510. g_hbmDst = hbmDst;
  511. }
  512. }
  513. void CImageList::SelectSrcBitmap(HBITMAP hbmSrc)
  514. {
  515. ASSERTCRITICAL;
  516. if (hbmSrc != g_hbmSrc)
  517. {
  518. // If it's selected in the dest DC, then deselect it first
  519. //
  520. if (hbmSrc && hbmSrc == g_hbmDst)
  521. CImageList::SelectDstBitmap(NULL);
  522. SelectBitmap(g_hdcSrc, hbmSrc ? hbmSrc : g_hbmDcDeselect);
  523. g_hbmSrc = hbmSrc;
  524. }
  525. }
  526. HDC ImageList_GetWorkDC(HDC hdc, BOOL f32bpp, int dx, int dy)
  527. {
  528. ASSERTCRITICAL;
  529. int iDepth = GetDeviceCaps(hdc, BITSPIXEL);
  530. if (g_hbmWork == NULL ||
  531. iDepth != g_bmWork.bmBitsPixel ||
  532. g_bmWork.bmWidth != dx ||
  533. g_bmWork.bmHeight != dy ||
  534. (f32bpp && iDepth != 32))
  535. {
  536. CImageList::_DeleteBitmap(g_hbmWork);
  537. g_hbmWork = NULL;
  538. if (dx == 0 || dy == 0)
  539. return NULL;
  540. if (f32bpp)
  541. g_hbmWork = CreateDIB(hdc, dx, dy, NULL);
  542. else
  543. g_hbmWork = CreateCompatibleBitmap(hdc, dx, dy);
  544. if (g_hbmWork)
  545. {
  546. GetObject(g_hbmWork, sizeof(g_bmWork), &g_bmWork);
  547. }
  548. }
  549. CImageList::SelectSrcBitmap(g_hbmWork);
  550. if (GetDeviceCaps(hdc, RASTERCAPS) & RC_PALETTE)
  551. {
  552. HPALETTE hpal = (HPALETTE)SelectPalette(hdc, (HPALETTE)GetStockObject(DEFAULT_PALETTE), TRUE);
  553. SelectPalette(g_hdcSrc, hpal, TRUE);
  554. }
  555. return g_hdcSrc;
  556. }
  557. void ImageList_ReleaseWorkDC(HDC hdc)
  558. {
  559. ASSERTCRITICAL;
  560. ASSERT(hdc == g_hdcSrc);
  561. if (GetDeviceCaps(hdc, RASTERCAPS) & RC_PALETTE)
  562. {
  563. SelectPalette(hdc, (HPALETTE)GetStockObject(DEFAULT_PALETTE), TRUE);
  564. }
  565. }
  566. void CImageList::_DeleteBitmap(HBITMAP hbm)
  567. {
  568. ASSERTCRITICAL;
  569. if (hbm)
  570. {
  571. if (g_hbmDst == hbm)
  572. CImageList::SelectDstBitmap(NULL);
  573. if (g_hbmSrc == hbm)
  574. CImageList::SelectSrcBitmap(NULL);
  575. DeleteBitmap(hbm);
  576. }
  577. }
  578. #define ILC_WIN95 (ILC_MASK | ILC_COLORMASK | ILC_SHARED | ILC_PALETTE)
  579. //============================================================================
  580. HRESULT CImageList::InitGlobals()
  581. {
  582. HRESULT hr = S_OK;
  583. ENTERCRITICAL;
  584. if (!g_iILRefCount)
  585. {
  586. if (!CImageList::GlobalInit())
  587. {
  588. hr = E_OUTOFMEMORY;
  589. }
  590. }
  591. LEAVECRITICAL;
  592. return S_OK;
  593. }
  594. CImageList* CImageList::Create(int cx, int cy, UINT flags, int cInitial, int cGrow)
  595. {
  596. CImageList* piml = NULL;
  597. HRESULT hr = S_OK;
  598. if (cx < 0 || cy < 0)
  599. return NULL;
  600. // Validate the flags
  601. if (flags & ~ILC_VALID)
  602. return NULL;
  603. hr = InitGlobals();
  604. ENTERCRITICAL;
  605. if (SUCCEEDED(hr))
  606. {
  607. piml = new CImageList();
  608. // allocate the bitmap PLUS one re-usable entry
  609. if (piml)
  610. {
  611. hr = piml->Initialize(cx, cy, flags, cInitial, cGrow);
  612. if (FAILED(hr))
  613. {
  614. piml->Release();
  615. piml = NULL;
  616. }
  617. }
  618. }
  619. LEAVECRITICAL;
  620. return piml;
  621. }
  622. void CImageList::_Destroy()
  623. {
  624. ENTERCRITICAL;
  625. // nuke dc's
  626. if (_hdcImage)
  627. {
  628. SelectObject(_hdcImage, g_hbmDcDeselect);
  629. DeleteDC(_hdcImage);
  630. }
  631. if (_hdcMask)
  632. {
  633. SelectObject(_hdcMask, g_hbmDcDeselect);
  634. DeleteDC(_hdcMask);
  635. }
  636. // nuke bitmaps
  637. if (_hbmImage)
  638. _DeleteBitmap(_hbmImage);
  639. if (_hbmMask)
  640. _DeleteBitmap(_hbmMask);
  641. if (_hbrBk)
  642. DeleteObject(_hbrBk);
  643. //Clean up DSA
  644. if (_dsaFlags)
  645. DSA_Destroy(_dsaFlags);
  646. if (_fInitialized)
  647. {
  648. // one less use of imagelists. if it's the last, terminate the imagelist
  649. g_iILRefCount--;
  650. if (!g_iILRefCount)
  651. CImageList::GlobalUninit();
  652. }
  653. LEAVECRITICAL;
  654. }
  655. HRESULT CImageList::GetImageCount(int* pi)
  656. {
  657. *pi = _cImage;
  658. return S_OK;
  659. }
  660. HRESULT CImageList::SetImageCount(UINT uAlloc)
  661. {
  662. ENTERCRITICAL;
  663. HRESULT hr = _ReAllocBitmaps(-((int)uAlloc + 2)); // Two because we need a spare image
  664. if (SUCCEEDED(hr))
  665. {
  666. _cImage = (int)uAlloc;
  667. }
  668. LEAVECRITICAL;
  669. return hr;
  670. }
  671. HRESULT CImageList::GetIconSize(int* pcx, int* pcy)
  672. {
  673. if (!pcx || !pcy)
  674. return E_INVALIDARG;
  675. *pcx = _cx;
  676. *pcy = _cy;
  677. return S_OK;
  678. }
  679. //
  680. // change the size of a existing image list
  681. // also removes all items
  682. //
  683. HRESULT CImageList::_SetIconSize(int cxImage, int cyImage)
  684. {
  685. if (_cx == cxImage && _cy == cyImage)
  686. return S_FALSE; // no change
  687. if (_cx < 0 || _cy < 0)
  688. return E_INVALIDARG; // invalid dimensions
  689. _cx = cxImage;
  690. _cy = cyImage;
  691. return Remove(-1);
  692. }
  693. HRESULT CImageList::SetIconSize(int cxImage, int cyImage)
  694. {
  695. if (_pimlMirror)
  696. {
  697. _pimlMirror->_SetIconSize(cxImage, cyImage);
  698. }
  699. return _SetIconSize(cxImage, cyImage);
  700. }
  701. //
  702. // ImageList_SetFlags
  703. //
  704. // change the image list flags, then rebuilds the bitmaps.
  705. //
  706. // the only reason to call this function is to change the
  707. // color depth of the image list, the shell needs to do this
  708. // when the screen depth changes and it wants to use HiColor icons.
  709. //
  710. HRESULT CImageList::SetFlags(UINT uFlags)
  711. {
  712. HBITMAP hOldImage;
  713. // check for valid input flags
  714. if (uFlags & ~ILC_VALID)
  715. return E_INVALIDARG;
  716. // you cant change these flags.
  717. if ((uFlags ^ _flags) & ILC_SHARED)
  718. return E_INVALIDARG;
  719. if (_pimlMirror)
  720. _pimlMirror->SetFlags(uFlags);
  721. // now change the flags and rebuild the bitmaps.
  722. _flags = uFlags;
  723. // set the old bitmap to NULL, so when Imagelist_remove calls
  724. // ImageList_createBitmap, it will not call CreatecomptibleBitmap,
  725. // it will create the spec for the bitmap from scratch..
  726. hOldImage = _hbmImage;
  727. _hbmImage = NULL;
  728. Remove(-1);
  729. // imagelist::remove will have ensured that the old image is no longer selected
  730. // thus we can now delete it...
  731. if ( hOldImage )
  732. DeleteObject( hOldImage );
  733. return S_OK;
  734. }
  735. HRESULT CImageList::GetFlags(UINT* puFlags)
  736. {
  737. *puFlags = (_flags & ILC_VALID) | (_pimlMirror ? ILC_MIRROR : 0);
  738. return S_OK;
  739. }
  740. // reset the background color of images iFirst through iLast
  741. void CImageList::_ResetBkColor(int iFirst, int iLast, COLORREF clr)
  742. {
  743. HBRUSH hbrT=NULL;
  744. DWORD rop;
  745. if (_hdcMask == NULL)
  746. return;
  747. if (clr == CLR_BLACK || clr == CLR_NONE)
  748. {
  749. rop = ROP_DSna;
  750. }
  751. else if (clr == CLR_WHITE)
  752. {
  753. rop = ROP_DSo;
  754. }
  755. else
  756. {
  757. ASSERT(_hbrBk);
  758. ASSERT(_clrBk == clr);
  759. rop = ROP_PatMask;
  760. hbrT = SelectBrush(_hdcImage, _hbrBk);
  761. }
  762. for ( ;iFirst <= iLast; iFirst++)
  763. {
  764. RECT rc;
  765. GetImageRect(iFirst, &rc);
  766. if (_GetItemFlags(iFirst) == 0)
  767. {
  768. BitBlt(_hdcImage, rc.left, rc.top, _cx, _cy,
  769. _hdcMask, rc.left, rc.top, rop);
  770. }
  771. }
  772. if (hbrT)
  773. SelectBrush(_hdcImage, hbrT);
  774. }
  775. //
  776. // GetNearestColor is problematic. If you have a 32-bit HDC with a 16-bit bitmap
  777. // selected into it, and you call GetNearestColor, GDI ignores the
  778. // color-depth of the bitmap and thinks you have a 32-bit bitmap inside,
  779. // so of course it returns the same color unchanged.
  780. //
  781. // So instead, we have to emulate GetNearestColor with SetPixel.
  782. //
  783. COLORREF GetNearestColor32(HDC hdc, COLORREF rgb)
  784. {
  785. COLORREF rgbT;
  786. rgbT = GetPixel(hdc, 0, 0);
  787. rgb = SetPixel(hdc, 0, 0, rgb);
  788. SetPixelV(hdc, 0, 0, rgbT);
  789. return rgb;
  790. }
  791. COLORREF CImageList::_SetBkColor(COLORREF clrBkI)
  792. {
  793. COLORREF clrBkOld;
  794. // Quick out if there is no change in color
  795. if (_clrBk == clrBkI)
  796. {
  797. return _clrBk;
  798. }
  799. // The following code deletes the brush, resets the background color etc.,
  800. // so, protect it with a critical section.
  801. ENTERCRITICAL;
  802. if (_hbrBk)
  803. {
  804. DeleteBrush(_hbrBk);
  805. }
  806. clrBkOld = _clrBk;
  807. _clrBk = clrBkI;
  808. if (_clrBk == CLR_NONE)
  809. {
  810. _hbrBk = (HBRUSH)GetStockObject(BLACK_BRUSH);
  811. _fSolidBk = TRUE;
  812. }
  813. else
  814. {
  815. _hbrBk = CreateSolidBrush(_clrBk);
  816. _fSolidBk = GetNearestColor32(_hdcImage, _clrBk) == _clrBk;
  817. }
  818. if (_cImage > 0)
  819. {
  820. _ResetBkColor(0, _cImage - 1, _clrBk);
  821. }
  822. LEAVECRITICAL;
  823. return clrBkOld;
  824. }
  825. HRESULT CImageList::SetBkColor(COLORREF clrBk, COLORREF* pclr)
  826. {
  827. if (_pimlMirror)
  828. {
  829. _pimlMirror->_SetBkColor(clrBk);
  830. }
  831. *pclr = _SetBkColor(clrBk);
  832. return S_OK;
  833. }
  834. HRESULT CImageList::GetBkColor(COLORREF* pclr)
  835. {
  836. *pclr = _clrBk;
  837. return S_OK;
  838. }
  839. HRESULT CImageList::_ReAllocBitmaps(int cAllocI)
  840. {
  841. HBITMAP hbmImageNew = NULL;
  842. HBITMAP hbmMaskNew = NULL;
  843. RGBQUAD* pargbImageNew = NULL;
  844. int cxL, cyL;
  845. // HACK: don't shrink unless the caller passes a negative count
  846. if (cAllocI > 0)
  847. {
  848. if (_cAlloc >= cAllocI)
  849. return S_OK;
  850. }
  851. else
  852. cAllocI *= -1;
  853. cxL = _cx * _cStrip;
  854. cyL = _cy * ((cAllocI + _cStrip - 1) / _cStrip);
  855. if (cAllocI > 0)
  856. {
  857. if (_flags & ILC_MASK)
  858. {
  859. hbmMaskNew = CreateMonoBitmap(cxL, cyL);
  860. if (!hbmMaskNew)
  861. {
  862. TraceMsg(TF_ERROR, "ImageList: Can't create bitmap");
  863. return E_OUTOFMEMORY;
  864. }
  865. }
  866. hbmImageNew = _CreateBitmap(cxL, cyL, &pargbImageNew);
  867. if (!hbmImageNew)
  868. {
  869. if (hbmMaskNew)
  870. CImageList::_DeleteBitmap(hbmMaskNew);
  871. TraceMsg(TF_ERROR, "ImageList: Can't create bitmap");
  872. return E_OUTOFMEMORY;
  873. }
  874. if (_dsaFlags == NULL)
  875. _dsaFlags = DSA_Create(sizeof(DWORD), _cGrow);
  876. if (!_dsaFlags)
  877. {
  878. if (hbmMaskNew)
  879. CImageList::_DeleteBitmap(hbmMaskNew);
  880. if (hbmImageNew)
  881. CImageList::_DeleteBitmap(hbmImageNew);
  882. TraceMsg(TF_ERROR, "ImageList: Can't create flags array");
  883. return E_OUTOFMEMORY;
  884. }
  885. }
  886. if (_cImage > 0)
  887. {
  888. int cyCopy = _cy * ((min(cAllocI, _cImage) + _cStrip - 1) / _cStrip);
  889. if (_flags & ILC_MASK)
  890. {
  891. CImageList::SelectDstBitmap(hbmMaskNew);
  892. BitBlt(g_hdcDst, 0, 0, cxL, cyCopy, _hdcMask, 0, 0, SRCCOPY);
  893. }
  894. CImageList::SelectDstBitmap(hbmImageNew);
  895. BitBlt(g_hdcDst, 0, 0, cxL, cyCopy, _hdcImage, 0, 0, SRCCOPY);
  896. }
  897. // select into DC's, delete then assign
  898. CImageList::SelectDstBitmap(NULL);
  899. CImageList::SelectSrcBitmap(NULL);
  900. SelectObject(_hdcImage, hbmImageNew);
  901. if (_hdcMask)
  902. SelectObject(_hdcMask, hbmMaskNew);
  903. if (_hbmMask)
  904. CImageList::_DeleteBitmap(_hbmMask);
  905. if (_hbmImage)
  906. CImageList::_DeleteBitmap(_hbmImage);
  907. _hbmMask = hbmMaskNew;
  908. _hbmImage = hbmImageNew;
  909. _pargbImage = pargbImageNew;
  910. _clrBlend = CLR_NONE;
  911. _cAlloc = cAllocI;
  912. return S_OK;
  913. }
  914. HBITMAP CImageList::_CreateMirroredBitmap(HBITMAP hbmOrig, BOOL fMirrorEach, int cx)
  915. {
  916. HBITMAP hbm = NULL, hOld_bm1, hOld_bm2;
  917. BITMAP bm;
  918. if (!hbmOrig)
  919. return NULL;
  920. if (!GetObject(hbmOrig, sizeof(BITMAP), &bm))
  921. return NULL;
  922. // Grab the screen DC
  923. HDC hdc = GetDC(NULL);
  924. HDC hdcMem1 = CreateCompatibleDC(hdc);
  925. if (!hdcMem1)
  926. {
  927. ReleaseDC(NULL, hdc);
  928. return NULL;
  929. }
  930. HDC hdcMem2 = CreateCompatibleDC(hdc);
  931. if (!hdcMem2)
  932. {
  933. DeleteDC(hdcMem1);
  934. ReleaseDC(NULL, hdc);
  935. return NULL;
  936. }
  937. if (bm.bmBitsPixel == 32)
  938. {
  939. void* p;
  940. BITMAPINFO bi = {sizeof(BITMAPINFOHEADER), bm.bmWidth, bm.bmHeight, 1, 32};
  941. hbm = CreateDIBSection(hdc, &bi, DIB_RGB_COLORS, &p, NULL, 0);
  942. }
  943. else
  944. {
  945. hbm = CreateColorBitmap(bm.bmWidth, bm.bmHeight);
  946. }
  947. if (!hbm)
  948. {
  949. DeleteDC(hdcMem2);
  950. DeleteDC(hdcMem1);
  951. ReleaseDC(NULL, hdc);
  952. return NULL;
  953. }
  954. //
  955. // Flip the bitmap
  956. //
  957. hOld_bm1 = (HBITMAP)SelectObject(hdcMem1, hbmOrig);
  958. hOld_bm2 = (HBITMAP)SelectObject(hdcMem2 , hbm );
  959. SET_DC_RTL_MIRRORED(hdcMem2);
  960. if (fMirrorEach)
  961. {
  962. for (int i = 0; i < bm.bmWidth; i += cx) // Flip the bits in the imagelist...
  963. {
  964. BitBlt(hdcMem2, bm.bmWidth - i - cx, 0, cx, bm.bmHeight, hdcMem1, i, 0, SRCCOPY);
  965. }
  966. }
  967. else
  968. {
  969. BitBlt(hdcMem2, 0, 0, bm.bmWidth, bm.bmHeight, hdcMem1, 0, 0, SRCCOPY);
  970. }
  971. SelectObject(hdcMem1, hOld_bm1 );
  972. SelectObject(hdcMem1, hOld_bm2 );
  973. DeleteDC(hdcMem2);
  974. DeleteDC(hdcMem1);
  975. ReleaseDC(NULL, hdc);
  976. return hbm;
  977. }
  978. HRESULT CImageList::SetColorTable(int start, int len, RGBQUAD *prgb, int* pi)
  979. {
  980. // mark it that we have set the color table so that it won't be overwritten
  981. // by the first bitmap add....
  982. _fColorsSet = TRUE;
  983. if (_hdcImage)
  984. {
  985. *pi = SetDIBColorTable(_hdcImage, start, len, prgb);
  986. return S_OK;
  987. }
  988. return E_FAIL;
  989. }
  990. BOOL CImageList::_HasAlpha(int i)
  991. {
  992. if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  993. {
  994. RECT rc;
  995. if (SUCCEEDED(GetImageRectInverted(i, &rc)))
  996. {
  997. for (int y = rc.top; y < rc.bottom; y++)
  998. {
  999. for (int x = rc.left; x < rc.right; x++)
  1000. {
  1001. if (_pargbImage[x + y * _cx].rgbReserved != 0)
  1002. return TRUE;
  1003. }
  1004. }
  1005. }
  1006. }
  1007. return FALSE;
  1008. }
  1009. void CImageList::_ScanForAlpha()
  1010. {
  1011. if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  1012. {
  1013. for (int i = 0; i < _cImage; i++)
  1014. {
  1015. SetItemFlags(i, _HasAlpha(i)? ILIF_ALPHA : 0);
  1016. }
  1017. }
  1018. }
  1019. BOOL CImageList::_PreProcessImage(int i)
  1020. {
  1021. if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  1022. {
  1023. RECT rc;
  1024. GetImageRectInverted(i, &rc);
  1025. #ifdef _X86_
  1026. if (IsProcessorFeaturePresent(PF_MMX_INSTRUCTIONS_AVAILABLE))
  1027. {
  1028. _asm
  1029. {
  1030. pxor mm0, mm0
  1031. pxor mm1, mm1
  1032. pxor mm5, mm5
  1033. movq mm6, qw128 // mm6 is filled with 128
  1034. movq mm7, qw1 // mm7 is filled with 1
  1035. }
  1036. for (int y = rc.top; y < rc.bottom; y++)
  1037. {
  1038. int Offset = y * _cx;
  1039. RGBQUAD* prgb = &_pargbImage[rc.left + Offset];
  1040. for (int x = rc.left; x < rc.right; x++)
  1041. {
  1042. _asm
  1043. {
  1044. push ecx
  1045. mov edx, dword ptr [prgb] // Read alpha channel
  1046. mov ecx, dword ptr [edx]
  1047. mov ebx, ecx
  1048. shr ebx, 24 // a >> 24
  1049. mov eax, ebx // a -> b
  1050. or eax, eax
  1051. jz EarlyOut
  1052. shl ebx, 8 // b << 8
  1053. or eax, ebx // a |= b
  1054. shl ebx, 8 // b << 8
  1055. or eax, ebx // a |= b
  1056. shl ebx, 8 // b << 8
  1057. // Note high byte of alpha is zero.
  1058. movd mm0, eax // a -> mm0
  1059. movd mm1, ecx // Load the pixel
  1060. punpcklbw mm0,mm5 // mm0 -> Expands <- mm0 Contains the Alpha channel for this multiply
  1061. punpcklbw mm1,mm5 // Unpack the pixel
  1062. pmullw mm1, mm0 // Multiply by the alpha channel <- mm1 contains c * alpha
  1063. paddusw mm1, mm6 // perform the (c * alpha) + 128
  1064. psrlw mm1, 8 // Divide by 255
  1065. paddusw mm1, mm7 // Add 1 to finish the divide by 255
  1066. packuswb mm1, mm5
  1067. movd eax, mm1
  1068. or eax, ebx // Transfer alpha channel
  1069. EarlyOut:
  1070. mov dword ptr [edx], eax
  1071. pop ecx
  1072. }
  1073. prgb++;
  1074. }
  1075. }
  1076. _asm emms
  1077. }
  1078. else
  1079. #endif
  1080. {
  1081. for (int y = rc.top; y < rc.bottom; y++)
  1082. {
  1083. int Offset = y * _cx;
  1084. for (int x = rc.left; x < rc.right; x++)
  1085. {
  1086. RGBQUAD* prgb = &_pargbImage[x + Offset];
  1087. if (prgb->rgbReserved)
  1088. {
  1089. prgb->rgbRed = ((prgb->rgbRed * prgb->rgbReserved) + 128) / 255;
  1090. prgb->rgbGreen = ((prgb->rgbGreen * prgb->rgbReserved) + 128) / 255;
  1091. prgb->rgbBlue = ((prgb->rgbBlue * prgb->rgbReserved) + 128) / 255;
  1092. }
  1093. else
  1094. {
  1095. *((DWORD*)prgb) = 0;
  1096. }
  1097. }
  1098. }
  1099. }
  1100. return TRUE;
  1101. }
  1102. return FALSE;
  1103. }
  1104. HRESULT CImageList::_Add(HBITMAP hbmImageI, HBITMAP hbmMaskI, int cImageI, int xStart, int yStart, int* pi)
  1105. {
  1106. int i = -1;
  1107. HRESULT hr = S_OK;
  1108. ENTERCRITICAL;
  1109. //
  1110. // if the ImageList is empty clone the color table of the first
  1111. // bitmap you add to the imagelist.
  1112. //
  1113. // the ImageList needs to be a 8bpp image list
  1114. // the bitmap being added needs to be a 8bpp DIBSection
  1115. //
  1116. if (hbmImageI && _cImage == 0 &&
  1117. (_flags & ILC_COLORMASK) != ILC_COLORDDB)
  1118. {
  1119. if (!_fColorsSet)
  1120. {
  1121. int n;
  1122. RGBQUAD argb[256];
  1123. CImageList::SelectDstBitmap(hbmImageI);
  1124. if (n = GetDIBColorTable(g_hdcDst, 0, 256, argb))
  1125. {
  1126. int i;
  1127. SetColorTable(0, n, argb, &i);
  1128. }
  1129. CImageList::SelectDstBitmap(NULL);
  1130. }
  1131. _clrBlend = CLR_NONE;
  1132. }
  1133. if (_cImage + cImageI + 1 > _cAlloc)
  1134. {
  1135. hr = _ReAllocBitmaps(_cAlloc + max(cImageI, _cGrow) + 1);
  1136. }
  1137. if (SUCCEEDED(hr))
  1138. {
  1139. i = _cImage;
  1140. _cImage += cImageI;
  1141. if (hbmImageI)
  1142. {
  1143. hr = _Replace(i, cImageI, hbmImageI, hbmMaskI, xStart, yStart);
  1144. if (FAILED(hr))
  1145. {
  1146. _cImage -= cImageI;
  1147. i = -1;
  1148. }
  1149. }
  1150. }
  1151. LEAVECRITICAL;
  1152. *pi = i;
  1153. return hr;
  1154. }
  1155. HRESULT CImageList::_AddValidated(HBITMAP hbmImage, HBITMAP hbmMask, int* pi)
  1156. {
  1157. BITMAP bm;
  1158. int cImageI;
  1159. if (GetObject(hbmImage, sizeof(bm), &bm) != sizeof(bm) || bm.bmWidth < _cx)
  1160. {
  1161. return E_INVALIDARG;
  1162. }
  1163. ASSERT(hbmImage);
  1164. ASSERT(_cx);
  1165. cImageI = bm.bmWidth / _cx; // # of images in source
  1166. // serialization handled within Add2.
  1167. return _Add(hbmImage, hbmMask, cImageI, 0, 0, pi);
  1168. }
  1169. HRESULT CImageList::Add(HBITMAP hbmImage, HBITMAP hbmMask, int* pi)
  1170. {
  1171. if (_pimlMirror)
  1172. {
  1173. HBITMAP hbmMirroredImage = _CreateMirroredBitmap(hbmImage, (ILC_PERITEMMIRROR & _flags), _cx);
  1174. HBITMAP hbmMirroredMask = _CreateMirroredBitmap(hbmMask, (ILC_PERITEMMIRROR & _flags), _cx);
  1175. _pimlMirror->_AddValidated(hbmMirroredImage, hbmMirroredMask, pi);
  1176. // The caller will take care of deleting hbmImage, hbmMask
  1177. // He knows nothing about hbmMirroredImage, hbmMirroredMask
  1178. DeleteObject(hbmMirroredImage);
  1179. DeleteObject(hbmMirroredMask);
  1180. }
  1181. return _AddValidated(hbmImage, hbmMask, pi);
  1182. }
  1183. HRESULT CImageList::_AddMasked(HBITMAP hbmImageI, COLORREF crMask, int* pi)
  1184. {
  1185. HRESULT hr = S_OK;
  1186. COLORREF crbO, crtO;
  1187. HBITMAP hbmMaskI;
  1188. int cImageI;
  1189. int n,i;
  1190. BITMAP bm;
  1191. DWORD ColorTableSave[256];
  1192. DWORD ColorTable[256];
  1193. *pi = -1;
  1194. if (GetObject(hbmImageI, sizeof(bm), &bm) != sizeof(bm))
  1195. return E_INVALIDARG;
  1196. hbmMaskI = CreateMonoBitmap(bm.bmWidth, bm.bmHeight);
  1197. if (!hbmMaskI)
  1198. return E_OUTOFMEMORY;
  1199. ENTERCRITICAL;
  1200. // copy color to mono, with crMask turning 1 and all others 0, then
  1201. // punch all crMask pixels in color to 0
  1202. CImageList::SelectSrcBitmap(hbmImageI);
  1203. CImageList::SelectDstBitmap(hbmMaskI);
  1204. // crMask == CLR_DEFAULT, means use the pixel in the upper left
  1205. //
  1206. if (crMask == CLR_DEFAULT)
  1207. crMask = GetPixel(g_hdcSrc, 0, 0);
  1208. // DIBSections dont do color->mono like DDBs do, so we have to do it.
  1209. // this only works for <=8bpp DIBSections, this method does not work
  1210. // for HiColor DIBSections.
  1211. //
  1212. // This code is a workaround for a problem in Win32 when a DIB is converted to
  1213. // monochrome. The conversion is done according to closeness to white or black
  1214. // and without regard to the background color. This workaround is is not required
  1215. // under MainWin.
  1216. //
  1217. // Please note, this code has an endianship problems the comparision in the if statement
  1218. // below is sensitive to endianship
  1219. // ----> if (ColorTableSave[i] == RGB(GetBValue(crMask),GetGValue(crMask),GetRValue(crMask))
  1220. //
  1221. if (bm.bmBits != NULL && bm.bmBitsPixel <= 8)
  1222. {
  1223. n = GetDIBColorTable(g_hdcSrc, 0, 256, (RGBQUAD*)ColorTableSave);
  1224. for (i=0; i<n; i++)
  1225. {
  1226. if (ColorTableSave[i] == RGB(GetBValue(crMask),GetGValue(crMask),GetRValue(crMask)))
  1227. ColorTable[i] = 0x00FFFFFF;
  1228. else
  1229. ColorTable[i] = 0x00000000;
  1230. }
  1231. SetDIBColorTable(g_hdcSrc, 0, n, (RGBQUAD*)ColorTable);
  1232. }
  1233. crbO = ::SetBkColor(g_hdcSrc, crMask);
  1234. BitBlt(g_hdcDst, 0, 0, bm.bmWidth, bm.bmHeight, g_hdcSrc, 0, 0, SRCCOPY);
  1235. ::SetBkColor(g_hdcSrc, 0x00FFFFFFL);
  1236. crtO = SetTextColor(g_hdcSrc, 0x00L);
  1237. BitBlt(g_hdcSrc, 0, 0, bm.bmWidth, bm.bmHeight, g_hdcDst, 0, 0, ROP_DSna);
  1238. ::SetBkColor(g_hdcSrc, crbO);
  1239. SetTextColor(g_hdcSrc, crtO);
  1240. if (bm.bmBits != NULL && bm.bmBitsPixel <= 8)
  1241. {
  1242. SetDIBColorTable(g_hdcSrc, 0, n, (RGBQUAD*)ColorTableSave);
  1243. }
  1244. CImageList::SelectSrcBitmap(NULL);
  1245. CImageList::SelectDstBitmap(NULL);
  1246. ASSERT(_cx);
  1247. cImageI = bm.bmWidth / _cx; // # of images in source
  1248. hr = _Add(hbmImageI, hbmMaskI, cImageI, 0, 0, pi);
  1249. DeleteObject(hbmMaskI);
  1250. LEAVECRITICAL;
  1251. return hr;
  1252. }
  1253. HRESULT CImageList::AddMasked(HBITMAP hbmImage, COLORREF crMask, int* pi)
  1254. {
  1255. if (_pimlMirror)
  1256. {
  1257. HBITMAP hbmMirroredImage = CImageList::_CreateMirroredBitmap(hbmImage, (ILC_PERITEMMIRROR & _flags), _cx);
  1258. _pimlMirror->_AddMasked(hbmMirroredImage, crMask, pi);
  1259. // The caller will take care of deleting hbmImage
  1260. // He knows nothing about hbmMirroredImage
  1261. DeleteObject(hbmMirroredImage);
  1262. }
  1263. return _AddMasked(hbmImage, crMask, pi);
  1264. }
  1265. HRESULT CImageList::_ReplaceValidated(int i, HBITMAP hbmImage, HBITMAP hbmMask)
  1266. {
  1267. HRESULT hr = E_INVALIDARG;
  1268. if (!IsImageListIndex(i))
  1269. return hr;
  1270. ENTERCRITICAL;
  1271. hr = _Replace(i, 1, hbmImage, hbmMask, 0, 0);
  1272. LEAVECRITICAL;
  1273. return hr;
  1274. }
  1275. HRESULT CImageList::Replace(int i, HBITMAP hbmImage, HBITMAP hbmMask)
  1276. {
  1277. if (_pimlMirror)
  1278. {
  1279. HBITMAP hbmMirroredImage = CImageList::_CreateMirroredBitmap(hbmImage, (ILC_PERITEMMIRROR & _flags), _cx);
  1280. if (hbmMirroredImage)
  1281. {
  1282. HBITMAP hbmMirroredMask = NULL;
  1283. if (hbmMask)
  1284. hbmMirroredMask = CImageList::_CreateMirroredBitmap(hbmMask, (ILC_PERITEMMIRROR & _flags), _cx);
  1285. _pimlMirror->_ReplaceValidated(i, hbmMirroredImage, hbmMirroredMask);
  1286. if (hbmMirroredMask)
  1287. DeleteObject(hbmMirroredMask);
  1288. DeleteObject(hbmMirroredImage);
  1289. }
  1290. }
  1291. return _ReplaceValidated(i, hbmImage, hbmMask);
  1292. }
  1293. // replaces images in piml with images from bitmaps
  1294. //
  1295. // in:
  1296. // piml
  1297. // i index in image list to start at (replace)
  1298. // _cImage count of images in source (hbmImage, hbmMask)
  1299. //
  1300. HRESULT CImageList::_Replace(int i, int cImageI, HBITMAP hbmImageI, HBITMAP hbmMaskI,
  1301. int xStart, int yStart)
  1302. {
  1303. RECT rcImage;
  1304. int x, iImage;
  1305. BOOL fBitmapIs32 = FALSE;
  1306. ASSERT(_hbmImage);
  1307. BITMAP bm;
  1308. GetObject(hbmImageI, sizeof(bm), &bm);
  1309. if (bm.bmBitsPixel == 32)
  1310. {
  1311. fBitmapIs32 = TRUE;
  1312. }
  1313. CImageList::SelectSrcBitmap(hbmImageI);
  1314. if (_hdcMask)
  1315. CImageList::SelectDstBitmap(hbmMaskI); // using as just a second source hdc
  1316. for (x = xStart, iImage = 0; iImage < cImageI; iImage++, x += _cx)
  1317. {
  1318. GetImageRect(i + iImage, &rcImage);
  1319. if (_hdcMask)
  1320. {
  1321. BitBlt(_hdcMask, rcImage.left, rcImage.top, _cx, _cy,
  1322. g_hdcDst, x, yStart, SRCCOPY);
  1323. }
  1324. BitBlt(_hdcImage, rcImage.left, rcImage.top, _cx, _cy,
  1325. g_hdcSrc, x, yStart, SRCCOPY);
  1326. if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  1327. {
  1328. DWORD dw = 0;
  1329. if (fBitmapIs32)
  1330. {
  1331. BOOL fHasAlpha = _HasAlpha(i + iImage);
  1332. if (fHasAlpha)
  1333. {
  1334. dw = ILIF_ALPHA;
  1335. _PreProcessImage(i + iImage);
  1336. }
  1337. }
  1338. SetItemFlags(i + iImage, dw);
  1339. }
  1340. }
  1341. _ResetBkColor(i, i + cImageI - 1, _clrBk);
  1342. CImageList::SelectSrcBitmap(NULL);
  1343. if (_hdcMask)
  1344. CImageList::SelectDstBitmap(NULL);
  1345. return S_OK;
  1346. }
  1347. void UnPremultiply(RGBQUAD* pargb, int cx, int cy)
  1348. {
  1349. int cTotal = cx * cy;
  1350. for (int i = 0; i < cTotal; i++)
  1351. {
  1352. RGBQUAD* prgb = &pargb[i];
  1353. if (prgb->rgbReserved != 0)
  1354. {
  1355. prgb->rgbRed = ((255 * prgb->rgbRed) - 128)/prgb->rgbReserved;
  1356. prgb->rgbGreen = ((255 * prgb->rgbGreen) - 128)/prgb->rgbReserved;
  1357. prgb->rgbBlue = ((255 * prgb->rgbBlue) - 128)/prgb->rgbReserved;
  1358. }
  1359. }
  1360. }
  1361. HRESULT CImageList::GetIcon(int i, UINT flags, HICON* phicon)
  1362. {
  1363. UINT cxImage, cyImage;
  1364. HICON hIcon = NULL;
  1365. HBITMAP hbmMask = NULL;
  1366. HBITMAP hbmColor = NULL;
  1367. ICONINFO ii;
  1368. HRESULT hr = E_OUTOFMEMORY;
  1369. RGBQUAD* prgb;
  1370. DWORD fHasAlpha = FALSE;
  1371. ENTERCRITICAL;
  1372. if (!IsImageListIndex(i))
  1373. {
  1374. hr = E_INVALIDARG;
  1375. }
  1376. else
  1377. {
  1378. fHasAlpha = (_GetItemFlags(i) & ILIF_ALPHA);
  1379. }
  1380. LEAVECRITICAL;
  1381. if (E_INVALIDARG == hr)
  1382. return hr;
  1383. cxImage = _cx;
  1384. cyImage = _cy;
  1385. if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  1386. {
  1387. // If the source image is not an alpha image, we need to create a lower than 32bpp dib.
  1388. // We need to do this because if the overlay contains an alpha channel, this will
  1389. // be propogated to the final icon, and the only visible portion will be the link item.
  1390. BITMAPINFO bi = {0};
  1391. bi.bmiHeader.biSize = sizeof(bi.bmiHeader);
  1392. bi.bmiHeader.biWidth = cxImage;
  1393. bi.bmiHeader.biHeight = cyImage;
  1394. bi.bmiHeader.biPlanes = 1;
  1395. bi.bmiHeader.biBitCount = fHasAlpha?32:24;
  1396. bi.bmiHeader.biCompression = BI_RGB;
  1397. HDC hdcScreen = GetDC(NULL);
  1398. if (hdcScreen)
  1399. {
  1400. hbmColor = CreateDIBSection(hdcScreen, &bi, DIB_RGB_COLORS, (void**)&prgb, NULL, 0);
  1401. ReleaseDC(NULL, hdcScreen);
  1402. }
  1403. flags |= ILD_PRESERVEALPHA;
  1404. }
  1405. else
  1406. {
  1407. hbmColor = CreateColorBitmap(cxImage, cyImage);
  1408. }
  1409. if (hbmColor)
  1410. {
  1411. hbmMask = CreateMonoBitmap(cxImage, cyImage);
  1412. if (hbmMask)
  1413. {
  1414. ENTERCRITICAL;
  1415. CImageList::SelectDstBitmap(hbmMask);
  1416. PatBlt(g_hdcDst, 0, 0, cxImage, cyImage, WHITENESS);
  1417. WimpyDraw(SAFECAST(this, IImageList*), i, g_hdcDst, 0, 0, ILD_MASK | flags);
  1418. CImageList::SelectDstBitmap(hbmColor);
  1419. PatBlt(g_hdcDst, 0, 0, cxImage, cyImage, BLACKNESS);
  1420. WimpyDraw(SAFECAST(this, IImageList*), i, g_hdcDst, 0, 0, ILD_TRANSPARENT | flags);
  1421. CImageList::SelectDstBitmap(NULL);
  1422. LEAVECRITICAL;
  1423. if (fHasAlpha)
  1424. {
  1425. UnPremultiply(prgb, _cx, _cy);
  1426. }
  1427. ii.fIcon = TRUE;
  1428. ii.xHotspot = 0;
  1429. ii.yHotspot = 0;
  1430. ii.hbmColor = hbmColor;
  1431. ii.hbmMask = hbmMask;
  1432. hIcon = CreateIconIndirect(&ii);
  1433. DeleteObject(hbmMask);
  1434. hr = S_OK;
  1435. }
  1436. DeleteObject(hbmColor);
  1437. }
  1438. *phicon = hIcon;
  1439. return hr;
  1440. }
  1441. // this removes an image from the bitmap but doing all the
  1442. // proper shuffling.
  1443. //
  1444. // this does the following:
  1445. // if the bitmap being removed is not the last in the row
  1446. // it blts the images to the right of the one being deleted
  1447. // to the location of the one being deleted (covering it up)
  1448. //
  1449. // for all rows until the last row (where the last image is)
  1450. // move the image from the next row up to the last position
  1451. // in the current row. then slide over all images in that
  1452. // row to the left.
  1453. void CImageList::_RemoveItemBitmap(int i)
  1454. {
  1455. RECT rc1;
  1456. RECT rc2;
  1457. int dx, y;
  1458. int x;
  1459. GetImageRect(i, &rc1);
  1460. GetImageRect(_cImage - 1, &rc2);
  1461. if (i < _cImage &&
  1462. (_flags & ILC_COLORMASK) == ILC_COLOR32)
  1463. {
  1464. DSA_DeleteItem(_dsaFlags, i);
  1465. }
  1466. SetItemFlags(_cImage, 0);
  1467. // the row with the image being deleted, do we need to shuffle?
  1468. // amount of stuff to shuffle
  1469. dx = _cStrip * _cx - rc1.right;
  1470. if (dx)
  1471. {
  1472. // yes, shuffle things left
  1473. BitBlt(_hdcImage, rc1.left, rc1.top, dx, _cy, _hdcImage, rc1.right, rc1.top, SRCCOPY);
  1474. if (_hdcMask)
  1475. BitBlt(_hdcMask, rc1.left, rc1.top, dx, _cy, _hdcMask, rc1.right, rc1.top, SRCCOPY);
  1476. }
  1477. y = rc1.top; // top of row we are working on
  1478. x = _cx * (_cStrip - 1); // x coord of last bitmaps in each row
  1479. while (y < rc2.top)
  1480. {
  1481. // copy first from row below to last image position on this row
  1482. BitBlt(_hdcImage, x, y,
  1483. _cx, _cy, _hdcImage, 0, y + _cy, SRCCOPY);
  1484. if (_hdcMask)
  1485. BitBlt(_hdcMask, x, y,
  1486. _cx, _cy, _hdcMask, 0, y + _cy, SRCCOPY);
  1487. y += _cy; // jump to row to slide left
  1488. if (y <= rc2.top)
  1489. {
  1490. // slide the rest over to the left
  1491. BitBlt(_hdcImage, 0, y, x, _cy,
  1492. _hdcImage, _cx, y, SRCCOPY);
  1493. // slide the rest over to the left
  1494. if (_hdcMask)
  1495. {
  1496. BitBlt(_hdcMask, 0, y, x, _cy,
  1497. _hdcMask, _cx, y, SRCCOPY);
  1498. }
  1499. }
  1500. }
  1501. }
  1502. //
  1503. // ImageList_Remove - remove a image from the image list
  1504. //
  1505. // i - image to remove, or -1 to remove all images.
  1506. //
  1507. // NOTE all images are "shifted" down, ie all image index's
  1508. // above the one deleted are changed by 1
  1509. //
  1510. HRESULT CImageList::_Remove(int i)
  1511. {
  1512. HRESULT hr = S_OK;
  1513. ENTERCRITICAL;
  1514. if (i == -1)
  1515. {
  1516. _cImage = 0;
  1517. _cAlloc = 0;
  1518. for (i=0; i<NUM_OVERLAY_IMAGES; i++)
  1519. _aOverlayIndexes[i] = -1;
  1520. if (_dsaFlags)
  1521. {
  1522. DSA_Destroy(_dsaFlags);
  1523. _dsaFlags = NULL;
  1524. }
  1525. _ReAllocBitmaps(-_cGrow);
  1526. }
  1527. else
  1528. {
  1529. if (!IsImageListIndex(i))
  1530. {
  1531. hr = E_INVALIDARG;
  1532. }
  1533. else
  1534. {
  1535. _RemoveItemBitmap(i);
  1536. --_cImage;
  1537. if (_cAlloc - (_cImage + 1) > _cGrow)
  1538. _ReAllocBitmaps(_cAlloc - _cGrow);
  1539. }
  1540. }
  1541. LEAVECRITICAL;
  1542. return hr;
  1543. }
  1544. HRESULT CImageList::Remove(int i)
  1545. {
  1546. if (_pimlMirror)
  1547. {
  1548. _pimlMirror->_Remove(i);
  1549. }
  1550. return _Remove(i);
  1551. }
  1552. BOOL CImageList::_IsSameObject(IUnknown* punk)
  1553. {
  1554. BOOL fRet = FALSE;
  1555. IUnknown* me;
  1556. IUnknown* them;
  1557. if (punk == NULL)
  1558. return FALSE;
  1559. QueryInterface(IID_PPV_ARG(IUnknown, &me));
  1560. if (SUCCEEDED(punk->QueryInterface(IID_PPV_ARG(IUnknown, &them))))
  1561. {
  1562. fRet = (me == them);
  1563. them->Release();
  1564. }
  1565. me->Release();
  1566. return fRet;
  1567. }
  1568. //
  1569. // ImageList_Copy - move an image in the image list
  1570. //
  1571. HRESULT CImageList::Copy(int iDst, IUnknown* punkSrc, int iSrc, UINT uFlags)
  1572. {
  1573. RECT rcDst, rcSrc, rcTmp;
  1574. CImageList* pimlTmp;
  1575. CImageList* pimlSrc;
  1576. HRESULT hr = E_FAIL;
  1577. if (uFlags & ~ILCF_VALID)
  1578. {
  1579. // don't let hosers pass bogus flags
  1580. RIPMSG(0, "ImageList_Copy: Invalid flags %08x", uFlags);
  1581. return E_INVALIDARG;
  1582. }
  1583. // Not supported
  1584. if (!_IsSameObject(punkSrc))
  1585. {
  1586. return E_INVALIDARG;
  1587. }
  1588. // We only support copies on ourself... Weird
  1589. pimlSrc = this;
  1590. ENTERCRITICAL;
  1591. pimlTmp = (uFlags & ILCF_SWAP)? pimlSrc : NULL;
  1592. if (SUCCEEDED(GetImageRect(iDst, &rcDst)) &&
  1593. SUCCEEDED(pimlSrc->GetImageRect(iSrc, &rcSrc)) &&
  1594. (!pimlTmp || pimlTmp->GetSpareImageRect(&rcTmp)))
  1595. {
  1596. int cx = pimlSrc->_cx;
  1597. int cy = pimlSrc->_cy;
  1598. //
  1599. // iff we are swapping we need to save the destination image
  1600. //
  1601. if (pimlTmp)
  1602. {
  1603. BitBlt(pimlTmp->_hdcImage, rcTmp.left, rcTmp.top, cx, cy,
  1604. _hdcImage, rcDst.left, rcDst.top, SRCCOPY);
  1605. if (pimlTmp->_hdcMask)
  1606. {
  1607. BitBlt(pimlTmp->_hdcMask, rcTmp.left, rcTmp.top, cx, cy,
  1608. _hdcMask, rcDst.left, rcDst.top, SRCCOPY);
  1609. }
  1610. }
  1611. //
  1612. // copy the image
  1613. //
  1614. BitBlt(_hdcImage, rcDst.left, rcDst.top, cx, cy,
  1615. pimlSrc->_hdcImage, rcSrc.left, rcSrc.top, SRCCOPY);
  1616. if (pimlSrc->_hdcMask)
  1617. {
  1618. BitBlt(_hdcMask, rcDst.left, rcDst.top, cx, cy,
  1619. pimlSrc->_hdcMask, rcSrc.left, rcSrc.top, SRCCOPY);
  1620. }
  1621. //
  1622. // iff we are swapping we need to copy the saved image too
  1623. //
  1624. if (pimlTmp)
  1625. {
  1626. BitBlt(pimlSrc->_hdcImage, rcSrc.left, rcSrc.top, cx, cy,
  1627. pimlTmp->_hdcImage, rcTmp.left, rcTmp.top, SRCCOPY);
  1628. if (pimlSrc->_hdcMask)
  1629. {
  1630. BitBlt(pimlSrc->_hdcMask, rcSrc.left, rcSrc.top, cx, cy,
  1631. pimlTmp->_hdcMask, rcTmp.left, rcTmp.top, SRCCOPY);
  1632. }
  1633. }
  1634. hr = S_OK;
  1635. }
  1636. LEAVECRITICAL;
  1637. return hr;
  1638. }
  1639. // IS_WHITE_PIXEL, BITS_ALL_WHITE are macros for looking at monochrome bits
  1640. // to determine if certain pixels are white or black. Note that within a byte
  1641. // the most significant bit represents the left most pixel.
  1642. //
  1643. #define IS_WHITE_PIXEL(pj,x,y,cScan) \
  1644. ((pj)[((y) * (cScan)) + ((x) >> 3)] & (1 << (7 - ((x) & 7))))
  1645. #define BITS_ALL_WHITE(b) (b == 0xff)
  1646. // Set the image iImage as one of the special images for us in combine
  1647. // drawing. to draw with these specify the index of this
  1648. // in:
  1649. // piml imagelist
  1650. // iImage image index to use in speical drawing
  1651. // iOverlay index of special image, values 1-4
  1652. HRESULT CImageList::_SetOverlayImage(int iImage, int iOverlay)
  1653. {
  1654. RECT rcImage;
  1655. RECT rc;
  1656. int x,y;
  1657. int cxI,cyI;
  1658. ULONG cScan;
  1659. ULONG cBits;
  1660. HBITMAP hbmMem;
  1661. HRESULT hr = S_FALSE;
  1662. iOverlay--; // make zero based
  1663. if (_hdcMask == NULL ||
  1664. iImage < 0 || iImage >= _cImage ||
  1665. iOverlay < 0 || iOverlay >= NUM_OVERLAY_IMAGES)
  1666. {
  1667. return E_INVALIDARG;
  1668. }
  1669. if (_aOverlayIndexes[iOverlay] == (SHORT)iImage)
  1670. return S_OK;
  1671. _aOverlayIndexes[iOverlay] = (SHORT)iImage;
  1672. //
  1673. // find minimal rect that bounds the image
  1674. //
  1675. GetImageRect(iImage, &rcImage);
  1676. SetRect(&rc, 0x7FFF, 0x7FFF, 0, 0);
  1677. //
  1678. // now compute the black box. This is much faster than GetPixel but
  1679. // could still be improved by doing more operations looking at entire
  1680. // bytes. We basicaly get the bits in monochrome form and then use
  1681. // a private GetPixel. This decreased time on NT from 50 milliseconds to
  1682. // 1 millisecond for a 32X32 image.
  1683. //
  1684. cxI = rcImage.right - rcImage.left;
  1685. cyI = rcImage.bottom - rcImage.top;
  1686. // compute the number of bytes in a scan. Note that they are WORD alligned
  1687. cScan = (((cxI + (sizeof(SHORT)*8 - 1)) / 16) * 2);
  1688. cBits = cScan * cyI;
  1689. hbmMem = CreateMonoBitmap(cxI,cyI);
  1690. if (hbmMem)
  1691. {
  1692. HDC hdcMem = CreateCompatibleDC(_hdcMask);
  1693. if (hdcMem)
  1694. {
  1695. PBYTE pBits = (PBYTE)LocalAlloc(LMEM_FIXED,cBits);
  1696. PBYTE pScan;
  1697. if (pBits)
  1698. {
  1699. SelectObject(hdcMem,hbmMem);
  1700. //
  1701. // map black pixels to 0, white to 1
  1702. //
  1703. BitBlt(hdcMem, 0, 0, cxI, cyI, _hdcMask, rcImage.left, rcImage.top, SRCCOPY);
  1704. //
  1705. // fill in the bits
  1706. //
  1707. GetBitmapBits(hbmMem,cBits,pBits);
  1708. //
  1709. // for each scan, find the bounds
  1710. //
  1711. for (y = 0, pScan = pBits; y < cyI; ++y,pScan += cScan)
  1712. {
  1713. int i;
  1714. //
  1715. // first go byte by byte through white space
  1716. //
  1717. for (x = 0, i = 0; (i < (cxI >> 3)) && BITS_ALL_WHITE(pScan[i]); ++i)
  1718. {
  1719. x += 8;
  1720. }
  1721. //
  1722. // now finish the scan bit by bit
  1723. //
  1724. for (; x < cxI; ++x)
  1725. {
  1726. if (!IS_WHITE_PIXEL(pBits, x,y,cScan))
  1727. {
  1728. rc.left = min(rc.left, x);
  1729. rc.right = max(rc.right, x+1);
  1730. rc.top = min(rc.top, y);
  1731. rc.bottom = max(rc.bottom, y+1);
  1732. // now that we found one, quickly jump to the known right edge
  1733. if ((x >= rc.left) && (x < rc.right))
  1734. {
  1735. x = rc.right-1;
  1736. }
  1737. }
  1738. }
  1739. }
  1740. if (rc.left == 0x7FFF)
  1741. {
  1742. rc.left = 0;
  1743. TraceMsg(TF_ERROR, "SetOverlayImage: Invalid image. No white pixels specified");
  1744. }
  1745. if (rc.top == 0x7FFF)
  1746. {
  1747. rc.top = 0;
  1748. TraceMsg(TF_ERROR, "SetOverlayImage: Invalid image. No white pixels specified");
  1749. }
  1750. _aOverlayDX[iOverlay] = (SHORT)(rc.right - rc.left);
  1751. _aOverlayDY[iOverlay] = (SHORT)(rc.bottom- rc.top);
  1752. _aOverlayX[iOverlay] = (SHORT)(rc.left);
  1753. _aOverlayY[iOverlay] = (SHORT)(rc.top);
  1754. _aOverlayF[iOverlay] = 0;
  1755. //
  1756. // see if the image is non-rectanglar
  1757. //
  1758. // if the overlay does not require a mask to be drawn set the
  1759. // ILD_IMAGE flag, this causes ImageList_DrawEx to just
  1760. // draw the image, ignoring the mask.
  1761. //
  1762. for (y=rc.top; y<rc.bottom; y++)
  1763. {
  1764. for (x=rc.left; x<rc.right; x++)
  1765. {
  1766. if (IS_WHITE_PIXEL(pBits, x, y,cScan))
  1767. break;
  1768. }
  1769. if (x != rc.right)
  1770. break;
  1771. }
  1772. if (y == rc.bottom)
  1773. _aOverlayF[iOverlay] = ILD_IMAGE;
  1774. LocalFree(pBits);
  1775. hr = S_OK;
  1776. }
  1777. DeleteDC(hdcMem);
  1778. }
  1779. DeleteObject(hbmMem);
  1780. }
  1781. return hr;
  1782. }
  1783. HRESULT CImageList::SetOverlayImage(int iImage, int iOverlay)
  1784. {
  1785. if (_pimlMirror)
  1786. {
  1787. _pimlMirror->_SetOverlayImage(iImage, iOverlay);
  1788. }
  1789. return _SetOverlayImage(iImage, iOverlay);
  1790. }
  1791. /*
  1792. ** BlendCT
  1793. **
  1794. */
  1795. void CImageList::BlendCTHelper(DWORD *pdw, DWORD rgb, UINT n, UINT count)
  1796. {
  1797. UINT i;
  1798. for (i=0; i<count; i++)
  1799. {
  1800. pdw[i] = RGB(
  1801. ((UINT)GetRValue(pdw[i]) * (100-n) + (UINT)GetBValue(rgb) * (n)) / 100,
  1802. ((UINT)GetGValue(pdw[i]) * (100-n) + (UINT)GetGValue(rgb) * (n)) / 100,
  1803. ((UINT)GetBValue(pdw[i]) * (100-n) + (UINT)GetRValue(rgb) * (n)) / 100);
  1804. }
  1805. }
  1806. /*
  1807. ** BlendDither
  1808. **
  1809. ** copy the source to the dest blended with the given color.
  1810. **
  1811. ** simulate a blend with a dither pattern.
  1812. **
  1813. */
  1814. void CImageList::BlendDither(HDC hdcDst, int xDst, int yDst, int x, int y, int cx, int cy, COLORREF rgb, UINT fStyle)
  1815. {
  1816. HBRUSH hbr;
  1817. HBRUSH hbrT;
  1818. HBRUSH hbrMask;
  1819. HBRUSH hbrFree = NULL; // free if non-null
  1820. ASSERT(GetTextColor(hdcDst) == CLR_BLACK);
  1821. ASSERT(::GetBkColor(hdcDst) == CLR_WHITE);
  1822. // choose a dither/blend brush
  1823. switch (fStyle & ILD_BLENDMASK)
  1824. {
  1825. default:
  1826. case ILD_BLEND50:
  1827. hbrMask = g_hbrMonoDither;
  1828. break;
  1829. }
  1830. // create (or use a existing) brush for the blend color
  1831. switch (rgb)
  1832. {
  1833. case CLR_DEFAULT:
  1834. hbr = g_hbrHighlight;
  1835. break;
  1836. case CLR_NONE:
  1837. hbr = _hbrBk;
  1838. break;
  1839. default:
  1840. if (rgb == _clrBk)
  1841. hbr = _hbrBk;
  1842. else
  1843. hbr = hbrFree = CreateSolidBrush(rgb);
  1844. break;
  1845. }
  1846. hbrT = (HBRUSH)SelectObject(hdcDst, hbr);
  1847. PatBlt(hdcDst, xDst, yDst, cx, cy, PATCOPY);
  1848. SelectObject(hdcDst, hbrT);
  1849. hbrT = (HBRUSH)SelectObject(hdcDst, hbrMask);
  1850. BitBlt(hdcDst, xDst, yDst, cx, cy, _hdcImage, x, y, ROP_MaskPat);
  1851. SelectObject(hdcDst, hbrT);
  1852. if (hbrFree)
  1853. DeleteBrush(hbrFree);
  1854. }
  1855. /*
  1856. ** BlendCT
  1857. **
  1858. ** copy the source to the dest blended with the given color.
  1859. **
  1860. */
  1861. void CImageList::BlendCT(HDC hdcDst, int xDst, int yDst, int x, int y, int cx, int cy, COLORREF rgb, UINT fStyle)
  1862. {
  1863. BITMAP bm;
  1864. GetObject(_hbmImage, sizeof(bm), &bm);
  1865. if (rgb == CLR_DEFAULT)
  1866. rgb = GetSysColor(COLOR_HIGHLIGHT);
  1867. ASSERT(rgb != CLR_NONE);
  1868. //
  1869. // get the DIB color table and blend it, only do this when the
  1870. // blend color changes
  1871. //
  1872. if (_clrBlend != rgb)
  1873. {
  1874. int n,cnt;
  1875. _clrBlend = rgb;
  1876. GetObject(_hbmImage, sizeof(dib), &dib.bm);
  1877. cnt = GetDIBColorTable(_hdcImage, 0, 256, (LPRGBQUAD)&dib.ct);
  1878. if ((fStyle & ILD_BLENDMASK) == ILD_BLEND50)
  1879. n = 50;
  1880. else
  1881. n = 25;
  1882. BlendCTHelper(dib.ct, rgb, n, cnt);
  1883. }
  1884. //
  1885. // draw the image with a different color table
  1886. //
  1887. StretchDIBits(hdcDst, xDst, yDst, cx, cy,
  1888. x, dib.bi.biHeight-(y+cy), cx, cy,
  1889. bm.bmBits, (LPBITMAPINFO)&dib.bi, DIB_RGB_COLORS, SRCCOPY);
  1890. }
  1891. // RGB555 macros
  1892. #define RGB555(r,g,b) (((((r)>>3)&0x1F)<<10) | ((((g)>>3)&0x1F)<<5) | (((b)>>3)&0x1F))
  1893. #define R_555(w) (int)(((w) >> 7) & 0xF8)
  1894. #define G_555(w) (int)(((w) >> 2) & 0xF8)
  1895. #define B_555(w) (int)(((w) << 3) & 0xF8)
  1896. void CImageList::Blend16Helper(int xSrc, int ySrc, int xDst, int yDst, int cx, int cy, COLORREF rgb, int a) // alpha value
  1897. {
  1898. // If it's odd, Adjust.
  1899. if ((cx & 1) == 1)
  1900. {
  1901. cx++;
  1902. }
  1903. if (rgb == CLR_NONE)
  1904. {
  1905. // blending with the destination, we ignore the alpha and always
  1906. // do 50% (this is what the old dither mask code did)
  1907. int ys = ySrc;
  1908. int yd = yDst;
  1909. for (; ys < ySrc + cy; ys++, yd++)
  1910. {
  1911. WORD* pSrc = &((WORD*)_pargbImage)[xSrc + ys * cx]; // Cast because we've gotten to this case because we are a 555 imagelist
  1912. WORD* pDst = &((WORD*)_pargbImage)[xDst + yd * cx];
  1913. for (int x = 0; x < cx; x++)
  1914. {
  1915. *pDst++ = ((*pDst & 0x7BDE) >> 1) + ((*pSrc++ & 0x7BDE) >> 1);
  1916. }
  1917. }
  1918. }
  1919. else
  1920. {
  1921. // blending with a solid color
  1922. // pre multiply source (constant) rgb by alpha
  1923. int sr = GetRValue(rgb) * a;
  1924. int sg = GetGValue(rgb) * a;
  1925. int sb = GetBValue(rgb) * a;
  1926. // compute inverse alpha for inner loop
  1927. a = 256 - a;
  1928. // special case a 50% blend, to avoid a multiply
  1929. if (a == 128)
  1930. {
  1931. sr = RGB555(sr>>8,sg>>8,sb>>8);
  1932. int ys = ySrc;
  1933. int yd = yDst;
  1934. for (; ys < ySrc + cy; ys++, yd++)
  1935. {
  1936. WORD* pSrc = &((WORD*)_pargbImage)[xSrc + ys * cx];
  1937. WORD* pDst = &((WORD*)_pargbImage)[xDst + yd * cx];
  1938. for (int x = 0; x < cx; x++)
  1939. {
  1940. int i = *pSrc++;
  1941. i = sr + ((i & 0x7BDE) >> 1);
  1942. *pDst++ = (WORD) i;
  1943. }
  1944. }
  1945. }
  1946. else
  1947. {
  1948. int ys = ySrc;
  1949. int yd = yDst;
  1950. for (; ys < ySrc + cy; ys++, yd++)
  1951. {
  1952. WORD* pSrc = &((WORD*)_pargbImage)[xSrc + ys * cx];
  1953. WORD* pDst = &((WORD*)_pargbImage)[xDst + yd * cx];
  1954. for (int x = 0; x < cx; x++)
  1955. {
  1956. int i = *pSrc++;
  1957. int r = (R_555(i) * a + sr) >> 8;
  1958. int g = (G_555(i) * a + sg) >> 8;
  1959. int b = (B_555(i) * a + sb) >> 8;
  1960. *pDst++ = RGB555(r,g,b);
  1961. }
  1962. }
  1963. }
  1964. }
  1965. }
  1966. /*
  1967. ** ImageList_Blend16
  1968. **
  1969. ** copy the source to the dest blended with the given color.
  1970. **
  1971. ** source is assumed to be a 16 bit (RGB 555) bottom-up DIBSection
  1972. ** (this is the only kind of DIBSection we create)
  1973. */
  1974. void CImageList::Blend16(HDC hdcDst, int xDst, int yDst, int iImage, int cx, int cy, COLORREF rgb, UINT fStyle)
  1975. {
  1976. BITMAP bm;
  1977. RECT rc;
  1978. RECT rcSpare;
  1979. RECT rcSpareInverted;
  1980. int a, x, y;
  1981. // get bitmap info for source bitmap
  1982. GetObject(_hbmImage, sizeof(bm), &bm);
  1983. ASSERT(bm.bmBitsPixel==16);
  1984. // get blend RGB
  1985. if (rgb == CLR_DEFAULT)
  1986. rgb = GetSysColor(COLOR_HIGHLIGHT);
  1987. // get blend factor as a fraction of 256
  1988. // only 50% or 25% is currently used.
  1989. if ((fStyle & ILD_BLENDMASK) == ILD_BLEND50)
  1990. a = 128;
  1991. else
  1992. a = 64;
  1993. GetImageRectInverted(iImage, &rc);
  1994. x = rc.left;
  1995. y = rc.top;
  1996. // blend the image with the specified color and place at end of image list
  1997. if (GetSpareImageRectInverted(&rcSpareInverted) &&
  1998. GetSpareImageRect(&rcSpare))
  1999. {
  2000. // if blending with the destination, copy the dest to our work buffer
  2001. if (rgb == CLR_NONE)
  2002. BitBlt(_hdcImage, rcSpare.left, rcSpare.top, cx, cy, hdcDst, xDst, yDst, SRCCOPY);
  2003. // sometimes the user can change the icon size (via plustab) between 32x32 and 48x48,
  2004. // thus the values we have might be bigger than the actual bitmap. To prevent us from
  2005. // crashing in Blend16 when this happens we do some bounds checks here
  2006. if (rc.left + cx <= bm.bmWidth &&
  2007. rc.top + cy <= bm.bmHeight &&
  2008. x + cx <= bm.bmWidth &&
  2009. y + cy <= bm.bmHeight)
  2010. {
  2011. // Needs inverted coordinates
  2012. Blend16Helper(x, y, rcSpareInverted.left, rcSpareInverted.top, cx, cy, rgb, a);
  2013. }
  2014. // blt blended image to the dest DC
  2015. BitBlt(hdcDst, xDst, yDst, cx, cy, _hdcImage, rcSpare.left, rcSpare.top, SRCCOPY);
  2016. }
  2017. }
  2018. #define ALPHA_50 128
  2019. #define ALPHA_25 64
  2020. void CImageList::_GenerateAlphaForImageUsingMask(int iImage, BOOL fSpare)
  2021. {
  2022. RECT rcImage;
  2023. RECT rcInverted;
  2024. HRESULT hr;
  2025. GetImageRect(iImage, &rcImage);
  2026. if (fSpare)
  2027. {
  2028. hr = GetSpareImageRectInverted(&rcInverted);
  2029. }
  2030. else
  2031. {
  2032. hr = GetImageRectInverted(iImage, &rcInverted);
  2033. }
  2034. if (!SUCCEEDED(hr))
  2035. return;
  2036. BITMAPINFO bi = {0};
  2037. bi.bmiHeader.biSize = sizeof(bi.bmiHeader);
  2038. bi.bmiHeader.biWidth = _cx;
  2039. bi.bmiHeader.biHeight = _cy;
  2040. bi.bmiHeader.biPlanes = 1;
  2041. bi.bmiHeader.biBitCount = 32;
  2042. bi.bmiHeader.biCompression = BI_RGB;
  2043. HDC hdcMem = CreateCompatibleDC(_hdcMask);
  2044. if (hdcMem)
  2045. {
  2046. RGBQUAD* pbMask;
  2047. HBITMAP hbmp = CreateDIBSection(hdcMem, &bi, DIB_RGB_COLORS, (void**)&pbMask, NULL, 0);
  2048. if (hbmp)
  2049. {
  2050. HBITMAP hbmpOld = (HBITMAP)SelectObject(hdcMem, hbmp);
  2051. SetTextColor(hdcMem, RGB(0xFF, 0xFF, 0xFF));
  2052. ::SetBkColor(hdcMem, RGB(0x0,0x0,0x0));
  2053. BitBlt(hdcMem, 0, 0, _cx, _cy, _hdcMask, rcImage.left, rcImage.top, SRCCOPY);
  2054. for (int y = 0; y < _cy; y++)
  2055. {
  2056. int Offset = (y + rcInverted.top) * _cx;
  2057. int MaskOffset = y * _cx;
  2058. for (int x = 0; x < _cx; x++)
  2059. {
  2060. RGBQUAD* prgb = &_pargbImage[x + rcInverted.left + Offset];
  2061. if (pbMask[x + MaskOffset].rgbBlue != 0)
  2062. {
  2063. prgb->rgbReserved = 255;
  2064. }
  2065. else
  2066. {
  2067. *(DWORD*)prgb = 0;
  2068. }
  2069. }
  2070. }
  2071. SelectObject(hdcMem, hbmpOld);
  2072. DeleteObject(hbmp);
  2073. }
  2074. DeleteDC(hdcMem);
  2075. }
  2076. if (!fSpare)
  2077. {
  2078. SetItemFlags(iImage, ILIF_ALPHA);
  2079. _PreProcessImage(iImage);
  2080. }
  2081. }
  2082. void ScaleAlpha(RGBQUAD* prgbImage, RECT* prc, int aScale)
  2083. {
  2084. int cx = RECTWIDTH(*prc);
  2085. for (int y = prc->top; y < prc->bottom; y++)
  2086. {
  2087. int Offset = y * cx;
  2088. for (int x = prc->left; x < prc->right; x++)
  2089. {
  2090. RGBQUAD* prgb = &prgbImage[x + Offset];
  2091. if (prgb->rgbReserved != 0)
  2092. {
  2093. prgb->rgbReserved = (BYTE)(prgb->rgbReserved / aScale); // New alpha
  2094. prgb->rgbRed = ((prgb->rgbRed * prgb->rgbReserved) + 128) / 255;
  2095. prgb->rgbGreen = ((prgb->rgbGreen * prgb->rgbReserved) + 128) / 255;
  2096. prgb->rgbBlue = ((prgb->rgbBlue * prgb->rgbReserved) + 128) / 255;
  2097. }
  2098. }
  2099. }
  2100. }
  2101. #define COLORBLEND_ALPHA 128
  2102. BOOL CImageList::Blend32(HDC hdcDst, int xDst, int yDst, int iImage, int cx, int cy, COLORREF rgb, UINT fStyle)
  2103. {
  2104. BITMAP bm;
  2105. RECT rc;
  2106. RECT rcSpare;
  2107. RECT rcSpareInverted;
  2108. int aScale;
  2109. BOOL fBlendWithColor = FALSE;
  2110. int r,g,b;
  2111. // get bitmap info for source bitmap
  2112. GetObject(_hbmImage, sizeof(bm), &bm);
  2113. ASSERT(bm.bmBitsPixel==32);
  2114. // get blend RGB
  2115. if (rgb == CLR_DEFAULT)
  2116. {
  2117. rgb = GetSysColor(COLOR_HIGHLIGHT);
  2118. fBlendWithColor = TRUE;
  2119. r = GetRValue(rgb) * COLORBLEND_ALPHA;
  2120. g = GetGValue(rgb) * COLORBLEND_ALPHA;
  2121. b = GetBValue(rgb) * COLORBLEND_ALPHA;
  2122. }
  2123. // get blend factor as a fraction of 256
  2124. // only 50% or 25% is currently used.
  2125. if ((fStyle & ILD_BLENDMASK) == ILD_BLEND50 || rgb == CLR_NONE)
  2126. aScale = 2;
  2127. else
  2128. aScale = 4;
  2129. GetImageRect(iImage, &rc);
  2130. if (GetSpareImageRectInverted(&rcSpareInverted) &&
  2131. GetSpareImageRect(&rcSpare))
  2132. {
  2133. BitBlt(_hdcImage, rcSpare.left, rcSpare.top, _cx, _cy, _hdcImage, rc.left, rc.top, SRCCOPY);
  2134. BOOL fHasAlpha = (_GetItemFlags(iImage) & ILIF_ALPHA);
  2135. if (!fHasAlpha)
  2136. {
  2137. _GenerateAlphaForImageUsingMask(iImage, TRUE);
  2138. }
  2139. // if blending with the destination, copy the dest to our work buffer
  2140. if (rgb == CLR_NONE)
  2141. {
  2142. ScaleAlpha(_pargbImage, &rcSpareInverted, aScale);
  2143. BLENDFUNCTION bf = {0};
  2144. bf.BlendOp = AC_SRC_OVER;
  2145. bf.SourceConstantAlpha = 255;
  2146. bf.AlphaFormat = AC_SRC_ALPHA;
  2147. bf.BlendFlags = AC_MIRRORBITMAP | ((fStyle & ILD_DPISCALE)?AC_USE_HIGHQUALITYFILTER:0);
  2148. GdiAlphaBlend(hdcDst, xDst, yDst, cx, cy, _hdcImage, rcSpare.left, rcSpare.top, _cx, _cy, bf);
  2149. return FALSE;
  2150. }
  2151. else
  2152. {
  2153. if (fBlendWithColor)
  2154. {
  2155. for (int y = rcSpareInverted.top; y < rcSpareInverted.bottom; y++)
  2156. {
  2157. int Offset = y * _cx;
  2158. for (int x = rcSpareInverted.left; x < rcSpareInverted.right; x++)
  2159. {
  2160. RGBQUAD* prgb = &_pargbImage[x + Offset];
  2161. if (prgb->rgbReserved > 128)
  2162. {
  2163. prgb->rgbRed = (prgb->rgbRed * COLORBLEND_ALPHA + r) / 255;
  2164. prgb->rgbGreen = (prgb->rgbGreen * COLORBLEND_ALPHA + g) / 255;
  2165. prgb->rgbBlue = (prgb->rgbBlue * COLORBLEND_ALPHA + b) / 255;
  2166. }
  2167. }
  2168. }
  2169. }
  2170. else
  2171. {
  2172. ScaleAlpha(_pargbImage, &rcSpareInverted, aScale);
  2173. }
  2174. BitBlt(hdcDst, xDst, yDst, cx, cy, _hdcImage, rcSpare.left, rcSpare.top, SRCCOPY);
  2175. return TRUE;
  2176. }
  2177. }
  2178. return FALSE;
  2179. }
  2180. /*
  2181. ** ImageList_Blend
  2182. **
  2183. ** copy the source to the dest blended with the given color.
  2184. ** top level function to decide what blend function to call
  2185. */
  2186. BOOL CImageList::Blend(HDC hdcDst, int xDst, int yDst, int iImage, int cx, int cy, COLORREF rgb, UINT fStyle)
  2187. {
  2188. BOOL fRet = FALSE;
  2189. BITMAP bm;
  2190. RECT rc;
  2191. int bpp = GetDeviceCaps(hdcDst, BITSPIXEL);
  2192. GetObject(_hbmImage, sizeof(bm), &bm);
  2193. GetImageRect(iImage, &rc);
  2194. //
  2195. // if _hbmImage is a DIBSection and we are on a HiColor device
  2196. // the do a "real" blend
  2197. //
  2198. if (bm.bmBits && bm.bmBitsPixel <= 8 && (bpp > 8 || bm.bmBitsPixel==8))
  2199. {
  2200. // blend from a 4bit or 8bit DIB
  2201. BlendCT(hdcDst, xDst, yDst, rc.left, rc.top, cx, cy, rgb, fStyle);
  2202. }
  2203. else if (bm.bmBits && bm.bmBitsPixel == 16 && bpp > 8)
  2204. {
  2205. // blend from a 16bit 555 DIB
  2206. Blend16(hdcDst, xDst, yDst, iImage, cx, cy, rgb, fStyle);
  2207. }
  2208. else if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  2209. {
  2210. fRet = Blend32(hdcDst, xDst, yDst, iImage, cx, cy, rgb, fStyle);
  2211. }
  2212. else
  2213. {
  2214. // simulate a blend with a dither pattern.
  2215. BlendDither(hdcDst, xDst, yDst, rc.left, rc.top, cx, cy, rgb, fStyle);
  2216. }
  2217. return fRet;
  2218. }
  2219. #define RGB_to_Gray(x) ((54 * GetRValue((x)) + 183 * GetGValue((x)) + 19 * GetBValue((x))) >> 8)
  2220. void TrueSaturateBits(void* pvBitmapBits, int Amount, int cx, int cy)
  2221. {
  2222. ULONG* pulSrc = (ULONG*)pvBitmapBits;
  2223. if ((cx > 0) && (cy > 0) && pulSrc)
  2224. {
  2225. for (int i = cx*cy - 1; i >= 0; i--)
  2226. {
  2227. /*
  2228. Enable this if you need true saturation adjustment
  2229. justmann 25-JAN-2001
  2230. ULONG ulR = GetRValue(*pulSrc);
  2231. ULONG ulG = GetGValue(*pulSrc);
  2232. ULONG ulB = GetBValue(*pulSrc);
  2233. ulGray = (54 * ulR + 183 * ulG + 19 * ulB) >> 8;
  2234. ULONG ulTemp = ulGray * (0xff - Amount);
  2235. ulR = (ulR * Amount + ulTemp) >> 8;
  2236. ulG = (ulG * Amount + ulTemp) >> 8;
  2237. ulB = (ulB * Amount + ulTemp) >> 8;
  2238. *pulSrc = (*pulSrc & 0xff000000) | RGB(R, G, B);
  2239. */
  2240. ULONG ulGray = RGB_to_Gray(*pulSrc);
  2241. *pulSrc = (*pulSrc & 0xff000000) | RGB(ulGray, ulGray, ulGray);
  2242. pulSrc++;
  2243. }
  2244. }
  2245. else
  2246. {
  2247. // This should never happen, if it does somebody has a bogus DIB section or does not
  2248. // understand what width or height is!
  2249. ASSERT(0);
  2250. }
  2251. }
  2252. BOOL CImageList::_MaskStretchBlt(BOOL fStretch, int i, HDC hdcDst, int xDst, int yDst, int cxDst, int cyDst,
  2253. HDC hdcImage, int xSrc, int ySrc, int cxSrc, int cySrc,
  2254. int xMask, int yMask,
  2255. DWORD dwRop)
  2256. {
  2257. BOOL fRet = TRUE;
  2258. if (fStretch == FALSE)
  2259. {
  2260. fRet = MaskBlt(hdcDst, xDst, yDst, cxDst, cyDst, hdcImage,
  2261. xSrc, ySrc, _hbmMask, xMask, yMask, dwRop);
  2262. }
  2263. else
  2264. {
  2265. //
  2266. // we have some special cases:
  2267. //
  2268. // if the background color is black, we just do a AND then OR
  2269. // if the background color is white, we just do a OR then AND
  2270. // otherwise change source, then AND then OR
  2271. //
  2272. COLORREF clrTextSave = SetTextColor(hdcDst, CLR_BLACK);
  2273. COLORREF clrBkSave = ::SetBkColor(hdcDst, CLR_WHITE);
  2274. // we cant do white/black special cases if we munged the mask or image
  2275. if (i != -1 && _clrBk == CLR_WHITE)
  2276. {
  2277. StretchBlt(hdcDst, xDst, yDst, cxDst, cyDst, _hdcMask, xMask, yMask, cxSrc, cySrc, ROP_DSno);
  2278. StretchBlt(hdcDst, xDst, yDst, cxDst, cyDst, hdcImage, xSrc, ySrc, cxSrc, cySrc, ROP_DSa);
  2279. }
  2280. else if (i != -1 && (_clrBk == CLR_BLACK || _clrBk == CLR_NONE))
  2281. {
  2282. StretchBlt(hdcDst, xDst, yDst, cxDst, cyDst, _hdcMask, xMask, yMask, cxSrc, cySrc, ROP_DSa);
  2283. StretchBlt(hdcDst, xDst, yDst, cxDst, cyDst, hdcImage, xSrc, ySrc, cxSrc, cySrc, ROP_DSo);
  2284. }
  2285. else
  2286. {
  2287. // black out the source image.
  2288. BitBlt(hdcImage, xSrc, ySrc, cxSrc, cySrc, _hdcMask, xMask, yMask, ROP_DSna);
  2289. StretchBlt(hdcDst, xDst, yDst, cxDst, cyDst, _hdcMask, xMask, yMask, cxSrc, cySrc, ROP_DSa);
  2290. StretchBlt(hdcDst, xDst, yDst, cxDst, cyDst, hdcImage, xSrc, ySrc, cxSrc, cySrc, ROP_DSo);
  2291. if (i != -1)
  2292. _ResetBkColor(i, i, _clrBk);
  2293. }
  2294. SetTextColor(hdcDst, clrTextSave);
  2295. ::SetBkColor(hdcDst, clrBkSave);
  2296. }
  2297. return fRet;
  2298. }
  2299. BOOL CImageList::_StretchBlt(BOOL fStretch, HDC hdc, int x, int y, int cx, int cy,
  2300. HDC hdcSrc, int xs, int ys, int cxs, int cys, DWORD dwRop)
  2301. {
  2302. if (fStretch)
  2303. return StretchBlt(hdc, x, y, cx, cy,
  2304. hdcSrc, xs, ys, cxs, cys, dwRop);
  2305. return BitBlt(hdc, x, y, cx, cy, hdcSrc, xs, ys, dwRop);
  2306. }
  2307. HRESULT CImageList::Draw(IMAGELISTDRAWPARAMS* pimldp)
  2308. {
  2309. RECT rcImage;
  2310. RECT rc;
  2311. HBRUSH hbrT;
  2312. BOOL fImage;
  2313. HDC hdcMaskI;
  2314. HDC hdcImageI;
  2315. int xMask, yMask;
  2316. int xImage, yImage;
  2317. int cxSource, cySource;
  2318. DWORD dwOldStretchBltMode;
  2319. BOOL fStretch;
  2320. BOOL fDPIScale = FALSE;
  2321. IMAGELISTDRAWPARAMS imldp = {0};
  2322. if (pimldp->cbSize != sizeof(IMAGELISTDRAWPARAMS))
  2323. {
  2324. if (pimldp->cbSize == IMAGELISTDRAWPARAMS_V3_SIZE)
  2325. {
  2326. memcpy(&imldp, pimldp, IMAGELISTDRAWPARAMS_V3_SIZE);
  2327. imldp.cbSize = sizeof(IMAGELISTDRAWPARAMS);
  2328. pimldp = &imldp;
  2329. }
  2330. else
  2331. return E_INVALIDARG;
  2332. }
  2333. if (!IsImageListIndex(pimldp->i))
  2334. return E_INVALIDARG;
  2335. //
  2336. // If we need to use the mirrored imagelist, then let's set it.
  2337. //
  2338. if (_pimlMirror &&
  2339. (IS_DC_RTL_MIRRORED(pimldp->hdcDst)))
  2340. {
  2341. return _pimlMirror->Draw(pimldp);
  2342. }
  2343. ENTERCRITICAL;
  2344. dwOldStretchBltMode = SetStretchBltMode(pimldp->hdcDst, COLORONCOLOR);
  2345. GetImageRect(pimldp->i, &rcImage);
  2346. rcImage.left += pimldp->xBitmap;
  2347. rcImage.top += pimldp->yBitmap;
  2348. if (pimldp->rgbBk == CLR_DEFAULT)
  2349. pimldp->rgbBk = _clrBk;
  2350. if (pimldp->rgbBk == CLR_NONE)
  2351. pimldp->fStyle |= ILD_TRANSPARENT;
  2352. if (pimldp->cx == 0)
  2353. pimldp->cx = RECTWIDTH(rcImage);
  2354. if (pimldp->cy == 0)
  2355. pimldp->cy = RECTHEIGHT(rcImage);
  2356. BOOL fImageHasAlpha = (_GetItemFlags(pimldp->i) & ILIF_ALPHA);
  2357. again:
  2358. cxSource = RECTWIDTH(rcImage);
  2359. cySource = RECTHEIGHT(rcImage);
  2360. if (pimldp->cx <= 0 || pimldp->cy <= 0)
  2361. {
  2362. // caller asked to draw no (or negative) pixels; that's easy!
  2363. // Early-out this case so other parts of the drawing
  2364. // don't get confused.
  2365. goto exit;
  2366. }
  2367. if (pimldp->fStyle & ILD_DPISCALE)
  2368. {
  2369. CCDPIScaleX(&pimldp->cx);
  2370. CCDPIScaleY(&pimldp->cy);
  2371. fDPIScale = TRUE;
  2372. }
  2373. fStretch = (pimldp->fStyle & ILD_SCALE) || (fDPIScale);
  2374. if (fStretch)
  2375. {
  2376. dwOldStretchBltMode = SetStretchBltMode(pimldp->hdcDst, HALFTONE);
  2377. }
  2378. hdcMaskI = _hdcMask;
  2379. xMask = rcImage.left;
  2380. yMask = rcImage.top;
  2381. hdcImageI = _hdcImage;
  2382. xImage = rcImage.left;
  2383. yImage = rcImage.top;
  2384. if (pimldp->fStyle & ILD_BLENDMASK)
  2385. {
  2386. // make a copy of the image, because we will have to modify it
  2387. hdcImageI = ImageList_GetWorkDC(pimldp->hdcDst, (_flags & ILC_COLORMASK) == ILC_COLOR32, pimldp->cx, pimldp->cy);
  2388. xImage = 0;
  2389. yImage = 0;
  2390. //
  2391. // blend with the destination
  2392. // by "oring" the mask with a 50% dither mask
  2393. //
  2394. if (pimldp->rgbFg == CLR_NONE && hdcMaskI)
  2395. {
  2396. fImageHasAlpha = FALSE;
  2397. if ((_flags & ILC_COLORMASK) == ILC_COLOR32 &&
  2398. !(pimldp->fStyle & ILD_MASK))
  2399. {
  2400. // copy dest to our work buffer
  2401. _StretchBlt(fStretch, hdcImageI, 0, 0, pimldp->cx, pimldp->cy, pimldp->hdcDst, pimldp->x, pimldp->y, cxSource, cySource, SRCCOPY);
  2402. Blend32(hdcImageI, 0, 0, pimldp->i, pimldp->cx, pimldp->cy, pimldp->rgbFg, pimldp->fStyle);
  2403. }
  2404. else if ((_flags & ILC_COLORMASK) == ILC_COLOR16 &&
  2405. !(pimldp->fStyle & ILD_MASK))
  2406. {
  2407. // copy dest to our work buffer
  2408. _StretchBlt(fStretch, hdcImageI, 0, 0, pimldp->cx, pimldp->cy, pimldp->hdcDst, pimldp->x, pimldp->y, cxSource, cySource, SRCCOPY);
  2409. // blend source into our work buffer
  2410. Blend16(hdcImageI, 0, 0, pimldp->i, pimldp->cx, pimldp->cy, pimldp->rgbFg, pimldp->fStyle);
  2411. pimldp->fStyle |= ILD_TRANSPARENT;
  2412. }
  2413. else
  2414. {
  2415. GetSpareImageRect(&rc);
  2416. xMask = rc.left;
  2417. yMask = rc.top;
  2418. // copy the source image
  2419. _StretchBlt(fStretch, hdcImageI, 0, 0, pimldp->cx, pimldp->cy,
  2420. _hdcImage, rcImage.left, rcImage.top, cxSource, cySource, SRCCOPY);
  2421. // make a dithered copy of the mask
  2422. hbrT = (HBRUSH)SelectObject(hdcMaskI, g_hbrMonoDither);
  2423. _StretchBlt(fStretch, hdcMaskI, rc.left, rc.top, pimldp->cx, pimldp->cy,
  2424. _hdcMask, rcImage.left, rcImage.top, cxSource, cySource, ROP_PSo);
  2425. SelectObject(hdcMaskI, hbrT);
  2426. pimldp->fStyle |= ILD_TRANSPARENT;
  2427. }
  2428. }
  2429. else
  2430. {
  2431. // blend source into our work buffer
  2432. if (Blend(hdcImageI, 0, 0, pimldp->i, pimldp->cx, pimldp->cy, pimldp->rgbFg, pimldp->fStyle))
  2433. {
  2434. fImageHasAlpha = (_flags & ILC_COLORMASK) == ILC_COLOR32;
  2435. }
  2436. }
  2437. }
  2438. // is the source image from the image list (not hdcWork)
  2439. fImage = hdcImageI == _hdcImage;
  2440. if (pimldp->cbSize >= sizeof(IMAGELISTDRAWPARAMS) &&
  2441. pimldp->fState & ILS_GLOW ||
  2442. pimldp->fState & ILS_SHADOW ||
  2443. pimldp->fState & ILS_SATURATE ||
  2444. pimldp->fState & ILS_ALPHA)
  2445. {
  2446. int z;
  2447. ULONG* pvBits;
  2448. HDC hdcMem = CreateCompatibleDC(pimldp->hdcDst);
  2449. HBITMAP hbmpOld;
  2450. HBITMAP hbmp;
  2451. BITMAPINFO bi = {0};
  2452. BLENDFUNCTION bf = {0};
  2453. DWORD dwAlphaAmount = 0x000000ff;
  2454. COLORREF crAlphaColor = pimldp->crEffect; // Need to make this a selectable color
  2455. int x, y;
  2456. int xOffset, yOffset;
  2457. SIZE size = {cxSource, cySource};
  2458. if (hdcMem)
  2459. {
  2460. if (pimldp->fState & ILS_SHADOW)
  2461. {
  2462. x = 5; // This is a "Blur fudge Factor"
  2463. y = 5; //
  2464. xOffset = -(DROP_SHADOW - x);
  2465. yOffset = -(DROP_SHADOW - y);
  2466. size.cx = pimldp->cx + 10;
  2467. size.cy = pimldp->cy + 10;
  2468. dwAlphaAmount = 0x00000050;
  2469. crAlphaColor = RGB(0, 0, 0);
  2470. }
  2471. else if (pimldp->fState & ILS_GLOW)
  2472. {
  2473. xOffset = x = 10;
  2474. yOffset = y = 10;
  2475. size.cx = pimldp->cx + (GLOW_RADIUS * 2);
  2476. size.cy = pimldp->cy + (GLOW_RADIUS * 2);
  2477. }
  2478. else if (pimldp->fState & ILS_ALPHA)
  2479. {
  2480. xOffset = x = 0;
  2481. yOffset = y = 0;
  2482. size.cx = pimldp->cx;
  2483. size.cy = pimldp->cy;
  2484. }
  2485. bi.bmiHeader.biSize = sizeof(bi.bmiHeader);
  2486. bi.bmiHeader.biWidth = size.cx;
  2487. bi.bmiHeader.biHeight = size.cy;
  2488. bi.bmiHeader.biPlanes = 1;
  2489. bi.bmiHeader.biBitCount = 32;
  2490. bi.bmiHeader.biCompression = BI_RGB;
  2491. hbmp = CreateDIBSection(hdcMem, &bi, DIB_RGB_COLORS, (VOID**)&pvBits, NULL, 0);
  2492. if (hbmp)
  2493. {
  2494. hbmpOld = (HBITMAP)SelectObject(hdcMem, hbmp);
  2495. ZeroMemory(pvBits, size.cx * size.cy);
  2496. if (pimldp->fState & ILS_SHADOW || pimldp->fState & ILS_GLOW || pimldp->fState & ILS_ALPHA)
  2497. {
  2498. if (_hbmMask)
  2499. {
  2500. MaskBlt(hdcMem, pimldp->x, pimldp->y, size.cx, size.cy,
  2501. hdcImageI, xImage, yImage, _hbmMask, xMask, yMask, 0xCCAA0000);
  2502. }
  2503. else if (pimldp->fState & ILS_SHADOW)
  2504. {
  2505. RECT rc = {x, y, size.cx, size.cy};
  2506. FillRectClr(hdcMem, &rc, RGB(0x0F, 0x0F, 0x0F)); // White so that it gets inverted into a shadow
  2507. }
  2508. else
  2509. {
  2510. BitBlt(hdcMem, x, y, size.cx, size.cy, hdcImageI, xImage, yImage, SRCCOPY);
  2511. }
  2512. int iTotalSize = size.cx * size.cy;
  2513. if (pimldp->fState & ILS_ALPHA)
  2514. {
  2515. for (z = 0; z < iTotalSize; z++)
  2516. {
  2517. RGBQUAD* prgb = &((RGBQUAD*)pvBits)[z];
  2518. prgb->rgbReserved = (BYTE)(pimldp->Frame & 0xFF);
  2519. prgb->rgbRed = ((prgb->rgbRed * prgb->rgbReserved) + 128) / 255;
  2520. prgb->rgbGreen = ((prgb->rgbGreen * prgb->rgbReserved) + 128) / 255;
  2521. prgb->rgbBlue = ((prgb->rgbBlue * prgb->rgbReserved) + 128) / 255;
  2522. }
  2523. }
  2524. else
  2525. {
  2526. for (z = 0; z < iTotalSize; z++)
  2527. {
  2528. if (((PULONG)pvBits)[z] != 0)
  2529. ((PULONG)pvBits)[z] = dwAlphaAmount;
  2530. }
  2531. BlurBitmap(pvBits, size.cx, size.cy, crAlphaColor);
  2532. if (!(pimldp->fState & ILS_SHADOW))
  2533. {
  2534. for (z = 0; z < iTotalSize; z++)
  2535. {
  2536. if (((PULONG)pvBits)[z] > 0x09000000)
  2537. ((PULONG)pvBits)[z] = dwAlphaAmount;
  2538. }
  2539. BlurBitmap(pvBits, size.cx, size.cy, crAlphaColor);
  2540. BlurBitmap(pvBits, size.cx, size.cy, crAlphaColor);
  2541. }
  2542. }
  2543. bf.BlendOp = AC_SRC_OVER;
  2544. bf.SourceConstantAlpha = 255;
  2545. bf.AlphaFormat = AC_SRC_ALPHA;
  2546. bf.BlendFlags = fDPIScale?AC_USE_HIGHQUALITYFILTER:0;
  2547. // Do not mirror the bitmap. By this point it is correctly mirrored
  2548. GdiAlphaBlend(pimldp->hdcDst, pimldp->x - xOffset, pimldp->y - yOffset, pimldp->cx, pimldp->cy,
  2549. hdcMem, 0, 0, size.cx, size.cy, bf);
  2550. }
  2551. else
  2552. {
  2553. BitBlt(hdcMem, 0, 0, pimldp->cx, pimldp->cy, hdcImageI, xImage, yImage, SRCCOPY);
  2554. TrueSaturateBits(pvBits, pimldp->Frame, size.cx, size.cy);
  2555. if (fImageHasAlpha)
  2556. {
  2557. bf.BlendOp = AC_SRC_OVER;
  2558. bf.SourceConstantAlpha = 150;
  2559. bf.AlphaFormat = AC_SRC_ALPHA;
  2560. // Do not mirror the bitmap. By this point it is correctly mirrored
  2561. GdiAlphaBlend(pimldp->hdcDst, pimldp->x, pimldp->y, pimldp->cx, pimldp->cy, hdcMem, 0, 0, cxSource, cySource, bf);
  2562. }
  2563. else if (_hbmMask)
  2564. {
  2565. _MaskStretchBlt(fStretch, -1, hdcMem, pimldp->x, pimldp->y, pimldp->cx, pimldp->cy,
  2566. hdcMem, 0, 0, cxSource, cySource, xMask, yMask, 0xCCAA0000);
  2567. }
  2568. else
  2569. {
  2570. _StretchBlt(fStretch, pimldp->hdcDst, pimldp->x, pimldp->y, pimldp->cx, pimldp->cy, hdcMem, 0, 0, cxSource,cySource, SRCCOPY);
  2571. }
  2572. }
  2573. SelectObject(hdcMem, hbmpOld);
  2574. DeleteObject(hbmp);
  2575. pimldp->fStyle |= ILD_TRANSPARENT;
  2576. }
  2577. DeleteDC(hdcMem);
  2578. }
  2579. if (pimldp->fState & ILS_SHADOW || pimldp->fState & ILS_GLOW)
  2580. {
  2581. if (pimldp->fState & ILS_SHADOW)
  2582. {
  2583. pimldp->fState &= ~ILS_SHADOW;
  2584. //pimldp->x -= DROP_SHADOW;
  2585. //pimldp->y -= DROP_SHADOW;
  2586. }
  2587. else
  2588. {
  2589. pimldp->fState &= ~ILS_GLOW;
  2590. }
  2591. goto again;
  2592. }
  2593. }
  2594. else if ((pimldp->fStyle & ILD_MASK) && hdcMaskI)
  2595. {
  2596. //
  2597. // ILD_MASK means draw the mask only
  2598. //
  2599. DWORD dwRop;
  2600. ASSERT(GetTextColor(pimldp->hdcDst) == CLR_BLACK);
  2601. ASSERT(::GetBkColor(pimldp->hdcDst) == CLR_WHITE);
  2602. if (pimldp->fStyle & ILD_ROP)
  2603. dwRop = pimldp->dwRop;
  2604. else if (pimldp->fStyle & ILD_TRANSPARENT)
  2605. dwRop = SRCAND;
  2606. else
  2607. dwRop = SRCCOPY;
  2608. _StretchBlt(fStretch, pimldp->hdcDst, pimldp->x, pimldp->y, pimldp->cx, pimldp->cy, hdcMaskI, xMask, yMask, cxSource, cySource, dwRop);
  2609. }
  2610. else if (fImageHasAlpha && // this image has alpha
  2611. !(pimldp->fStyle & ILD_PRESERVEALPHA)) // But not if we're trying to preserve it.
  2612. {
  2613. if (!(pimldp->fStyle & ILD_TRANSPARENT))
  2614. {
  2615. COLORREF clr = pimldp->rgbBk;
  2616. if (clr == CLR_DEFAULT)
  2617. clr = _clrBk;
  2618. RECT rc = {pimldp->x, pimldp->y, pimldp->x + pimldp->cx, pimldp->y + pimldp->cy};
  2619. FillRectClr(pimldp->hdcDst, &rc, clr);
  2620. }
  2621. BLENDFUNCTION bf = {0};
  2622. bf.BlendOp = AC_SRC_OVER;
  2623. bf.SourceConstantAlpha = 255;
  2624. bf.AlphaFormat = AC_SRC_ALPHA;
  2625. bf.BlendFlags = AC_MIRRORBITMAP | (fDPIScale?AC_USE_HIGHQUALITYFILTER:0);
  2626. GdiAlphaBlend(pimldp->hdcDst, pimldp->x, pimldp->y, pimldp->cx, pimldp->cy, hdcImageI, xImage, yImage, cxSource, cySource, bf);
  2627. }
  2628. else if (pimldp->fStyle & ILD_IMAGE)
  2629. {
  2630. COLORREF clrBk = ::GetBkColor(hdcImageI);
  2631. DWORD dwRop;
  2632. if (pimldp->rgbBk != CLR_DEFAULT)
  2633. {
  2634. ::SetBkColor(hdcImageI, pimldp->rgbBk);
  2635. }
  2636. if (pimldp->fStyle & ILD_ROP)
  2637. dwRop = pimldp->dwRop;
  2638. else
  2639. dwRop = SRCCOPY;
  2640. _StretchBlt(fStretch, pimldp->hdcDst, pimldp->x, pimldp->y, pimldp->cx, pimldp->cy, hdcImageI, xImage, yImage, cxSource, cySource, dwRop);
  2641. ::SetBkColor(hdcImageI, clrBk);
  2642. }
  2643. else if ((pimldp->fStyle & ILD_TRANSPARENT) && hdcMaskI)
  2644. {
  2645. _MaskStretchBlt(fStretch, fImage?pimldp->i:-1,pimldp->hdcDst, pimldp->x, pimldp->y, pimldp->cx, pimldp->cy,
  2646. hdcImageI, xImage, yImage, cxSource, cySource, xMask, yMask, 0xCCAA0000);
  2647. }
  2648. else if (fImage && pimldp->rgbBk == _clrBk && _fSolidBk)
  2649. {
  2650. _StretchBlt(fStretch, pimldp->hdcDst, pimldp->x, pimldp->y, pimldp->cx, pimldp->cy, hdcImageI, xImage, yImage, cxSource, cySource, SRCCOPY);
  2651. }
  2652. else if (hdcMaskI && !fImageHasAlpha)
  2653. {
  2654. if (fImage &&
  2655. ((pimldp->rgbBk == _clrBk &&
  2656. !_fSolidBk) ||
  2657. GetNearestColor32(hdcImageI, pimldp->rgbBk) != pimldp->rgbBk))
  2658. {
  2659. // make a copy of the image, because we will have to modify it
  2660. hdcImageI = ImageList_GetWorkDC(pimldp->hdcDst, (_flags & ILC_COLORMASK) == ILC_COLOR32, pimldp->cx, pimldp->cy);
  2661. xImage = 0;
  2662. yImage = 0;
  2663. fImage = FALSE;
  2664. BitBlt(hdcImageI, 0, 0, pimldp->cx, pimldp->cy, _hdcImage, rcImage.left, rcImage.top, SRCCOPY);
  2665. }
  2666. SetBrushOrgEx(hdcImageI, xImage-pimldp->x, yImage-pimldp->y, NULL);
  2667. hbrT = SelectBrush(hdcImageI, CreateSolidBrush(pimldp->rgbBk));
  2668. BitBlt(hdcImageI, xImage, yImage, pimldp->cx, pimldp->cy, hdcMaskI, xMask, yMask, ROP_PatMask);
  2669. DeleteObject(SelectBrush(hdcImageI, hbrT));
  2670. SetBrushOrgEx(hdcImageI, 0, 0, NULL);
  2671. _StretchBlt(fStretch, pimldp->hdcDst, pimldp->x, pimldp->y, pimldp->cx, pimldp->cy, hdcImageI, xImage, yImage, cxSource, cySource, SRCCOPY);
  2672. if (fImage)
  2673. _ResetBkColor(pimldp->i, pimldp->i, _clrBk);
  2674. }
  2675. else
  2676. {
  2677. _StretchBlt(fStretch, pimldp->hdcDst, pimldp->x, pimldp->y, pimldp->cx, pimldp->cy, hdcImageI, xImage, yImage, cxSource, cySource, SRCCOPY);
  2678. }
  2679. //
  2680. // now deal with a overlay image, use the minimal bounding rect (and flags)
  2681. // we computed in ImageList_SetOverlayImage()
  2682. //
  2683. if (pimldp->fStyle & ILD_OVERLAYMASK)
  2684. {
  2685. int n = OVERLAYMASKTOINDEX(pimldp->fStyle);
  2686. if (n < NUM_OVERLAY_IMAGES)
  2687. {
  2688. pimldp->i = _aOverlayIndexes[n];
  2689. if (!fImageHasAlpha)
  2690. pimldp->fStyle &= ~ILD_PRESERVEALPHA;
  2691. if (pimldp->fStyle & ILD_PRESERVEALPHA &&
  2692. !(_GetItemFlags(pimldp->i) & ILIF_ALPHA))
  2693. {
  2694. _GenerateAlphaForImageUsingMask(pimldp->i, FALSE);
  2695. }
  2696. fImageHasAlpha = (_GetItemFlags(pimldp->i) & ILIF_ALPHA);
  2697. GetImageRect(pimldp->i, &rcImage);
  2698. int xOverlay = _aOverlayX[n];
  2699. int yOverlay = _aOverlayY[n];
  2700. int cxOverlay = _aOverlayDX[n];
  2701. int cyOverlay = _aOverlayDY[n];
  2702. if (fDPIScale)
  2703. {
  2704. CCDPIScaleX(&xOverlay );
  2705. CCDPIScaleY(&yOverlay );
  2706. }
  2707. pimldp->cx = cxOverlay;
  2708. pimldp->cy = cyOverlay;
  2709. pimldp->x += xOverlay;
  2710. pimldp->y += yOverlay;
  2711. rcImage.left += _aOverlayX[n] + pimldp->xBitmap;
  2712. rcImage.top += _aOverlayY[n] + pimldp->yBitmap;
  2713. rcImage.right = rcImage.left + _aOverlayDX[n];
  2714. rcImage.bottom = rcImage.top + _aOverlayDY[n];
  2715. pimldp->fStyle &= ILD_MASK;
  2716. pimldp->fStyle |= ILD_TRANSPARENT;
  2717. pimldp->fStyle |= (fDPIScale?ILD_DPISCALE:0);
  2718. pimldp->fStyle |= _aOverlayF[n];
  2719. if (fImageHasAlpha)
  2720. pimldp->fStyle &= ~(ILD_IMAGE);
  2721. if (pimldp->cx > 0 && pimldp->cy > 0)
  2722. goto again;
  2723. }
  2724. }
  2725. if (!fImage)
  2726. {
  2727. ImageList_ReleaseWorkDC(hdcImageI);
  2728. }
  2729. exit:
  2730. SetStretchBltMode(pimldp->hdcDst, dwOldStretchBltMode);
  2731. LEAVECRITICAL;
  2732. return S_OK;
  2733. }
  2734. HRESULT CImageList::GetImageInfo(int i, IMAGEINFO * pImageInfo)
  2735. {
  2736. RIPMSG(pImageInfo != NULL, "ImageList_GetImageInfo: Invalid NULL pointer");
  2737. RIPMSG(IsImageListIndex(i), "ImageList_GetImageInfo: Invalid image index %d", i);
  2738. if (!pImageInfo || !IsImageListIndex(i))
  2739. return E_POINTER;
  2740. pImageInfo->hbmImage = _hbmImage;
  2741. pImageInfo->hbmMask = _hbmMask;
  2742. return GetImageRect(i, &pImageInfo->rcImage);
  2743. }
  2744. //
  2745. // Parameter:
  2746. // i -- -1 to add
  2747. //
  2748. HRESULT CImageList::_ReplaceIcon(int i, HICON hIcon, int* pi)
  2749. {
  2750. HICON hIconT = hIcon;
  2751. RECT rc;
  2752. HRESULT hr = S_OK;
  2753. TraceMsg(TF_IMAGELIST, "ImageList_ReplaceIcon");
  2754. *pi = -1;
  2755. // be win95 compatible
  2756. if (i < -1)
  2757. return E_INVALIDARG;
  2758. //
  2759. // re-size the icon (iff needed) by calling CopyImage
  2760. //
  2761. hIcon = (HICON)CopyImage(hIconT, IMAGE_ICON, _cx, _cy, LR_COPYFROMRESOURCE | LR_COPYRETURNORG);
  2762. if (hIcon == NULL)
  2763. return E_OUTOFMEMORY;
  2764. //
  2765. // alocate a slot for the icon
  2766. //
  2767. if (i == -1)
  2768. hr = _Add(NULL,NULL,1,0,0,&i);
  2769. if (i == -1)
  2770. goto exit;
  2771. ENTERCRITICAL;
  2772. //
  2773. // now draw it into the image bitmaps
  2774. //
  2775. hr = GetImageRect(i, &rc);
  2776. if (FAILED(hr))
  2777. goto LeaveCritical;
  2778. if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  2779. {
  2780. BOOL fSuccess = FALSE;
  2781. ICONINFO io;
  2782. if (GetIconInfo(hIcon, &io))
  2783. {
  2784. BITMAP bm;
  2785. if (GetObject(io.hbmColor, sizeof(bm), &bm))
  2786. {
  2787. if (bm.bmBitsPixel == 32)
  2788. {
  2789. HDC h = CreateCompatibleDC(_hdcImage);
  2790. if (h)
  2791. {
  2792. HBITMAP hbmpOld = (HBITMAP)SelectObject(h, io.hbmColor);
  2793. BitBlt(_hdcImage, rc.left, rc.top, RECTWIDTH(rc), RECTHEIGHT(rc), h, 0, 0, SRCCOPY);
  2794. if (_HasAlpha(i))
  2795. {
  2796. SetItemFlags(i, ILIF_ALPHA);
  2797. _PreProcessImage(i);
  2798. fSuccess = TRUE;
  2799. }
  2800. SelectObject(h, hbmpOld);
  2801. DeleteDC(h);
  2802. }
  2803. }
  2804. }
  2805. DeleteObject(io.hbmColor);
  2806. DeleteObject(io.hbmMask);
  2807. }
  2808. if (!fSuccess)
  2809. {
  2810. // If it doesn't have alpha or we can't get info
  2811. SetItemFlags(i, 0);
  2812. }
  2813. }
  2814. if (_GetItemFlags(i) == 0)
  2815. {
  2816. FillRect(_hdcImage, &rc, _hbrBk);
  2817. DrawIconEx(_hdcImage, rc.left, rc.top, hIcon, 0, 0, 0, NULL, DI_NORMAL);
  2818. }
  2819. if (_hdcMask)
  2820. DrawIconEx(_hdcMask, rc.left, rc.top, hIcon, 0, 0, 0, NULL, DI_MASK);
  2821. hr = S_OK;
  2822. *pi = i;
  2823. LeaveCritical:
  2824. LEAVECRITICAL;
  2825. exit:
  2826. //
  2827. // if we had user size a new icon, delete it.
  2828. //
  2829. if (hIcon != hIconT)
  2830. DestroyIcon(hIcon);
  2831. return hr;
  2832. }
  2833. HRESULT CImageList::ReplaceIcon(int i, HICON hIcon, int* pi)
  2834. {
  2835. // Let's add it first to the mirrored image list, if one exists
  2836. if (_pimlMirror)
  2837. {
  2838. HICON hIconT = CopyIcon(hIcon);
  2839. if (hIconT)
  2840. {
  2841. MirrorIcon(&hIconT, NULL);
  2842. _pimlMirror->_ReplaceIcon(i, hIconT, pi);
  2843. DestroyIcon(hIconT);
  2844. }
  2845. }
  2846. return _ReplaceIcon(i, hIcon,pi);
  2847. }
  2848. // make a dithered copy of the source image in the destination image.
  2849. // allows placing of the final image in the destination.
  2850. HRESULT CImageList::CopyDitherImage(WORD iDst, int xDst, int yDst, IUnknown* punkSrc, int iSrc, UINT fStyle)
  2851. {
  2852. IImageList* pux;
  2853. HRESULT hr = punkSrc->QueryInterface(IID_PPV_ARG(IImageList, &pux));
  2854. if (FAILED(hr))
  2855. return hr;
  2856. RECT rc;
  2857. int x, y;
  2858. GetImageRect(iDst, &rc);
  2859. // coordinates in destination image list
  2860. x = xDst + rc.left;
  2861. y = yDst + rc.top;
  2862. fStyle &= ILD_OVERLAYMASK;
  2863. WimpyDrawEx(pux, iSrc, _hdcImage, x, y, 0, 0, CLR_DEFAULT, CLR_NONE, ILD_IMAGE | fStyle);
  2864. //
  2865. // dont dither the mask on a hicolor device, we will draw the image
  2866. // with blending while dragging.
  2867. //
  2868. if (_hdcMask && GetScreenDepth() > 8)
  2869. {
  2870. WimpyDrawEx(pux, iSrc, _hdcMask, x, y, 0, 0, CLR_NONE, CLR_NONE, ILD_MASK | fStyle);
  2871. }
  2872. else if (_hdcMask)
  2873. {
  2874. WimpyDrawEx(pux, iSrc, _hdcMask, x, y, 0, 0, CLR_NONE, CLR_NONE, ILD_BLEND50|ILD_MASK | fStyle);
  2875. }
  2876. if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  2877. {
  2878. SetItemFlags(iDst, _HasAlpha(iDst)?ILIF_ALPHA:0);
  2879. }
  2880. _ResetBkColor(iDst, iDst+1, _clrBk);
  2881. pux->Release();
  2882. return hr;
  2883. }
  2884. //
  2885. // ImageList_CopyBitmap
  2886. //
  2887. // Worker function for ImageList_Duplicate.
  2888. //
  2889. // Given a bitmap and an hdc, creates and returns a copy of the passed in bitmap.
  2890. //
  2891. HBITMAP CImageList::_CopyBitmap(HBITMAP hbm, HDC hdc)
  2892. {
  2893. ASSERT(hbm);
  2894. BITMAP bm;
  2895. HBITMAP hbmCopy = NULL;
  2896. if (GetObject(hbm, sizeof(bm), &bm) == sizeof(bm))
  2897. {
  2898. ENTERCRITICAL;
  2899. if (hbmCopy = CreateCompatibleBitmap(hdc, bm.bmWidth, bm.bmHeight))
  2900. {
  2901. CImageList::SelectDstBitmap(hbmCopy);
  2902. BitBlt(g_hdcDst, 0, 0, bm.bmWidth, bm.bmHeight,
  2903. hdc, 0, 0, SRCCOPY);
  2904. CImageList::SelectDstBitmap(NULL);
  2905. }
  2906. LEAVECRITICAL;
  2907. }
  2908. return hbmCopy;
  2909. }
  2910. HBITMAP CImageList::_CopyDIBBitmap(HBITMAP hbm, HDC hdc, RGBQUAD** ppargb)
  2911. {
  2912. ASSERT(hbm);
  2913. BITMAP bm;
  2914. HBITMAP hbmCopy = NULL;
  2915. if (GetObject(hbm, sizeof(bm), &bm) == sizeof(bm))
  2916. {
  2917. ENTERCRITICAL;
  2918. hbmCopy = _CreateBitmap(bm.bmWidth, bm.bmHeight, ppargb);
  2919. if (hbmCopy)
  2920. {
  2921. CImageList::SelectDstBitmap(hbmCopy);
  2922. BitBlt(g_hdcDst, 0, 0, bm.bmWidth, bm.bmHeight,
  2923. hdc, 0, 0, SRCCOPY);
  2924. CImageList::SelectDstBitmap(NULL);
  2925. }
  2926. LEAVECRITICAL;
  2927. }
  2928. return hbmCopy;
  2929. }
  2930. HRESULT CImageList::Clone(REFIID riid, void** ppv)
  2931. {
  2932. HBITMAP hbmImageI;
  2933. HBITMAP hbmMaskI = NULL;
  2934. RGBQUAD* pargbImageI;
  2935. HDSA dsaFlags = NULL;
  2936. HRESULT hr = S_OK;
  2937. CImageList* pimlCopy = NULL;
  2938. *ppv = NULL;
  2939. ENTERCRITICAL;
  2940. hbmImageI = _CopyDIBBitmap(_hbmImage, _hdcImage, &pargbImageI);
  2941. if (!hbmImageI)
  2942. hr = E_OUTOFMEMORY;
  2943. if (SUCCEEDED(hr))
  2944. {
  2945. if (_hdcMask)
  2946. {
  2947. hbmMaskI = _CopyBitmap(_hbmMask, _hdcMask);
  2948. if (!hbmMaskI)
  2949. hr = E_OUTOFMEMORY;
  2950. }
  2951. if (SUCCEEDED(hr) && (_flags & ILC_COLORMASK) == ILC_COLOR32)
  2952. {
  2953. dsaFlags = DSA_Create(sizeof(DWORD), _cGrow);
  2954. if (dsaFlags)
  2955. {
  2956. DWORD dw;
  2957. for (int i = 0; i < _cImage; i++)
  2958. {
  2959. DSA_GetItem(_dsaFlags, i, &dw);
  2960. if (!DSA_SetItem(dsaFlags, i, &dw))
  2961. {
  2962. hr = E_OUTOFMEMORY;
  2963. break;
  2964. }
  2965. }
  2966. }
  2967. }
  2968. if (SUCCEEDED(hr))
  2969. {
  2970. pimlCopy = CImageList::Create(_cx, _cy, _flags, 0, _cGrow);
  2971. if (pimlCopy)
  2972. {
  2973. // Slam in our bitmap copies and delete the old ones
  2974. SelectObject(pimlCopy->_hdcImage, hbmImageI);
  2975. CImageList::_DeleteBitmap(pimlCopy->_hbmImage);
  2976. if (pimlCopy->_hdcMask)
  2977. {
  2978. SelectObject(pimlCopy->_hdcMask, hbmMaskI);
  2979. CImageList::_DeleteBitmap(pimlCopy->_hbmMask);
  2980. }
  2981. if (pimlCopy->_dsaFlags)
  2982. DSA_Destroy(pimlCopy->_dsaFlags);
  2983. pimlCopy->_dsaFlags = dsaFlags;
  2984. pimlCopy->_hbmImage = hbmImageI;
  2985. pimlCopy->_pargbImage = pargbImageI;
  2986. pimlCopy->_hbmMask = hbmMaskI;
  2987. // Make sure other info is correct
  2988. pimlCopy->_cImage = _cImage;
  2989. pimlCopy->_cAlloc = _cAlloc;
  2990. pimlCopy->_cStrip = _cStrip;
  2991. pimlCopy->_clrBlend = _clrBlend;
  2992. pimlCopy->_clrBk = _clrBk;
  2993. // Delete the old brush and create the correct one
  2994. if (pimlCopy->_hbrBk)
  2995. DeleteObject(pimlCopy->_hbrBk);
  2996. if (pimlCopy->_clrBk == CLR_NONE)
  2997. {
  2998. pimlCopy->_hbrBk = (HBRUSH)GetStockObject(BLACK_BRUSH);
  2999. pimlCopy->_fSolidBk = TRUE;
  3000. }
  3001. else
  3002. {
  3003. pimlCopy->_hbrBk = CreateSolidBrush(pimlCopy->_clrBk);
  3004. pimlCopy->_fSolidBk = GetNearestColor32(pimlCopy->_hdcImage, pimlCopy->_clrBk) == pimlCopy->_clrBk;
  3005. }
  3006. }
  3007. }
  3008. LEAVECRITICAL;
  3009. }
  3010. if (FAILED(hr))
  3011. {
  3012. if (hbmImageI)
  3013. CImageList::_DeleteBitmap(hbmImageI);
  3014. if (hbmMaskI)
  3015. CImageList::_DeleteBitmap(hbmMaskI);
  3016. if (dsaFlags)
  3017. DSA_Destroy(dsaFlags);
  3018. }
  3019. if (pimlCopy)
  3020. {
  3021. hr = pimlCopy->QueryInterface(riid, ppv);
  3022. pimlCopy->Release();
  3023. }
  3024. return hr;
  3025. }
  3026. void CImageList::_Merge(IImageList* pux, int i, int dx, int dy)
  3027. {
  3028. if (_hdcMask)
  3029. {
  3030. IImageListPriv* puxp;
  3031. if (SUCCEEDED(pux->QueryInterface(IID_PPV_ARG(IImageListPriv, &puxp))))
  3032. {
  3033. HDC hdcMaskI;
  3034. if (SUCCEEDED(puxp->GetPrivateGoo(NULL, NULL, NULL, &hdcMaskI)) && hdcMaskI)
  3035. {
  3036. RECT rcMerge;
  3037. int cxI, cyI;
  3038. pux->GetIconSize(&cxI, &cyI);
  3039. UINT uFlags = 0;
  3040. puxp->GetFlags(&uFlags);
  3041. pux->GetImageRect(i, &rcMerge);
  3042. BitBlt(_hdcMask, dx, dy, cxI, cyI,
  3043. hdcMaskI, rcMerge.left, rcMerge.top, SRCAND);
  3044. }
  3045. puxp->Release();
  3046. }
  3047. }
  3048. WimpyDraw(pux, i, _hdcImage, dx, dy, ILD_TRANSPARENT | ILD_PRESERVEALPHA);
  3049. if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  3050. SetItemFlags(i, _HasAlpha(i)? ILIF_ALPHA : 0);
  3051. }
  3052. HRESULT CImageList::_Merge(int i1, IUnknown* punk, int i2, int dx, int dy, CImageList** ppiml)
  3053. {
  3054. CImageList* pimlNew = NULL;
  3055. IImageListPriv* puxp;
  3056. HRESULT hr = punk->QueryInterface(IID_PPV_ARG(IImageListPriv, &puxp));
  3057. if (SUCCEEDED(hr))
  3058. {
  3059. IImageList* pux;
  3060. hr = punk->QueryInterface(IID_PPV_ARG(IImageList, &pux));
  3061. if (SUCCEEDED(hr))
  3062. {
  3063. RECT rcNew;
  3064. RECT rc1;
  3065. RECT rc2;
  3066. int cxI, cyI;
  3067. int c1, c2;
  3068. UINT wFlags;
  3069. UINT uSrcFlags;
  3070. puxp->GetFlags(&uSrcFlags);
  3071. pux->GetIconSize(&cxI, &cyI);
  3072. ENTERCRITICAL;
  3073. SetRect(&rc1, 0, 0, _cx, _cy);
  3074. SetRect(&rc2, dx, dy, cxI + dx, cyI + dy);
  3075. UnionRect(&rcNew, &rc1, &rc2);
  3076. cxI = RECTWIDTH(rcNew);
  3077. cyI = RECTHEIGHT(rcNew);
  3078. //
  3079. // If one of images are shared, create a shared image.
  3080. //
  3081. wFlags = (_flags | uSrcFlags) & ~ILC_COLORMASK;
  3082. c1 = (_flags & ILC_COLORMASK);
  3083. c2 = (uSrcFlags & ILC_COLORMASK);
  3084. if ((c1 == 16 || c1 == 32) && c2 == ILC_COLORDDB)
  3085. {
  3086. c2 = c1;
  3087. }
  3088. wFlags |= max(c1,c2);
  3089. pimlNew = CImageList::Create(cxI, cyI, ILC_MASK|wFlags, 1, 0);
  3090. if (pimlNew)
  3091. {
  3092. pimlNew->_cImage++;
  3093. if (pimlNew->_hdcMask)
  3094. PatBlt(pimlNew->_hdcMask, 0, 0, cxI, cyI, WHITENESS);
  3095. PatBlt(pimlNew->_hdcImage, 0, 0, cxI, cyI, BLACKNESS);
  3096. pimlNew->_Merge(SAFECAST(this, IImageList*), i1, rc1.left - rcNew.left, rc1.top - rcNew.top);
  3097. pimlNew->_Merge(pux, i2, rc2.left - rcNew.left, rc2.top - rcNew.top);
  3098. }
  3099. else
  3100. hr = E_OUTOFMEMORY;
  3101. LEAVECRITICAL;
  3102. pux->Release();
  3103. }
  3104. puxp->Release();
  3105. }
  3106. *ppiml = pimlNew;
  3107. return hr;
  3108. }
  3109. HRESULT CImageList::Merge(int i1, IUnknown* punk, int i2, int dx, int dy, REFIID riid, void** ppv)
  3110. {
  3111. CImageList* piml;
  3112. HRESULT hr = _Merge(i1, punk, i2, dx, dy, &piml);
  3113. if (piml)
  3114. {
  3115. hr = piml->QueryInterface(riid, ppv);
  3116. piml->Release();
  3117. }
  3118. return hr;
  3119. }
  3120. HRESULT CImageList::GetImageRectInverted(int i, RECT * prcImage)
  3121. {
  3122. int x, y;
  3123. ASSERT(prcImage);
  3124. ASSERT(_cStrip == 1); // If not, modify below to accomodate
  3125. if (!prcImage || !IsImageListIndex(i))
  3126. return E_FAIL;
  3127. x = 0;
  3128. y = (_cy * _cAlloc) - (_cy * i) - _cy;
  3129. SetRect(prcImage, x, y, x + _cx, y + _cy);
  3130. return S_OK;
  3131. }
  3132. HRESULT CImageList::GetImageRect(int i, RECT * prcImage)
  3133. {
  3134. int x, y;
  3135. ASSERT(prcImage);
  3136. if (!prcImage || !IsImageListIndex(i))
  3137. return E_FAIL;
  3138. x = _cx * (i % _cStrip);
  3139. y = _cy * (i / _cStrip);
  3140. SetRect(prcImage, x, y, x + _cx, y + _cy);
  3141. return S_OK;
  3142. }
  3143. BOOL CImageList::GetSpareImageRect(RECT * prcImage)
  3144. {
  3145. BOOL fRet = FALSE;
  3146. if (_cImage < _cAlloc)
  3147. {
  3148. // special hacking to use the one scratch image at tail of list :)
  3149. _cImage++;
  3150. fRet = (S_OK == GetImageRect(_cImage-1, prcImage));
  3151. _cImage--;
  3152. }
  3153. return fRet;
  3154. }
  3155. BOOL CImageList::GetSpareImageRectInverted(RECT * prcImage)
  3156. {
  3157. BOOL fRet = FALSE;
  3158. if (_cImage < _cAlloc)
  3159. {
  3160. // special hacking to use the one scratch image at tail of list :)
  3161. _cImage++;
  3162. fRet = (S_OK == GetImageRectInverted(_cImage-1, prcImage));
  3163. _cImage--;
  3164. }
  3165. return fRet;
  3166. }
  3167. // Drag Drop
  3168. // copy an image from one imagelist to another at x,y within iDst in pimlDst.
  3169. // pimlDst's image size should be larger than pimlSrc
  3170. void CImageList::_CopyOneImage(int iDst, int x, int y, CImageList* piml, int iSrc)
  3171. {
  3172. RECT rcSrc, rcDst;
  3173. piml->GetImageRect(iSrc, &rcSrc);
  3174. GetImageRect(iDst, &rcDst);
  3175. if (piml->_hdcMask && _hdcMask)
  3176. {
  3177. BitBlt(_hdcMask, rcDst.left + x, rcDst.top + y, piml->_cx, piml->_cy,
  3178. piml->_hdcMask, rcSrc.left, rcSrc.top, SRCCOPY);
  3179. }
  3180. BitBlt(_hdcImage, rcDst.left + x, rcDst.top + y, piml->_cx, piml->_cy,
  3181. piml->_hdcImage, rcSrc.left, rcSrc.top, SRCCOPY);
  3182. if ((_flags & ILC_COLORMASK) == ILC_COLOR32)
  3183. SetItemFlags(iDst, _HasAlpha(iDst)? ILIF_ALPHA : 0);
  3184. }
  3185. //
  3186. // Cached bitmaps that we use during drag&drop. We re-use those bitmaps
  3187. // across multiple drag session as far as the image size is the same.
  3188. //
  3189. struct DRAGRESTOREBMP
  3190. {
  3191. int BitsPixel;
  3192. HBITMAP hbmOffScreen;
  3193. HBITMAP hbmRestore;
  3194. SIZE sizeRestore;
  3195. }
  3196. g_drb =
  3197. {
  3198. 0, NULL, NULL, {-1,-1}
  3199. };
  3200. BOOL CImageList::CreateDragBitmaps()
  3201. {
  3202. HDC hdc;
  3203. hdc = GetDC(NULL);
  3204. if (_cx != g_drb.sizeRestore.cx ||
  3205. _cy != g_drb.sizeRestore.cy ||
  3206. GetDeviceCaps(hdc, BITSPIXEL) != g_drb.BitsPixel)
  3207. {
  3208. ImageList_DeleteDragBitmaps();
  3209. g_drb.BitsPixel = GetDeviceCaps(hdc, BITSPIXEL);
  3210. g_drb.sizeRestore.cx = _cx;
  3211. g_drb.sizeRestore.cy = _cy;
  3212. g_drb.hbmRestore = CreateColorBitmap(g_drb.sizeRestore.cx, g_drb.sizeRestore.cy);
  3213. g_drb.hbmOffScreen = CreateColorBitmap(g_drb.sizeRestore.cx * 2 - 1, g_drb.sizeRestore.cy * 2 - 1);
  3214. if (!g_drb.hbmRestore || !g_drb.hbmOffScreen)
  3215. {
  3216. ImageList_DeleteDragBitmaps();
  3217. ReleaseDC(NULL, hdc);
  3218. return FALSE;
  3219. }
  3220. }
  3221. ReleaseDC(NULL, hdc);
  3222. return TRUE;
  3223. }
  3224. void ImageList_DeleteDragBitmaps()
  3225. {
  3226. if (g_drb.hbmRestore)
  3227. {
  3228. CImageList::_DeleteBitmap(g_drb.hbmRestore);
  3229. g_drb.hbmRestore = NULL;
  3230. }
  3231. if (g_drb.hbmOffScreen)
  3232. {
  3233. CImageList::_DeleteBitmap(g_drb.hbmOffScreen);
  3234. g_drb.hbmOffScreen = NULL;
  3235. }
  3236. g_drb.sizeRestore.cx = -1;
  3237. g_drb.sizeRestore.cy = -1;
  3238. }
  3239. //
  3240. // Drag context. We don't reuse none of them across two different
  3241. // drag sessions. I'm planning to allocate it for each session
  3242. // to minimize critical sections.
  3243. //
  3244. struct DRAGCONTEXT
  3245. {
  3246. CImageList* pimlDrag; // Image to be drawin while dragging
  3247. IImageList* puxCursor; // Overlap cursor image
  3248. CImageList* pimlDither; // Dithered image
  3249. IImageList* puxDragImage; // The context of the drag.
  3250. int iCursor; // Image index of the cursor
  3251. POINT ptDrag; // current drag position (hwndDC coords)
  3252. POINT ptDragHotspot;
  3253. POINT ptCursor;
  3254. BOOL fDragShow;
  3255. BOOL fHiColor;
  3256. HWND hwndDC;
  3257. }
  3258. g_dctx =
  3259. {
  3260. (CImageList*)NULL, (CImageList*)NULL, (CImageList*)NULL, (IImageList*)NULL,
  3261. -1,
  3262. {0, 0}, {0, 0}, {0, 0},
  3263. FALSE,
  3264. FALSE,
  3265. (HWND)NULL
  3266. };
  3267. HDC ImageList_GetDragDC()
  3268. {
  3269. HDC hdc = GetDCEx(g_dctx.hwndDC, NULL, DCX_WINDOW | DCX_CACHE | DCX_LOCKWINDOWUPDATE);
  3270. //
  3271. // If hdc is mirrored then mirror the 2 globals DCs.
  3272. //
  3273. if (IS_DC_RTL_MIRRORED(hdc))
  3274. {
  3275. SET_DC_RTL_MIRRORED(g_hdcDst);
  3276. SET_DC_RTL_MIRRORED(g_hdcSrc);
  3277. }
  3278. return hdc;
  3279. }
  3280. void ImageList_ReleaseDragDC(HDC hdc)
  3281. {
  3282. //
  3283. // If the hdc is mirrored then unmirror the 2 globals DCs.
  3284. //
  3285. if (IS_DC_RTL_MIRRORED(hdc))
  3286. {
  3287. SET_DC_LAYOUT(g_hdcDst, 0);
  3288. SET_DC_LAYOUT(g_hdcSrc, 0);
  3289. }
  3290. ReleaseDC(g_dctx.hwndDC, hdc);
  3291. }
  3292. //
  3293. // x, y -- Specifies the initial cursor position in the coords of hwndLock,
  3294. // which is specified by the previous ImageList_StartDrag call.
  3295. //
  3296. HRESULT CImageList::DragMove(int x, int y)
  3297. {
  3298. int IncOne = 0;
  3299. ENTERCRITICAL;
  3300. if (g_dctx.fDragShow)
  3301. {
  3302. RECT rcOld, rcNew, rcBounds;
  3303. int dx, dy;
  3304. dx = x - g_dctx.ptDrag.x;
  3305. dy = y - g_dctx.ptDrag.y;
  3306. rcOld.left = g_dctx.ptDrag.x - g_dctx.ptDragHotspot.x;
  3307. rcOld.top = g_dctx.ptDrag.y - g_dctx.ptDragHotspot.y;
  3308. rcOld.right = rcOld.left + g_drb.sizeRestore.cx;
  3309. rcOld.bottom = rcOld.top + g_drb.sizeRestore.cy;
  3310. rcNew = rcOld;
  3311. OffsetRect(&rcNew, dx, dy);
  3312. if (!IntersectRect(&rcBounds, &rcOld, &rcNew))
  3313. {
  3314. //
  3315. // No intersection. Simply hide the old one and show the new one.
  3316. //
  3317. ImageList_DragShowNolock(FALSE);
  3318. g_dctx.ptDrag.x = x;
  3319. g_dctx.ptDrag.y = y;
  3320. ImageList_DragShowNolock(TRUE);
  3321. }
  3322. else
  3323. {
  3324. //
  3325. // Some intersection.
  3326. //
  3327. HDC hdcScreen;
  3328. int cx, cy;
  3329. UnionRect(&rcBounds, &rcOld, &rcNew);
  3330. hdcScreen = ImageList_GetDragDC();
  3331. if (hdcScreen)
  3332. {
  3333. //
  3334. // If the DC is RTL mirrored, then restrict the
  3335. // screen bitmap not to go beyond the screen since
  3336. // we will end up copying the wrong bits from the
  3337. // hdcScreen to the hbmOffScreen when the DC is mirrored.
  3338. // GDI will skip invalid screen coord from the screen into
  3339. // the destination bitmap. This will result in copying un-init
  3340. // bits back to the screen (since the screen is mirrored).
  3341. // [samera]
  3342. //
  3343. if (IS_DC_RTL_MIRRORED(hdcScreen))
  3344. {
  3345. RECT rcWindow;
  3346. GetWindowRect(g_dctx.hwndDC, &rcWindow);
  3347. rcWindow.right -= rcWindow.left;
  3348. if (rcBounds.right > rcWindow.right)
  3349. {
  3350. rcBounds.right = rcWindow.right;
  3351. }
  3352. if (rcBounds.left < 0)
  3353. {
  3354. rcBounds.left = 0;
  3355. }
  3356. }
  3357. cx = rcBounds.right - rcBounds.left;
  3358. cy = rcBounds.bottom - rcBounds.top;
  3359. //
  3360. // Copy the union rect from the screen to hbmOffScreen.
  3361. //
  3362. CImageList::SelectDstBitmap(g_drb.hbmOffScreen);
  3363. BitBlt(g_hdcDst, 0, 0, cx, cy,
  3364. hdcScreen, rcBounds.left, rcBounds.top, SRCCOPY);
  3365. //
  3366. // Hide the cursor on the hbmOffScreen by copying hbmRestore.
  3367. //
  3368. CImageList::SelectSrcBitmap(g_drb.hbmRestore);
  3369. BitBlt(g_hdcDst,
  3370. rcOld.left - rcBounds.left,
  3371. rcOld.top - rcBounds.top,
  3372. g_drb.sizeRestore.cx, g_drb.sizeRestore.cy,
  3373. g_hdcSrc, 0, 0, SRCCOPY);
  3374. //
  3375. // Copy the original screen bits to hbmRestore
  3376. //
  3377. BitBlt(g_hdcSrc, 0, 0, g_drb.sizeRestore.cx, g_drb.sizeRestore.cy,
  3378. g_hdcDst,
  3379. rcNew.left - rcBounds.left,
  3380. rcNew.top - rcBounds.top,
  3381. SRCCOPY);
  3382. //
  3383. // Draw the image on hbmOffScreen
  3384. //
  3385. if (g_dctx.fHiColor)
  3386. {
  3387. WimpyDrawEx(SAFECAST(g_dctx.pimlDrag, IImageList*), 0, g_hdcDst,
  3388. rcNew.left - rcBounds.left + IncOne,
  3389. rcNew.top - rcBounds.top, 0, 0, CLR_NONE, CLR_NONE, ILD_BLEND50);
  3390. if (g_dctx.puxCursor)
  3391. {
  3392. WimpyDraw(g_dctx.puxCursor, g_dctx.iCursor, g_hdcDst,
  3393. rcNew.left - rcBounds.left + g_dctx.ptCursor.x + IncOne,
  3394. rcNew.top - rcBounds.top + g_dctx.ptCursor.y,
  3395. ILD_NORMAL);
  3396. }
  3397. }
  3398. else
  3399. {
  3400. WimpyDraw(SAFECAST(g_dctx.pimlDrag, IImageList*), 0, g_hdcDst,
  3401. rcNew.left - rcBounds.left + IncOne,
  3402. rcNew.top - rcBounds.top, ILD_NORMAL);
  3403. }
  3404. //
  3405. // Copy the hbmOffScreen back to the screen.
  3406. //
  3407. BitBlt(hdcScreen, rcBounds.left, rcBounds.top, cx, cy,
  3408. g_hdcDst, 0, 0, SRCCOPY);
  3409. ImageList_ReleaseDragDC(hdcScreen);
  3410. }
  3411. g_dctx.ptDrag.x = x;
  3412. g_dctx.ptDrag.y = y;
  3413. }
  3414. }
  3415. LEAVECRITICAL;
  3416. return S_OK;
  3417. }
  3418. HRESULT CImageList::BeginDrag(int iTrack, int dxHotspot, int dyHotspot)
  3419. {
  3420. HRESULT hr = E_ACCESSDENIED;
  3421. ENTERCRITICAL;
  3422. if (!g_dctx.pimlDrag)
  3423. {
  3424. UINT newflags;
  3425. int cxI = 0, cyI = 0;
  3426. g_dctx.fDragShow = FALSE;
  3427. g_dctx.hwndDC = NULL;
  3428. g_dctx.fHiColor = GetScreenDepth() > 8;
  3429. newflags = _flags|ILC_SHARED;
  3430. if (g_dctx.fHiColor)
  3431. {
  3432. UINT uColorFlag = ILC_COLOR16;
  3433. if (GetScreenDepth() == 32 || GetScreenDepth() == 24)
  3434. {
  3435. uColorFlag = ILC_COLOR32;
  3436. }
  3437. newflags = (newflags & ~ILC_COLORMASK) | uColorFlag;
  3438. }
  3439. g_dctx.pimlDither = CImageList::Create(_cx, _cy, newflags, 1, 0);
  3440. if (g_dctx.pimlDither)
  3441. {
  3442. g_dctx.pimlDither->_cImage++;
  3443. g_dctx.ptDragHotspot.x = dxHotspot;
  3444. g_dctx.ptDragHotspot.y = dyHotspot;
  3445. g_dctx.pimlDither->_CopyOneImage(0, 0, 0, this, iTrack);
  3446. hr = ImageList_SetDragImage(NULL, 0, dxHotspot, dyHotspot)? S_OK : E_FAIL;
  3447. }
  3448. }
  3449. LEAVECRITICAL;
  3450. return hr;
  3451. }
  3452. HRESULT CImageList::DragEnter(HWND hwndLock, int x, int y)
  3453. {
  3454. HRESULT hr = S_FALSE;
  3455. hwndLock = hwndLock ? hwndLock : GetDesktopWindow();
  3456. ENTERCRITICAL;
  3457. if (!g_dctx.hwndDC)
  3458. {
  3459. g_dctx.hwndDC = hwndLock;
  3460. g_dctx.ptDrag.x = x;
  3461. g_dctx.ptDrag.y = y;
  3462. ImageList_DragShowNolock(TRUE);
  3463. hr = S_OK;
  3464. }
  3465. LEAVECRITICAL;
  3466. return hr;
  3467. }
  3468. HRESULT CImageList::DragLeave(HWND hwndLock)
  3469. {
  3470. HRESULT hr = S_FALSE;
  3471. hwndLock = hwndLock ? hwndLock : GetDesktopWindow();
  3472. ENTERCRITICAL;
  3473. if (g_dctx.hwndDC == hwndLock)
  3474. {
  3475. ImageList_DragShowNolock(FALSE);
  3476. g_dctx.hwndDC = NULL;
  3477. hr = S_OK;
  3478. }
  3479. LEAVECRITICAL;
  3480. return hr;
  3481. }
  3482. HRESULT CImageList::DragShowNolock(BOOL fShow)
  3483. {
  3484. HDC hdcScreen;
  3485. int x, y;
  3486. int IncOne = 0;
  3487. x = g_dctx.ptDrag.x - g_dctx.ptDragHotspot.x;
  3488. y = g_dctx.ptDrag.y - g_dctx.ptDragHotspot.y;
  3489. if (!g_dctx.pimlDrag)
  3490. return E_ACCESSDENIED;
  3491. //
  3492. // REVIEW: Why this block is in the critical section? We are supposed
  3493. // to have only one dragging at a time, aren't we?
  3494. //
  3495. ENTERCRITICAL;
  3496. if (fShow && !g_dctx.fDragShow)
  3497. {
  3498. hdcScreen = ImageList_GetDragDC();
  3499. CImageList::SelectSrcBitmap(g_drb.hbmRestore);
  3500. BitBlt(g_hdcSrc, 0, 0, g_drb.sizeRestore.cx, g_drb.sizeRestore.cy,
  3501. hdcScreen, x, y, SRCCOPY);
  3502. if (g_dctx.fHiColor)
  3503. {
  3504. WimpyDrawEx(SAFECAST(g_dctx.pimlDrag, IImageList*), 0, hdcScreen, x + IncOne, y, 0, 0, CLR_NONE, CLR_NONE, ILD_BLEND50);
  3505. if (g_dctx.puxCursor)
  3506. {
  3507. WimpyDraw(g_dctx.puxCursor, g_dctx.iCursor, hdcScreen,
  3508. x + g_dctx.ptCursor.x + IncOne, y + g_dctx.ptCursor.y, ILD_NORMAL);
  3509. }
  3510. }
  3511. else
  3512. {
  3513. WimpyDraw(SAFECAST(g_dctx.pimlDrag, IImageList*), 0, hdcScreen, x + IncOne, y, ILD_NORMAL);
  3514. }
  3515. ImageList_ReleaseDragDC(hdcScreen);
  3516. }
  3517. else if (!fShow && g_dctx.fDragShow)
  3518. {
  3519. hdcScreen = ImageList_GetDragDC();
  3520. CImageList::SelectSrcBitmap(g_drb.hbmRestore);
  3521. BitBlt(hdcScreen, x, y, g_drb.sizeRestore.cx, g_drb.sizeRestore.cy,
  3522. g_hdcSrc, 0, 0, SRCCOPY);
  3523. ImageList_ReleaseDragDC(hdcScreen);
  3524. }
  3525. g_dctx.fDragShow = fShow;
  3526. LEAVECRITICAL;
  3527. return S_OK;
  3528. }
  3529. // this hotspot stuff is broken in design
  3530. BOOL ImageList_MergeDragImages(int dxHotspot, int dyHotspot)
  3531. {
  3532. CImageList* pimlNew;
  3533. BOOL fRet = FALSE;
  3534. if (g_dctx.pimlDither)
  3535. {
  3536. if (g_dctx.puxCursor)
  3537. {
  3538. IImageList* pux = NULL;
  3539. IImageListPriv* puxpCursor;
  3540. if (SUCCEEDED(g_dctx.puxCursor->QueryInterface(IID_PPV_ARG(IImageListPriv, &puxpCursor))))
  3541. {
  3542. // If the cursor list has a mirrored list, let's use that.
  3543. if (FAILED(puxpCursor->GetMirror(IID_PPV_ARG(IImageList, &pux))))
  3544. {
  3545. pux = g_dctx.puxCursor;
  3546. if (pux)
  3547. pux->AddRef();
  3548. }
  3549. puxpCursor->Release();
  3550. }
  3551. g_dctx.pimlDither->_Merge(0, pux, g_dctx.iCursor, dxHotspot, dyHotspot, &pimlNew);
  3552. if (pimlNew && pimlNew->CreateDragBitmaps())
  3553. {
  3554. // WARNING: Don't destroy pimlDrag if it is pimlDither.
  3555. if (g_dctx.pimlDrag && (g_dctx.pimlDrag != g_dctx.pimlDither))
  3556. {
  3557. g_dctx.pimlDrag->Release();
  3558. }
  3559. g_dctx.pimlDrag = pimlNew;
  3560. fRet = TRUE;
  3561. }
  3562. pux->Release();
  3563. }
  3564. else
  3565. {
  3566. if (g_dctx.pimlDither->CreateDragBitmaps())
  3567. {
  3568. g_dctx.pimlDrag = g_dctx.pimlDither;
  3569. fRet = TRUE;
  3570. }
  3571. }
  3572. }
  3573. else
  3574. {
  3575. // not an error case if both aren't set yet
  3576. // only an error if we actually tried the merge and failed
  3577. fRet = TRUE;
  3578. }
  3579. return fRet;
  3580. }
  3581. BOOL ImageList_SetDragImage(HIMAGELIST piml, int i, int dxHotspot, int dyHotspot)
  3582. {
  3583. BOOL fVisible = g_dctx.fDragShow;
  3584. BOOL fRet;
  3585. ENTERCRITICAL;
  3586. if (fVisible)
  3587. ImageList_DragShowNolock(FALSE);
  3588. // only do this last step if everything is there.
  3589. fRet = ImageList_MergeDragImages(dxHotspot, dyHotspot);
  3590. if (fVisible)
  3591. ImageList_DragShowNolock(TRUE);
  3592. LEAVECRITICAL;
  3593. return fRet;
  3594. }
  3595. HRESULT CImageList::GetDragImage(POINT * ppt, POINT * pptHotspot, REFIID riid, void** ppv)
  3596. {
  3597. if (ppt)
  3598. {
  3599. ppt->x = g_dctx.ptDrag.x;
  3600. ppt->y = g_dctx.ptDrag.y;
  3601. }
  3602. if (pptHotspot)
  3603. {
  3604. pptHotspot->x = g_dctx.ptDragHotspot.x;
  3605. pptHotspot->y = g_dctx.ptDragHotspot.y;
  3606. }
  3607. if (g_dctx.pimlDrag)
  3608. {
  3609. return g_dctx.pimlDrag->QueryInterface(riid, ppv);
  3610. }
  3611. return E_ACCESSDENIED;
  3612. }
  3613. HRESULT CImageList::GetItemFlags(int i, DWORD *dwFlags)
  3614. {
  3615. if (IsImageListIndex(i) && _dsaFlags)
  3616. {
  3617. *dwFlags = _GetItemFlags(i);
  3618. return S_OK;
  3619. }
  3620. return E_INVALIDARG;
  3621. }
  3622. HRESULT CImageList::GetOverlayImage(int iOverlay, int* piIndex)
  3623. {
  3624. if (iOverlay <= 0 || iOverlay >= NUM_OVERLAY_IMAGES)
  3625. return E_INVALIDARG;
  3626. *piIndex = _aOverlayIndexes[iOverlay - 1];
  3627. return S_OK;
  3628. }
  3629. HRESULT CImageList::SetDragCursorImage(IUnknown* punk, int i, int dxHotspot, int dyHotspot)
  3630. {
  3631. HRESULT hr = E_INVALIDARG;
  3632. BOOL fVisible = g_dctx.fDragShow;
  3633. IImageList* pux;
  3634. if (SUCCEEDED(punk->QueryInterface(IID_PPV_ARG(IImageList, &pux))))
  3635. {
  3636. ENTERCRITICAL;
  3637. // do work only if something has changed
  3638. if ((g_dctx.puxCursor != pux) || (g_dctx.iCursor != i))
  3639. {
  3640. if (fVisible)
  3641. ImageList_DragShowNolock(FALSE);
  3642. IImageList* puxOld = g_dctx.puxCursor;
  3643. g_dctx.puxCursor = pux;
  3644. g_dctx.puxCursor->AddRef();
  3645. if (puxOld)
  3646. puxOld->Release();
  3647. g_dctx.iCursor = i;
  3648. g_dctx.ptCursor.x = dxHotspot;
  3649. g_dctx.ptCursor.y = dyHotspot;
  3650. hr = ImageList_MergeDragImages(dxHotspot, dyHotspot)? S_OK : E_FAIL;
  3651. if (fVisible)
  3652. ImageList_DragShowNolock(TRUE);
  3653. }
  3654. LEAVECRITICAL;
  3655. pux->Release();
  3656. }
  3657. return hr;
  3658. }
  3659. HRESULT CImageList::EndDrag()
  3660. {
  3661. ENTERCRITICAL;
  3662. ImageList_DragShowNolock(FALSE);
  3663. // WARNING: Don't destroy pimlDrag if it is pimlDither.
  3664. if (g_dctx.pimlDrag && (g_dctx.pimlDrag != g_dctx.pimlDither))
  3665. {
  3666. g_dctx.pimlDrag->Release();
  3667. }
  3668. g_dctx.pimlDrag = NULL;
  3669. if (g_dctx.pimlDither)
  3670. {
  3671. g_dctx.pimlDither->Release();
  3672. g_dctx.pimlDither = NULL;
  3673. }
  3674. if (g_dctx.puxCursor)
  3675. {
  3676. g_dctx.puxCursor->Release();
  3677. g_dctx.puxCursor = NULL;
  3678. }
  3679. g_dctx.iCursor = -1;
  3680. g_dctx.hwndDC = NULL;
  3681. LEAVECRITICAL;
  3682. return S_OK;
  3683. }
  3684. // APIs
  3685. BOOL WINAPI ImageList_SetDragCursorImage(HIMAGELIST piml, int i, int dxHotspot, int dyHotspot)
  3686. {
  3687. BOOL fRet = FALSE;
  3688. IUnknown* punk;
  3689. HRESULT hr = HIMAGELIST_QueryInterface(piml, IID_PPV_ARG(IUnknown, &punk));
  3690. if (SUCCEEDED(hr))
  3691. {
  3692. if (g_dctx.puxDragImage)
  3693. {
  3694. fRet = (S_OK == g_dctx.puxDragImage->SetDragCursorImage(punk, i, dxHotspot, dyHotspot));
  3695. }
  3696. punk->Release();
  3697. }
  3698. return fRet;
  3699. }
  3700. HIMAGELIST WINAPI ImageList_GetDragImage(POINT * ppt, POINT * pptHotspot)
  3701. {
  3702. if (g_dctx.puxDragImage)
  3703. {
  3704. IImageList* punk;
  3705. g_dctx.puxDragImage->GetDragImage(ppt, pptHotspot, IID_PPV_ARG(IImageList, &punk));
  3706. return reinterpret_cast<HIMAGELIST>(punk);
  3707. }
  3708. return NULL;
  3709. }
  3710. void WINAPI ImageList_EndDrag()
  3711. {
  3712. ENTERCRITICAL;
  3713. if (g_dctx.puxDragImage)
  3714. {
  3715. g_dctx.puxDragImage->EndDrag();
  3716. g_dctx.puxDragImage->Release();
  3717. g_dctx.puxDragImage = NULL;
  3718. }
  3719. LEAVECRITICAL;
  3720. }
  3721. BOOL WINAPI ImageList_BeginDrag(HIMAGELIST pimlTrack, int iTrack, int dxHotspot, int dyHotspot)
  3722. {
  3723. IImageList* pux;
  3724. if (SUCCEEDED(HIMAGELIST_QueryInterface(pimlTrack, IID_PPV_ARG(IImageList, &pux))))
  3725. {
  3726. if (SUCCEEDED(pux->BeginDrag(iTrack, dxHotspot, dyHotspot)))
  3727. {
  3728. g_dctx.puxDragImage = pux;
  3729. return TRUE;
  3730. }
  3731. }
  3732. return FALSE;
  3733. }
  3734. BOOL WINAPI ImageList_DragEnter(HWND hwndLock, int x, int y)
  3735. {
  3736. BOOL fRet = FALSE;
  3737. if (g_dctx.puxDragImage)
  3738. {
  3739. fRet = (S_OK == g_dctx.puxDragImage->DragEnter(hwndLock, x, y));
  3740. }
  3741. return fRet;
  3742. }
  3743. BOOL WINAPI ImageList_DragMove(int x, int y)
  3744. {
  3745. BOOL fRet = FALSE;
  3746. if (g_dctx.puxDragImage)
  3747. {
  3748. fRet = (S_OK == g_dctx.puxDragImage->DragMove(x, y));
  3749. }
  3750. return fRet;
  3751. }
  3752. BOOL WINAPI ImageList_DragLeave(HWND hwndLock)
  3753. {
  3754. BOOL fRet = FALSE;
  3755. if (g_dctx.puxDragImage)
  3756. {
  3757. fRet = (S_OK == g_dctx.puxDragImage->DragLeave(hwndLock));
  3758. }
  3759. return fRet;
  3760. }
  3761. BOOL WINAPI ImageList_DragShowNolock(BOOL fShow)
  3762. {
  3763. BOOL fRet = FALSE;
  3764. if (g_dctx.puxDragImage)
  3765. {
  3766. fRet = (S_OK == g_dctx.puxDragImage->DragShowNolock(fShow));
  3767. }
  3768. return fRet;
  3769. }
  3770. //============================================================================
  3771. // ImageList_Clone - clone a image list
  3772. //
  3773. // create a new imagelist with the same properties as the given
  3774. // imagelist, except mabey a new icon size
  3775. //
  3776. // piml - imagelist to clone
  3777. // cx,cy - new icon size (0,0) to use clone icon size.
  3778. // flags - new flags (used if no clone)
  3779. // cInitial- initial size
  3780. // cGrow - grow value (used if no clone)
  3781. //============================================================================
  3782. EXTERN_C HIMAGELIST WINAPI ImageList_Clone(HIMAGELIST himl, int cx, int cy, UINT flags, int cInitial, int cGrow)
  3783. {
  3784. IImageListPriv* puxp;
  3785. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageListPriv, &puxp))))
  3786. {
  3787. // always use the clone flags
  3788. puxp->GetFlags(&flags);
  3789. IUnknown* punkMirror;
  3790. if (SUCCEEDED(puxp->GetMirror(IID_PPV_ARG(IUnknown, &punkMirror))))
  3791. {
  3792. flags |= ILC_MIRROR;
  3793. punkMirror->Release();
  3794. }
  3795. IImageList* pux;
  3796. if (SUCCEEDED(puxp->QueryInterface(IID_PPV_ARG(IImageList, &pux))))
  3797. {
  3798. int cxI, cyI;
  3799. pux->GetIconSize(&cxI, &cyI);
  3800. if (cx == 0)
  3801. cx = cxI;
  3802. if (cy == 0)
  3803. cy = cyI;
  3804. pux->Release();
  3805. }
  3806. puxp->Release();
  3807. }
  3808. return ImageList_Create(cx,cy,flags,cInitial,cGrow);
  3809. }
  3810. HRESULT WINAPI ImageList_CreateInstance(int cx, int cy, UINT flags, int cInitial, int cGrow, REFIID riid, void** ppv)
  3811. {
  3812. CImageList* piml=NULL;
  3813. HRESULT hr = E_OUTOFMEMORY;
  3814. *ppv = NULL;
  3815. piml = CImageList::Create(cx, cy, flags, cInitial, cGrow);
  3816. if (piml)
  3817. {
  3818. //
  3819. // Let's create a mirrored imagelist, if requested.
  3820. //
  3821. if (piml->_flags & ILC_MIRROR)
  3822. {
  3823. piml->_flags &= ~ILC_MIRROR;
  3824. piml->_pimlMirror = CImageList::Create(cx, cy, flags, cInitial, cGrow);
  3825. if (piml->_pimlMirror)
  3826. {
  3827. piml->_pimlMirror->_flags &= ~ILC_MIRROR;
  3828. }
  3829. }
  3830. hr = piml->QueryInterface(riid, ppv);
  3831. piml->Release();
  3832. }
  3833. return hr;
  3834. }
  3835. HIMAGELIST WINAPI ImageList_Create(int cx, int cy, UINT flags, int cInitial, int cGrow)
  3836. {
  3837. IImageList* pux;
  3838. ImageList_CreateInstance(cx, cy, flags, cInitial, cGrow, IID_PPV_ARG(IImageList, &pux));
  3839. return reinterpret_cast<HIMAGELIST>(pux);
  3840. }
  3841. //
  3842. // When this code is compiled Unicode, this implements the
  3843. // ANSI version of the ImageList_LoadImage api.
  3844. //
  3845. HIMAGELIST WINAPI ImageList_LoadImageA(HINSTANCE hi, LPCSTR lpbmp, int cx, int cGrow, COLORREF crMask, UINT uType, UINT uFlags)
  3846. {
  3847. HIMAGELIST lpResult;
  3848. LPWSTR lpBmpW;
  3849. if (!IS_INTRESOURCE(lpbmp))
  3850. {
  3851. lpBmpW = ProduceWFromA(CP_ACP, lpbmp);
  3852. if (!lpBmpW)
  3853. {
  3854. return NULL;
  3855. }
  3856. }
  3857. else
  3858. {
  3859. lpBmpW = (LPWSTR)lpbmp;
  3860. }
  3861. lpResult = ImageList_LoadImageW(hi, lpBmpW, cx, cGrow, crMask, uType, uFlags);
  3862. if (!IS_INTRESOURCE(lpbmp))
  3863. FreeProducedString(lpBmpW);
  3864. return lpResult;
  3865. }
  3866. HIMAGELIST WINAPI ImageList_LoadImageW(HINSTANCE hi, LPCTSTR lpbmp, int cx, int cGrow, COLORREF crMask, UINT uType, UINT uFlags)
  3867. {
  3868. HBITMAP hbmImage;
  3869. HIMAGELIST piml = NULL;
  3870. BITMAP bm;
  3871. int cy, cInitial;
  3872. UINT flags;
  3873. hbmImage = (HBITMAP)LoadImage(hi, lpbmp, uType, 0, 0, uFlags);
  3874. if (hbmImage && (sizeof(bm) == GetObject(hbmImage, sizeof(bm), &bm)))
  3875. {
  3876. // If cx is not stated assume it is the same as cy.
  3877. // ASSERT(cx);
  3878. cy = bm.bmHeight;
  3879. if (cx == 0)
  3880. cx = cy;
  3881. cInitial = bm.bmWidth / cx;
  3882. ENTERCRITICAL;
  3883. flags = 0;
  3884. if (crMask != CLR_NONE)
  3885. flags |= ILC_MASK;
  3886. if (bm.bmBits)
  3887. flags |= (bm.bmBitsPixel & ILC_COLORMASK);
  3888. piml = ImageList_Create(cx, cy, flags, cInitial, cGrow);
  3889. if (piml)
  3890. {
  3891. int added;
  3892. if (crMask == CLR_NONE)
  3893. added = ImageList_Add(piml, hbmImage, NULL);
  3894. else
  3895. added = ImageList_AddMasked(piml, hbmImage, crMask);
  3896. if (added < 0)
  3897. {
  3898. ImageList_Destroy(piml);
  3899. piml = NULL;
  3900. }
  3901. }
  3902. LEAVECRITICAL;
  3903. }
  3904. if (hbmImage)
  3905. DeleteObject(hbmImage);
  3906. return reinterpret_cast<HIMAGELIST>((IImageList*)piml);
  3907. }
  3908. //
  3909. //
  3910. #undef ImageList_AddIcon
  3911. EXTERN_C int WINAPI ImageList_AddIcon(HIMAGELIST himl, HICON hIcon)
  3912. {
  3913. return ImageList_ReplaceIcon(himl, -1, hIcon);
  3914. }
  3915. EXTERN_C void WINAPI ImageList_CopyDitherImage(HIMAGELIST himlDst, WORD iDst,
  3916. int xDst, int yDst, HIMAGELIST himlSrc, int iSrc, UINT fStyle)
  3917. {
  3918. IImageListPriv* puxp;
  3919. if (SUCCEEDED(HIMAGELIST_QueryInterface(himlDst, IID_PPV_ARG(IImageListPriv, &puxp))))
  3920. {
  3921. IUnknown* punk;
  3922. if (SUCCEEDED(HIMAGELIST_QueryInterface(himlSrc, IID_PPV_ARG(IUnknown, &punk))))
  3923. {
  3924. puxp->CopyDitherImage(iDst, xDst, yDst, punk, iSrc, fStyle);
  3925. punk->Release();
  3926. }
  3927. puxp->Release();
  3928. }
  3929. }
  3930. //
  3931. // ImageList_Duplicate
  3932. //
  3933. // Makes a copy of the passed in imagelist.
  3934. //
  3935. HIMAGELIST WINAPI ImageList_Duplicate(HIMAGELIST himl)
  3936. {
  3937. IImageList* pret = NULL;
  3938. IImageList* pux;
  3939. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  3940. {
  3941. pux->Clone(IID_PPV_ARG(IImageList, &pret));
  3942. pux->Release();
  3943. }
  3944. return reinterpret_cast<HIMAGELIST>(pret);
  3945. }
  3946. BOOL WINAPI ImageList_Write(HIMAGELIST himl, LPSTREAM pstm)
  3947. {
  3948. BOOL fRet = FALSE;
  3949. IPersistStream* pps;
  3950. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IPersistStream, &pps))))
  3951. {
  3952. if (SUCCEEDED(pps->Save(pstm, TRUE)))
  3953. {
  3954. fRet = TRUE;
  3955. }
  3956. pps->Release();
  3957. }
  3958. return fRet;
  3959. }
  3960. HIMAGELIST WINAPI ImageList_Read(LPSTREAM pstm)
  3961. {
  3962. CImageList* piml = new CImageList();
  3963. if (piml)
  3964. {
  3965. if (SUCCEEDED(piml->Load(pstm)))
  3966. {
  3967. return reinterpret_cast<HIMAGELIST>((IImageList*)piml);
  3968. }
  3969. piml->Release();
  3970. }
  3971. return NULL;
  3972. }
  3973. WINCOMMCTRLAPI HRESULT WINAPI ImageList_ReadEx(DWORD dwFlags, LPSTREAM pstm, REFIID riid, PVOID* ppv)
  3974. {
  3975. HRESULT hr = E_OUTOFMEMORY;
  3976. CImageList* piml = new CImageList();
  3977. if (piml)
  3978. {
  3979. hr = piml->LoadEx(dwFlags, pstm);
  3980. if (SUCCEEDED(hr))
  3981. {
  3982. hr = piml->QueryInterface(riid, ppv);
  3983. }
  3984. piml->Release();
  3985. }
  3986. return hr;
  3987. }
  3988. WINCOMMCTRLAPI HRESULT WINAPI ImageList_WriteEx(HIMAGELIST himl, DWORD dwFlags, LPSTREAM pstm)
  3989. {
  3990. IImageListPersistStream* pps;
  3991. HRESULT hr = HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageListPersistStream, &pps));
  3992. if (SUCCEEDED(hr))
  3993. {
  3994. hr = pps->SaveEx(dwFlags, pstm);
  3995. pps->Release();
  3996. }
  3997. return hr;
  3998. }
  3999. BOOL WINAPI ImageList_GetImageRect(HIMAGELIST himl, int i, RECT * prcImage)
  4000. {
  4001. BOOL fRet = FALSE;
  4002. IImageList* pux;
  4003. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4004. {
  4005. if (SUCCEEDED(pux->GetImageRect(i, prcImage)))
  4006. {
  4007. fRet = TRUE;
  4008. }
  4009. pux->Release();
  4010. }
  4011. return fRet;
  4012. }
  4013. BOOL WINAPI ImageList_Destroy(HIMAGELIST himl)
  4014. {
  4015. BOOL fRet = FALSE;
  4016. IImageList* pux;
  4017. // Weirdness: We are doing a Query Interface first to verify that
  4018. // this is actually a valid imagelist, then we are calling release twice
  4019. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4020. {
  4021. // Release the interface we QI'd for
  4022. pux->Release();
  4023. // Release a second time to destroy the object
  4024. pux->Release();
  4025. fRet = TRUE;
  4026. }
  4027. return fRet;
  4028. }
  4029. int WINAPI ImageList_GetImageCount(HIMAGELIST himl)
  4030. {
  4031. int fRet = 0;
  4032. IImageList* pux;
  4033. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4034. {
  4035. pux->GetImageCount(&fRet);
  4036. pux->Release();
  4037. }
  4038. return fRet;
  4039. }
  4040. BOOL WINAPI ImageList_SetImageCount(HIMAGELIST himl, UINT uNewCount)
  4041. {
  4042. BOOL fRet = FALSE;
  4043. IImageList* pux;
  4044. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4045. {
  4046. fRet = (S_OK == pux->SetImageCount(uNewCount));
  4047. pux->Release();
  4048. }
  4049. return fRet;
  4050. }
  4051. int WINAPI ImageList_Add(HIMAGELIST himl, HBITMAP hbmImage, HBITMAP hbmMask)
  4052. {
  4053. int fRet = -1;
  4054. IImageList* pux;
  4055. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4056. {
  4057. pux->Add(hbmImage, hbmMask, &fRet);
  4058. pux->Release();
  4059. }
  4060. return fRet;
  4061. }
  4062. int WINAPI ImageList_ReplaceIcon(HIMAGELIST himl, int i, HICON hicon)
  4063. {
  4064. int fRet = -1;
  4065. IImageList* pux;
  4066. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4067. {
  4068. pux->ReplaceIcon(i, hicon, &fRet);
  4069. pux->Release();
  4070. }
  4071. return fRet;
  4072. }
  4073. COLORREF WINAPI ImageList_SetBkColor(HIMAGELIST himl, COLORREF clrBk)
  4074. {
  4075. COLORREF fRet = clrBk;
  4076. IImageList* pux;
  4077. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4078. {
  4079. pux->SetBkColor(clrBk, &fRet);
  4080. pux->Release();
  4081. }
  4082. return fRet;
  4083. }
  4084. COLORREF WINAPI ImageList_GetBkColor(HIMAGELIST himl)
  4085. {
  4086. COLORREF fRet = RGB(0,0,0);
  4087. IImageList* pux;
  4088. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4089. {
  4090. pux->GetBkColor(&fRet);
  4091. pux->Release();
  4092. }
  4093. return fRet;
  4094. }
  4095. BOOL WINAPI ImageList_SetOverlayImage(HIMAGELIST himl, int iImage, int iOverlay)
  4096. {
  4097. BOOL fRet = FALSE;
  4098. IImageList* pux;
  4099. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4100. {
  4101. fRet = (S_OK == pux->SetOverlayImage(iImage, iOverlay));
  4102. pux->Release();
  4103. }
  4104. return fRet;
  4105. }
  4106. BOOL WINAPI ImageList_Replace(HIMAGELIST himl, int i, HBITMAP hbmImage, HBITMAP hbmMask)
  4107. {
  4108. BOOL fRet = FALSE;
  4109. IImageList* pux;
  4110. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4111. {
  4112. fRet = (S_OK == pux->Replace(i, hbmImage, hbmMask));
  4113. pux->Release();
  4114. }
  4115. return fRet;
  4116. }
  4117. int WINAPI ImageList_AddMasked(HIMAGELIST himl, HBITMAP hbmImage, COLORREF crMask)
  4118. {
  4119. int fRet = -1;
  4120. IImageList* pux;
  4121. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4122. {
  4123. pux->AddMasked(hbmImage, crMask, &fRet);
  4124. pux->Release();
  4125. }
  4126. return fRet;
  4127. }
  4128. BOOL WINAPI ImageList_DrawEx(HIMAGELIST himl, int i, HDC hdcDst, int x, int y, int dx, int dy, COLORREF rgbBk, COLORREF rgbFg, UINT fStyle)
  4129. {
  4130. BOOL fRet = FALSE;
  4131. IImageList* pux;
  4132. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4133. {
  4134. IMAGELISTDRAWPARAMS imldp = {0};
  4135. imldp.cbSize = sizeof(imldp);
  4136. imldp.himl = himl;
  4137. imldp.i = i;
  4138. imldp.hdcDst = hdcDst;
  4139. imldp.x = x;
  4140. imldp.y = y;
  4141. imldp.cx = dx;
  4142. imldp.cy = dy;
  4143. imldp.rgbBk = rgbBk;
  4144. imldp.rgbFg = rgbFg;
  4145. imldp.fStyle = fStyle;
  4146. fRet = (S_OK == pux->Draw(&imldp));
  4147. pux->Release();
  4148. }
  4149. return fRet;
  4150. }
  4151. BOOL WINAPI ImageList_Draw(HIMAGELIST himl, int i, HDC hdcDst, int x, int y, UINT fStyle)
  4152. {
  4153. BOOL fRet = FALSE;
  4154. IImageList* pux;
  4155. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4156. {
  4157. IMAGELISTDRAWPARAMS imldp = {0};
  4158. imldp.cbSize = sizeof(imldp);
  4159. imldp.himl = himl;
  4160. imldp.i = i;
  4161. imldp.hdcDst = hdcDst;
  4162. imldp.x = x;
  4163. imldp.y = y;
  4164. imldp.rgbBk = CLR_DEFAULT;
  4165. imldp.rgbFg = CLR_DEFAULT;
  4166. imldp.fStyle = fStyle;
  4167. fRet = (S_OK == pux->Draw(&imldp));
  4168. pux->Release();
  4169. }
  4170. return fRet;
  4171. }
  4172. // Note: no distinction between failure case (bad himl) and no flags set
  4173. DWORD WINAPI ImageList_GetItemFlags(HIMAGELIST himl, int i)
  4174. {
  4175. DWORD dwFlags = 0;
  4176. if (himl)
  4177. {
  4178. IImageList* pux;
  4179. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4180. {
  4181. pux->GetItemFlags(i, &dwFlags);
  4182. pux->Release();
  4183. }
  4184. }
  4185. return dwFlags;
  4186. }
  4187. BOOL WINAPI ImageList_DrawIndirect(IMAGELISTDRAWPARAMS* pimldp)
  4188. {
  4189. BOOL fRet = FALSE;
  4190. IImageList* pux;
  4191. if (!pimldp)
  4192. return fRet;
  4193. if (SUCCEEDED(HIMAGELIST_QueryInterface(pimldp->himl, IID_PPV_ARG(IImageList, &pux))))
  4194. {
  4195. fRet = (S_OK == pux->Draw(pimldp));
  4196. pux->Release();
  4197. }
  4198. return fRet;
  4199. }
  4200. BOOL WINAPI ImageList_Remove(HIMAGELIST himl, int i)
  4201. {
  4202. BOOL fRet = FALSE;
  4203. IImageList* pux;
  4204. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4205. {
  4206. fRet = (S_OK == pux->Remove(i));
  4207. pux->Release();
  4208. }
  4209. return fRet;
  4210. }
  4211. HICON WINAPI ImageList_GetIcon(HIMAGELIST himl, int i, UINT flags)
  4212. {
  4213. HICON fRet = NULL;
  4214. IImageList* pux;
  4215. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4216. {
  4217. pux->GetIcon(i, flags, &fRet);
  4218. pux->Release();
  4219. }
  4220. return fRet;
  4221. }
  4222. BOOL WINAPI ImageList_Copy(HIMAGELIST himlDst, int iDst, HIMAGELIST himlSrc, int iSrc, UINT uFlags)
  4223. {
  4224. BOOL fRet = FALSE;
  4225. if (himlDst == himlSrc)
  4226. {
  4227. IImageList* pux;
  4228. if (SUCCEEDED(HIMAGELIST_QueryInterface(himlDst, IID_PPV_ARG(IImageList, &pux))))
  4229. {
  4230. fRet = (S_OK == pux->Copy(iDst,(IUnknown*)pux, iSrc, uFlags));
  4231. pux->Release();
  4232. }
  4233. }
  4234. return fRet;
  4235. }
  4236. BOOL WINAPI ImageList_GetIconSize(HIMAGELIST himl, int *cx, int *cy)
  4237. {
  4238. BOOL fRet = FALSE;
  4239. IImageList* pux;
  4240. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4241. {
  4242. fRet = (S_OK == pux->GetIconSize(cx, cy));
  4243. pux->Release();
  4244. }
  4245. return fRet;
  4246. }
  4247. BOOL WINAPI ImageList_SetIconSize(HIMAGELIST himl, int cx, int cy)
  4248. {
  4249. BOOL fRet = FALSE;
  4250. IImageList* pux;
  4251. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4252. {
  4253. fRet = (S_OK == pux->SetIconSize(cx, cy));
  4254. pux->Release();
  4255. }
  4256. return fRet;
  4257. }
  4258. BOOL WINAPI ImageList_GetImageInfo(HIMAGELIST himl, int i, IMAGEINFO* pImageInfo)
  4259. {
  4260. BOOL fRet = FALSE;
  4261. IImageList* pux;
  4262. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageList, &pux))))
  4263. {
  4264. fRet = (S_OK == pux->GetImageInfo(i, pImageInfo));
  4265. pux->Release();
  4266. }
  4267. return fRet;
  4268. }
  4269. HIMAGELIST WINAPI ImageList_Merge(HIMAGELIST himl1, int i1, HIMAGELIST himl2, int i2, int dx, int dy)
  4270. {
  4271. IImageList* fRet = NULL;
  4272. IImageList* pux1;
  4273. IImageList* pux2;
  4274. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl1, IID_PPV_ARG(IImageList, &pux1))))
  4275. {
  4276. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl2, IID_PPV_ARG(IImageList, &pux2))))
  4277. {
  4278. pux1->Merge(i1, (IUnknown*)pux2, i2, dx, dy, IID_PPV_ARG(IImageList, &fRet));
  4279. pux2->Release();
  4280. }
  4281. pux1->Release();
  4282. }
  4283. return reinterpret_cast<HIMAGELIST>(fRet);
  4284. }
  4285. BOOL WINAPI ImageList_SetFlags(HIMAGELIST himl, UINT flags)
  4286. {
  4287. BOOL fRet = FALSE;
  4288. IImageListPriv* pux;
  4289. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageListPriv, &pux))))
  4290. {
  4291. fRet = (S_OK == pux->SetFlags(flags));
  4292. pux->Release();
  4293. }
  4294. return fRet;
  4295. }
  4296. BOOL WINAPI ImageList_SetFilter(HIMAGELIST himl, PFNIMLFILTER pfnFilter, LPARAM lParamFilter)
  4297. {
  4298. return FALSE;
  4299. }
  4300. int ImageList_SetColorTable(HIMAGELIST himl, int start, int len, RGBQUAD *prgb)
  4301. {
  4302. int fRet = -1;
  4303. IImageListPriv* pux;
  4304. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageListPriv, &pux))))
  4305. {
  4306. pux->SetColorTable(start, len, prgb, &fRet);
  4307. pux->Release();
  4308. }
  4309. return fRet;
  4310. }
  4311. UINT WINAPI ImageList_GetFlags(HIMAGELIST himl)
  4312. {
  4313. UINT fRet = 0;
  4314. IImageListPriv* pux;
  4315. if (SUCCEEDED(HIMAGELIST_QueryInterface(himl, IID_PPV_ARG(IImageListPriv, &pux))))
  4316. {
  4317. pux->GetFlags(&fRet);
  4318. pux->Release();
  4319. }
  4320. return fRet;
  4321. }