ShowImgs.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508
  1. #include "StdAfx.h"
  2. #include "ShowImgs.h"
  3. CShowImgs::CShowImgs(void)
  4. {
  5. m_nLoadType = Load_Thumbnail;
  6. m_nMaxLoadImg = 5;
  7. m_pShowDC = NULL;
  8. m_pAryOfImgPath = NULL;
  9. m_pAryOfThumbImgPath = NULL;
  10. m_pCurShowImg = NULL;
  11. m_pCurShowThumImg = NULL;
  12. m_nCurShowIndex = 0;
  13. // 获得屏幕的宽(像素值);
  14. double fx = GetSystemMetrics(SM_CXSCREEN);
  15. // 获得屏幕的高(像素值);
  16. double fy = GetSystemMetrics(SM_CYSCREEN);
  17. }
  18. CShowImgs::~CShowImgs(void)
  19. {
  20. #if NOUSECLASS
  21. // 释放原图;
  22. if ( m_pAryLoadImgPtr.GetSize() )
  23. {
  24. for (INT i = m_pAryLoadImgPtr.GetSize() -1 ; i >= 0; i--)
  25. {
  26. DELETEP(m_pAryLoadImgPtr.ElementAt(i))
  27. m_pAryLoadImgPtr.RemoveAt(i);
  28. }
  29. }
  30. // 释放缩略图;
  31. if ( m_pAryLoadThumbImgPtr.GetSize() )
  32. {
  33. for (INT i = m_pAryLoadThumbImgPtr.GetSize() -1 ; i >= 0; i--)
  34. {
  35. DELETEP(m_pAryLoadThumbImgPtr.ElementAt(i))
  36. m_pAryLoadThumbImgPtr.RemoveAt(i);
  37. }
  38. }
  39. // 释放图片路径索引;
  40. if ( m_AryLoadIndex.GetSize() )
  41. {
  42. m_AryLoadIndex.RemoveAll();
  43. }
  44. #else
  45. if ( m_AryOfLoadedImg.GetSize() )
  46. {
  47. for ( INT i = m_AryOfLoadedImg.GetSize() -1; i >= 0; i-- )
  48. {
  49. DELETEP(m_AryOfLoadedImg.ElementAt(i))
  50. m_AryOfLoadedImg.RemoveAt(i);
  51. }
  52. }
  53. #endif
  54. // 释放显示的图片;
  55. DELETEP(m_pCurShowImg)
  56. DELETEP(m_pCurShowThumImg)
  57. }
  58. //////////////////////////////////////////////////////////////////////////
  59. BOOL CShowImgs::LoadImgFromFile(IN Image** pImg, LPCTSTR lpPath)
  60. {
  61. if ( !PathFileExists(lpPath) )
  62. return FALSE;
  63. if ( *pImg )
  64. delete *pImg;
  65. *pImg = NULL;
  66. #ifdef UNICODE
  67. *pImg = Image::FromFile(lpPath);
  68. #else
  69. BSTR strtmp = _bstr_t(lpPath);
  70. *pImg = Image::FromFile(strtmp);
  71. SysFreeString(strtmp);
  72. #endif
  73. return (*pImg ? TRUE : FALSE);
  74. }
  75. BOOL CShowImgs::LoadImgFromBuffer(IN Image** pImg, IN BYTE* pBuffer, IN CONST INT& nBufLen)
  76. {
  77. if ( pBuffer == NULL )
  78. return FALSE;
  79. if ( *pImg )
  80. delete *pImg;
  81. *pImg = NULL;
  82. HGLOBAL hMemery = GlobalAlloc(GMEM_MOVEABLE, nBufLen);
  83. if ( hMemery == NULL )
  84. return FALSE;
  85. BYTE *pMem = (BYTE*)GlobalLock(hMemery);
  86. memcpy(pMem, pBuffer, nBufLen);
  87. IStream *pstream = NULL;
  88. CreateStreamOnHGlobal(hMemery, TRUE, &pstream);
  89. *pImg = Image::FromStream(pstream);
  90. GlobalUnlock(hMemery);
  91. pstream->Release();
  92. return (*pImg ? TRUE : FALSE);
  93. }
  94. // 先以只读方式从文件中读取二进制流出来,可以做到不独占文件;
  95. BOOL CShowImgs::LoadImgFromBuffer(IN Image** pImg, LPCTSTR lpPath)
  96. {
  97. if ( !PathFileExists(lpPath) )
  98. return FALSE;
  99. if ( *pImg )
  100. delete *pImg;
  101. *pImg = NULL;
  102. CFile fp;
  103. CFileException e;
  104. BOOL bRet = FALSE;
  105. if ( fp.Open(lpPath, CFile::modeRead, &e))
  106. {
  107. DWORD dwLength = (DWORD)fp.GetLength();
  108. BYTE *pData = new BYTE[dwLength];
  109. fp.Read(pData,dwLength);
  110. fp.Close();
  111. bRet = LoadImgFromBuffer(pImg, pData, dwLength);
  112. if( pData )
  113. delete []pData;
  114. }
  115. return bRet;
  116. }
  117. //////////////////////////////////////////////////////////////////////////
  118. BOOL CShowImgs::GetOrientation(IN Image *pImg)
  119. {
  120. if (pImg == NULL) return FALSE;
  121. UINT totalBufferSize;
  122. UINT numProperties;
  123. pImg->GetPropertySize(&totalBufferSize, &numProperties);
  124. // Allocate the buffer that will receive the property items.
  125. PropertyItem* pAllItems = (PropertyItem*)malloc(totalBufferSize);
  126. // Fill the buffer.
  127. pImg->GetAllPropertyItems(totalBufferSize, numProperties, pAllItems);
  128. // Print the id data member of each property item.
  129. for (UINT j = 0; j < numProperties; ++j)
  130. {
  131. if (PropertyTagOrientation == pAllItems[j].id)
  132. {
  133. short* ptrLong = (short*)(pAllItems[j].value);
  134. int ret = (int)*ptrLong;
  135. free(pAllItems);
  136. return ret;
  137. }
  138. }
  139. free(pAllItems);
  140. return TRUE;
  141. }
  142. /************************************************************************/
  143. /*
  144. 函数:GetEncoderClsid
  145. 描述:获取GDI+支持的图像格式编码器种类,以及所有种类编码器信息;
  146. 参数:
  147. IN: format 要获取的图像格式;
  148. OUT: pClsid 返回符合条件的图像编码器信息;
  149. 返回:成功返回编码器索引,否则返回-1;
  150. */
  151. /************************************************************************/
  152. int CShowImgs::GetEncoderClsid(IN CONST WCHAR* format, OUT CLSID* pClsid)
  153. {
  154. // CD+支持的图像编码器数量;
  155. UINT numEncoders = 0;
  156. // 每个图像编码器以数组形式保存的大小;
  157. UINT nSize = 0;
  158. ImageCodecInfo* pImageCodecInfo = NULL;
  159. // 获取GDI+支持的图像格式编码器种类数以及ImageCodecInfo数组的存放大小;
  160. GetImageEncodersSize(&numEncoders, &nSize);
  161. if (nSize == 0)
  162. return -1;
  163. // 为ImageCodecInfo数组分配足额空间;
  164. pImageCodecInfo = (ImageCodecInfo*)(malloc(nSize));
  165. if (pImageCodecInfo == NULL)
  166. return -1;
  167. // 获取所有的图像编码器信息;
  168. GetImageEncoders(numEncoders, nSize, pImageCodecInfo);
  169. // 查找符合的图像编码器的Clsid;
  170. for (UINT i = 0; i < numEncoders; ++i)
  171. {
  172. if (wcscmp(pImageCodecInfo[i].MimeType, format) == 0)
  173. {
  174. *pClsid = pImageCodecInfo[i].Clsid;
  175. free(pImageCodecInfo);
  176. return i; // Success
  177. }
  178. }
  179. // 释放步骤3分配的内存;
  180. free(pImageCodecInfo);
  181. return -1;
  182. }
  183. /************************************************************************/
  184. /*
  185. 函数:ZoomImg
  186. 描述:缩放到指定大小的区域中,返回缩放后的尺寸;
  187. 参数:
  188. IN: Imgrc 图像的大小;
  189. IN: dwDisplayPix 图像要显示的区域的大小;
  190. OUT: dwZoomPix 图像在显示区域内缩放后的尺寸;
  191. 返回:
  192. 注意:dwZoomPix的尺寸必须在dwDisplayPix内;
  193. */
  194. /************************************************************************/
  195. int CShowImgs::ZoomImg(IN CRect &Imgrc, IN CONST DWORD &dwDisplayPix, OUT DWORD &dwZoomPix)
  196. {
  197. double fDisplayWidth = GET_XPIX(dwDisplayPix);
  198. double fDisplayHeight = GET_YPIX(dwDisplayPix);
  199. // 显示区域长宽比;
  200. double fDisplayAspectRatio = fDisplayWidth / fDisplayHeight;
  201. // 图片长宽比;
  202. double fImgAspectRatio = ((double)Imgrc.Width()) / ((double)Imgrc.Height());
  203. double fZoomWidth;
  204. double fZoomHeight;
  205. if (fDisplayAspectRatio > fImgAspectRatio)
  206. {
  207. fZoomWidth = fDisplayHeight*fImgAspectRatio;
  208. fZoomHeight = fDisplayHeight;
  209. }
  210. else
  211. {
  212. fZoomWidth = fDisplayWidth;
  213. fZoomHeight = fDisplayWidth / fImgAspectRatio;
  214. }
  215. dwZoomPix = SET_PIX((int)fZoomWidth, (int)fZoomHeight);
  216. //////////////////////////////////////////////////////////////////////////
  217. int nRetval = 0;
  218. if ((fDisplayWidth == Imgrc.Width()) && (fDisplayHeight == Imgrc.Height()))
  219. {
  220. nRetval = ZoomNull;
  221. }
  222. else if ((fDisplayWidth > Imgrc.Width()) && (fDisplayHeight > Imgrc.Height()))
  223. {
  224. nRetval = ZoomOut;
  225. }
  226. else
  227. {
  228. nRetval = ZoomIn;
  229. }
  230. return nRetval;
  231. }
  232. int CShowImgs::ZoomImg(IN CONST DWORD &dwImgPix, IN CONST DWORD &dwDisplayPix, OUT DWORD &dwZoomPix)
  233. {
  234. double fDisplayWidth = GET_XPIX(dwDisplayPix);
  235. double fDisplayHeight = GET_YPIX(dwDisplayPix);
  236. // 显示区域长宽比;
  237. double fDisplayAspectRatio = fDisplayWidth / fDisplayHeight;
  238. // 图片长宽比;
  239. double fImgAspectRatio = ((double)GET_XPIX(dwImgPix)) / ((double)GET_YPIX(dwImgPix));
  240. double fZoomWidth;
  241. double fZoomHeight;
  242. if (fDisplayAspectRatio > fImgAspectRatio)
  243. {
  244. fZoomWidth = fDisplayHeight*fImgAspectRatio;
  245. fZoomHeight = fDisplayHeight;
  246. }
  247. else
  248. {
  249. fZoomWidth = fDisplayWidth;
  250. fZoomHeight = fDisplayWidth / fImgAspectRatio;
  251. }
  252. dwZoomPix = SET_PIX((int)fZoomWidth, (int)fZoomHeight);
  253. //////////////////////////////////////////////////////////////////////////
  254. int nRetval = 0;
  255. if ((fDisplayWidth == GET_XPIX(dwImgPix)) && (fDisplayHeight == GET_YPIX(dwImgPix)))
  256. {
  257. nRetval = ZoomNull;
  258. }
  259. else if ((fDisplayWidth > GET_XPIX(dwImgPix)) && (fDisplayHeight > GET_YPIX(dwImgPix)))
  260. {
  261. nRetval = ZoomOut;
  262. }
  263. else
  264. {
  265. nRetval = ZoomIn;
  266. }
  267. return nRetval;
  268. }
  269. /************************************************************************/
  270. /* 函数:LoadImgByIndex[3/18/2016 IT];
  271. /* 描述:根据图片路径索引来加载要显示在DC上的图片;
  272. /* 参数:;
  273. /* [IN] nIndex:图片路径的索引,默认-1,表示使用m_nCurShowIndex;
  274. /* 返回:void;
  275. /* 注意:;
  276. /* 示例:;
  277. /*
  278. /* 修改:;
  279. /* 日期:;
  280. /* 内容:;
  281. /************************************************************************/
  282. void CShowImgs::LoadImgByIndex(IN CONST INT& nIndex /* = -1 */)
  283. {
  284. if ( nIndex != -1 )
  285. m_nCurShowIndex = nIndex;
  286. if ( m_nCurShowIndex < 0 ){
  287. OutputDebugString(_T("要加载的图片已超出索引范围(负数)!\n"));
  288. return;
  289. }
  290. BOOL bHasLoad = FALSE;
  291. if ( m_nLoadType == Load_Original || m_nLoadType == Load_Thumbnail)
  292. {// 原图;
  293. if ( m_nCurShowIndex >= m_pAryOfImgPath->GetSize() ){
  294. OutputDebugString(_T("要加载的原图片已超出索引范围!\n"));
  295. return;
  296. }
  297. m_pCurShowImg = NULL;
  298. if ( IsLoadTheIndexImg(m_nCurShowIndex) != -1)
  299. {// 加载过该路径索引的图片;
  300. m_pCurShowImg = m_AryOfLoadedImg.ElementAt(m_nCurShowIndex)->GetImgPtr();
  301. }
  302. else
  303. {// 未加载过该路径索引的图片;
  304. if ( !LoadImgFromFile(&m_pCurShowImg, m_pAryOfImgPath->ElementAt(m_nCurShowIndex)) )
  305. {// 从文件中加载失败,使用空白图片代替;
  306. Bitmap *pNULLBmp = new Bitmap(100, 100, PixelFormat24bppRGB);
  307. Graphics graph(pNULLBmp);
  308. graph.Clear(Color(255, 0, 0, 0));
  309. CPerImg *pPerImg = new CPerImg;
  310. pPerImg->SetImgPtr(pNULLBmp);
  311. pPerImg->SetIndex(m_nCurShowIndex);
  312. m_AryOfLoadedImg.Add(pPerImg);
  313. }
  314. else
  315. {// 从文件中加载成功;
  316. CPerImg *pPerImg = new CPerImg;
  317. pPerImg->SetIndex(m_nCurShowIndex);
  318. pPerImg->SetImgPtr(m_pCurShowImg);
  319. m_AryOfLoadedImg.Add(pPerImg);
  320. }
  321. }
  322. }
  323. else if ( m_nLoadType == Load_both )
  324. {// 同时加载原图和缩略图;
  325. if ( m_pAryOfImgPath->GetSize() != m_pAryOfThumbImgPath->GetSize() ){
  326. OutputDebugString(_T("要加载的原图和缩略图数量不等!\n"));
  327. return;
  328. }
  329. if ( m_nCurShowIndex >= m_pAryOfImgPath->GetSize() ){
  330. OutputDebugString(_T("要加载的图片已超出索引范围!\n"));
  331. return;
  332. }
  333. m_pCurShowImg = m_pCurShowThumImg = NULL;
  334. if ( IsLoadTheIndexImg(m_nCurShowIndex) != -1)
  335. {// 加载过该路径索引的图片;
  336. m_pCurShowImg = m_AryOfLoadedImg.ElementAt(m_nCurShowIndex)->GetImgPtr();
  337. m_pCurShowThumImg = m_AryOfLoadedImg.ElementAt(m_nCurShowIndex)->GetThumbImgPtr();
  338. }
  339. else
  340. {// 未加载过该路径索引的图片;
  341. if ( !LoadImgFromFile(&m_pCurShowImg, m_pAryOfImgPath->ElementAt(m_nCurShowIndex)) )
  342. {// 从文件中加载失败,使用空白图片代替;
  343. Bitmap *pNULLBmp = new Bitmap(100, 100, PixelFormat24bppRGB);
  344. Graphics graph(pNULLBmp);
  345. graph.Clear(Color(255, 0, 0, 0));
  346. Bitmap *pNULLThumbBmp = new Bitmap(100, 100, PixelFormat24bppRGB);
  347. Graphics graphOfThumb(pNULLThumbBmp);
  348. graphOfThumb.Clear(Color(255, 0, 0, 0));
  349. CPerImg *pPerImg = new CPerImg;
  350. pPerImg->SetImgPtr(pNULLBmp);
  351. pPerImg->SetThumbImgPtr(pNULLThumbBmp);
  352. pPerImg->SetIndex(m_nCurShowIndex);
  353. m_AryOfLoadedImg.Add(pPerImg);
  354. }
  355. else
  356. {// 从文件中加载成功;
  357. if ( !LoadImgFromFile(&m_pCurShowThumImg, m_pAryOfThumbImgPath->ElementAt(m_nCurShowIndex)) )
  358. {// 从文件中加载失败,使用空白图片代替;
  359. Bitmap *pNULLBmp = new Bitmap(100, 100, PixelFormat24bppRGB);
  360. Graphics graph(pNULLBmp);
  361. graph.Clear(Color(255, 0, 0, 0));
  362. m_pCurShowThumImg = pNULLBmp;
  363. }
  364. CPerImg *pPerImg = new CPerImg;
  365. pPerImg->SetIndex(m_nCurShowIndex);
  366. pPerImg->SetImgPtr(m_pCurShowImg);
  367. pPerImg->SetThumbImgPtr(m_pCurShowThumImg);
  368. m_AryOfLoadedImg.Add(pPerImg);
  369. }
  370. }
  371. }
  372. else
  373. {
  374. ATLTRACE("没有该类型的图片路径!\n");
  375. }
  376. }
  377. /************************************************************************/
  378. /* 函数:[3/18/2016 IT];
  379. /* 描述:;
  380. /* 参数:;
  381. /* [IN] :;
  382. /* [OUT] :;
  383. /* [IN/OUT] :;
  384. /* 返回:void;
  385. /* 注意:;
  386. /* 示例:;
  387. /*
  388. /* 修改:;
  389. /* 日期:;
  390. /* 内容:;
  391. /************************************************************************/
  392. void CShowImgs::ReleaseExtraImg()
  393. {
  394. if ( m_AryOfLoadedImg.GetSize() > m_nMaxLoadImg )
  395. {
  396. for ( INT i = m_AryOfLoadedImg.GetSize() -1; i > m_nMaxLoadImg; i-- )
  397. {
  398. DELETEP(m_AryOfLoadedImg.ElementAt(i))
  399. m_AryOfLoadedImg.RemoveAt(i);
  400. }
  401. }
  402. }
  403. /************************************************************************/
  404. /* 函数:IsLoadTheIndexImg[3/18/2016 IT];
  405. /* 描述:指定路径索引的图片是否已加载过;
  406. /* 参数:;
  407. /* [IN] nIndex:要加载的图片路径索引,默认-1,表示使用m_nCurShowIndex;;
  408. /* 返回:已加载返回路径索引, 否则返回-1;
  409. /* 注意:;
  410. /* 示例:;
  411. /*
  412. /* 修改:;
  413. /* 日期:;
  414. /* 内容:;
  415. /************************************************************************/
  416. INT CShowImgs::IsLoadTheIndexImg(IN CONST INT& nIndex /* = -1 */)
  417. {
  418. if ( nIndex != -1 )
  419. m_nCurShowIndex = nIndex;
  420. INT nIndexOfPath = -1;
  421. #if NOUSECLASS
  422. for ( INT i = 0; i < m_AryLoadIndex.GetSize(); i++ ){
  423. if ( m_nCurShowIndex == m_AryLoadIndex.ElementAt(i) ){
  424. nIndexOfPath = i;
  425. break;
  426. }//if;
  427. }//for;
  428. #else
  429. for ( INT i = 0; i < m_AryOfLoadedImg.GetSize(); i++ )
  430. {
  431. if ( m_nCurShowIndex == m_AryOfLoadedImg.ElementAt(i)->GetIndex() ){
  432. nIndexOfPath = i;
  433. break;
  434. }
  435. }
  436. #endif
  437. return nIndexOfPath;
  438. }