IMGCommon.cpp 28 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097
  1. #include "StdAfx.h"
  2. #include "IMGCommon.h"
  3. #include <strsafe.h>
  4. #include <comutil.h>
  5. #pragma comment(lib, "comsuppw.lib")
  6. const ColorMatrix _ClrMatrix =
  7. {
  8. 1.0, 0.0, 0.0, 0.0, 0.0,
  9. 0.0, 1.0, 0.0, 0.0, 0.0,
  10. 0.0, 0.0, 0.3, 0.0, 0.0,
  11. 0.0, 0.0, 0.0, 1.0, 0.0,
  12. 0.0, 0.0, 0.0, 0.0, 1.0,
  13. };
  14. ULONG_PTR IMGCommon::m_gdiplusToken;
  15. IMGCommon::IMGCommon(void)
  16. {
  17. GdiplusStartupInput gdiplusStartupInput;
  18. GdiplusStartup(&m_gdiplusToken, &gdiplusStartupInput, NULL);
  19. }
  20. IMGCommon::~IMGCommon(void)
  21. {
  22. Gdiplus::GdiplusShutdown(m_gdiplusToken);
  23. }
  24. /************************************************************************/
  25. /* 函数:IsCorrectExt[6/1/2016 IT];
  26. /* 描述:扩展名是否正确;
  27. /* 参数:;
  28. /* [IN] fext:要判断的扩展名;
  29. /* [OUT] lpMistakenExt:返回错误的扩展名;
  30. /* 返回:扩展名错误返回-1、扩展名为*.*返回0、扩展符合要求返回1;
  31. /* 注意:;
  32. /* 示例:;
  33. /*
  34. /* 修改:;
  35. /* 日期:;
  36. /* 内容:;
  37. /************************************************************************/
  38. INT IMGCommon::IsCorrectExt( IN const TString &fext, OUT TString* lpMistakenExt /* = NULL */ )
  39. {
  40. if (fext.size() == 0)
  41. {
  42. WriteTextLog(_T("要查找的扩展名字符空!"));
  43. return -1;
  44. }
  45. if (fext.find(_T("*.*")) != TString::npos)
  46. {
  47. WriteTextLog(_T("要查找的扩展名为\"*.*\""));
  48. return 0;
  49. }
  50. TString ext = fext;
  51. if (ext[ext.length() - 1] != _T('|'))
  52. ext.append(_T("|"));
  53. INT bRet = 1;
  54. CONST TString sCorrectExt = _T("*.jpg|*.jpeg|*.png|*.bmp|*.cr2|*.nef|*.raw|*.ra2|*.ofr|*.pef|*.sr2");
  55. int nIndex = 0;
  56. do
  57. {
  58. nIndex = ext.find(_T('|'));
  59. if (nIndex != TString::npos)
  60. {
  61. if (sCorrectExt.find(ext.substr(0, nIndex)) == TString::npos)
  62. {
  63. if (lpMistakenExt)
  64. *lpMistakenExt = ext.substr(0, nIndex);
  65. bRet = -1;
  66. break;
  67. }
  68. ext = ext.substr(nIndex + 1);
  69. }
  70. } while (ext.find(_T('|')) != TString::npos);
  71. return bRet;
  72. }
  73. /************************************************************************/
  74. /* 函数:ExtMerge[6/1/2016 IT];
  75. /* 描述:合并扩展名;
  76. /* 参数:;
  77. /* [IN] ext1:要合并的扩展名1;
  78. /* [IN] ext2:要合并的扩展名2;
  79. /* [OUT] merge:合并后的扩展名;
  80. /* 返回:只合并合法的扩展名,若两扩展名都非法,返回FALSE;
  81. /* 注意:;
  82. /* 示例:;
  83. /*
  84. /* 修改:;
  85. /* 日期:;
  86. /* 内容:;
  87. /************************************************************************/
  88. BOOL IMGCommon::ExtMerge(IN CONST TString& ext1, IN CONST TString& ext2, OUT TString &merge) // 合并扩展名;
  89. {
  90. // 1.判断扩展名1合法性;
  91. INT nRet = IsCorrectExt(ext1);
  92. if ( nRet == -1 )
  93. {
  94. nRet = IsCorrectExt(ext2);
  95. if ( nRet == -1)
  96. return FALSE;
  97. merge = ext2;
  98. return TRUE;
  99. }
  100. if (nRet == 0)
  101. {
  102. merge = _T("*.*");
  103. return TRUE;
  104. }
  105. merge = ext1;
  106. // 2.判断扩展名2合法性;
  107. nRet = IsCorrectExt(ext2);
  108. if (nRet != 1) return TRUE;
  109. // 3.合并2扩展名;
  110. if (merge[merge.length() - 1] != _T('|'))
  111. merge.append(_T("|"));
  112. merge.append(ext2);
  113. return TRUE;
  114. }
  115. /************************************************************************/
  116. /* 函数:GetFileName[6/1/2016 IT];
  117. /* 描述:根据参数,获取参数中的文件名;
  118. /* 参数:;
  119. /* [IN] strfile:要获取文件名的全路径名;
  120. /* 返回:获取成功返回文件名,否则返回空;
  121. /* 注意:;
  122. /* 示例:;
  123. /*
  124. /* 修改:;
  125. /* 日期:;
  126. /* 内容:;
  127. /************************************************************************/
  128. CString IMGCommon::GetFileName(IN CONST CString& strfile)
  129. {
  130. CString name = _T("");
  131. int nIndex = strfile.ReverseFind(_T('\\'));
  132. if (nIndex == -1)
  133. return _T("");
  134. name = strfile.Mid(nIndex + 1);
  135. nIndex = name.ReverseFind(_T('.'));
  136. if (nIndex == -1)
  137. return _T("");
  138. name = name.Mid(0, nIndex);
  139. return name;
  140. }
  141. /************************************************************************/
  142. /* 函数:GetFilteringImg[6/1/2016 IT];
  143. /* 描述:过滤掉扩展名不符合要求的文件名;
  144. /* 参数:;
  145. /* [IN] AryImgs:要被过滤的文件名数组;
  146. /* [IN] lpCopyExt:要保留的文件的扩展名;
  147. /* [IN] bPickoutThumbnail:是否剔除缩略图;
  148. /* 返回:void;
  149. /* 注意:;
  150. /* 示例:;
  151. /*
  152. /* 修改:;
  153. /* 日期:;
  154. /* 内容:;
  155. /************************************************************************/
  156. void IMGCommon::GetFilteringImg(IN CStringArray &AryImgs, IN LPCTSTR lpCopyExt, IN BOOL bPickoutThumbnail /*= TRUE*/)
  157. {
  158. if (AryImgs.GetSize() == 0 || lpCopyExt == NULL) return;
  159. // 获取复制扩展名;
  160. int nIndex = 0;
  161. TString strtmp;
  162. TString strCopyExt(lpCopyExt);
  163. if (strCopyExt.find(_T("*.*")) != TString::npos) return;
  164. strCopyExt.append(_T("|"));
  165. STR_VEC vtCopyExt;
  166. // 将所有扩展名解析到数组里;
  167. do
  168. {
  169. nIndex = strCopyExt.find(_T('|'));
  170. if (nIndex != TString::npos)
  171. {
  172. strtmp = strCopyExt.substr(0, nIndex);
  173. strCopyExt = strCopyExt.substr(nIndex + 1);
  174. if (strtmp.compare(_T("*.*")))
  175. vtCopyExt.push_back(strtmp.substr(1));
  176. }
  177. } while (strCopyExt.find(_T('|')) != TString::npos);
  178. // 过滤非复制扩展名的文件,同时去掉缩略图,保留指定复制扩展名的文件;
  179. for (int i = AryImgs.GetSize() - 1; i >= 0; i--)
  180. {
  181. BOOL bExsit = FALSE;
  182. for (STR_VEC::iterator itExt = vtCopyExt.begin(); itExt != vtCopyExt.end(); itExt++)
  183. {
  184. if (AryImgs.ElementAt(i).MakeLower().Find(itExt->c_str()) != -1) {
  185. bExsit = TRUE;
  186. break;
  187. }
  188. }
  189. if (!bExsit) {
  190. AryImgs.RemoveAt(i);
  191. continue;
  192. }
  193. if (bPickoutThumbnail)
  194. {// 这里应该去除"s"和"m"开头的缩略图;// 但新的做法是在订单里创建缩略图子文件夹;// 总而言之,都要去掉缩略图;
  195. nIndex = AryImgs.ElementAt(i).ReverseFind(_T('\\'));
  196. if (nIndex != -1)
  197. {
  198. strtmp = AryImgs.ElementAt(i).Mid(nIndex + 1);
  199. if (strtmp[0] == _T('s') || strtmp[0] == _T('m')) {
  200. AryImgs.RemoveAt(i);
  201. continue;
  202. }
  203. }
  204. }
  205. }
  206. }
  207. /************************************************************************/
  208. /* 函数:GetFilteringImg[6/1/2016 IT];
  209. /* 描述:过滤掉扩展名不符合要求的文件名;
  210. /* 参数:;
  211. /* [IN] vtImgs:要被过滤的文件名数组;
  212. /* [IN] lpCopyExt:要保留的文件的扩展名;
  213. /* [IN] bPickoutThumbnail:是否剔除缩略图;
  214. /* 返回:void;
  215. /* 注意:;
  216. /* 示例:;
  217. /*
  218. /* 修改:;
  219. /* 日期:;
  220. /* 内容:;
  221. /************************************************************************/
  222. void IMGCommon::GetFilteringImg(IN STR_VEC &vtImgs, IN LPCTSTR lpCopyExt, IN BOOL bPickoutThumbnail /*= TRUE*/)
  223. {
  224. if (vtImgs.size() == 0 || lpCopyExt == NULL) return;
  225. // 获取复制扩展名;
  226. int nIndex = 0;
  227. TString strtmp;
  228. TString strCopyExt(lpCopyExt);
  229. if (strCopyExt.find(_T("*.*")) != TString::npos) return;
  230. strCopyExt.append(_T("|"));
  231. STR_VEC vtCopyExt;
  232. // 将所有扩展名解析到数组里;
  233. do
  234. {
  235. nIndex = strCopyExt.find(_T('|'));
  236. if (nIndex != TString::npos)
  237. {
  238. strtmp = strCopyExt.substr(0, nIndex);
  239. strCopyExt = strCopyExt.substr(nIndex + 1);
  240. if (strtmp.compare(_T("*.*")))
  241. vtCopyExt.push_back(strtmp);
  242. }
  243. } while (strCopyExt.find(_T('|')) != TString::npos);
  244. // 过滤非复制扩展名的文件,同时去掉缩略图,保留指定复制扩展名的文件;
  245. for (STR_VEC::iterator it = vtImgs.begin(); it != vtImgs.end();)
  246. {
  247. BOOL bExsit = FALSE;
  248. for (STR_VEC::iterator itExt = vtCopyExt.begin(); itExt != vtCopyExt.end(); itExt++)
  249. {
  250. if (match(itExt->c_str(), it->c_str())) {
  251. bExsit = TRUE;
  252. break;
  253. }
  254. }
  255. if (!bExsit) {
  256. it = vtImgs.erase(it);
  257. continue;
  258. }
  259. if (bPickoutThumbnail)
  260. {// 这里应该去除"s"和"m"开头的缩略图;// 但新的做法是在订单里创建缩略图子文件夹;// 总而言之,都要去掉缩略图;
  261. nIndex = it->find_last_of(_T('\\'));
  262. if (nIndex != TString::npos) {
  263. strtmp = it->substr(nIndex + 1);
  264. if (strtmp[0] == _T('s') || strtmp[0] == _T('m')) {
  265. it = vtImgs.erase(it);
  266. continue;
  267. }
  268. }
  269. }
  270. it++;
  271. }
  272. }
  273. /************************************************************************/
  274. /* 函数:SubgroupExt[6/1/2016 IT];
  275. /* 描述:将指定的扩展名文件分离出来;
  276. /* 参数:;
  277. /* [IN] vtAllImgs:源,要被分离指定扩展名的文件路径数组;
  278. /* [IN] lpSubgroupExt:要分离出来的扩展名;
  279. /* [OUT] AryImgs:返回分离出来的文件路径数组;
  280. /* 返回:void;
  281. /* 注意:;
  282. /* 示例:;
  283. /*
  284. /* 修改:;
  285. /* 日期:;
  286. /* 内容:;
  287. /************************************************************************/
  288. void IMGCommon::SubgroupExt(IN STR_VEC &vtAllImgs, IN LPCTSTR lpSubgroupExt, OUT CStringArray &AryImgs)
  289. {
  290. if (vtAllImgs.size() == 0 || lpSubgroupExt == NULL) return;
  291. // 获取复制扩展名;
  292. int nIndex = 0;
  293. TString strtmp;
  294. TString strSubgroupExt(lpSubgroupExt);
  295. if (strSubgroupExt.find(_T("*.*")) != TString::npos)
  296. {
  297. for (STR_VEC::iterator it = vtAllImgs.begin(); it != vtAllImgs.end(); it++)
  298. AryImgs.Add(CString(it->c_str()));
  299. return;
  300. }
  301. strSubgroupExt.append(_T("|"));
  302. STR_VEC vtSubgroupExt;
  303. // 将所有扩展名解析到数组里;
  304. do
  305. {
  306. nIndex = strSubgroupExt.find(_T('|'));
  307. if (nIndex != TString::npos)
  308. {
  309. strtmp = strSubgroupExt.substr(0, nIndex);
  310. strSubgroupExt = strSubgroupExt.substr(nIndex + 1);
  311. if (strtmp.compare(_T("*.*")))
  312. vtSubgroupExt.push_back(strtmp);
  313. }
  314. } while (strSubgroupExt.find(_T('|')) != TString::npos);
  315. // 过滤非复制扩展名的文件,同时去掉缩略图,保留指定复制扩展名的文件;
  316. for (STR_VEC::iterator it = vtAllImgs.begin(); it != vtAllImgs.end(); it++)
  317. {
  318. BOOL bExsit = FALSE;
  319. for (STR_VEC::iterator itExt = vtSubgroupExt.begin(); itExt != vtSubgroupExt.end(); itExt++)
  320. {
  321. if (match(itExt->c_str(), it->c_str()))
  322. {
  323. #if 1// 这里应该去除"s"和"m"开头的缩略图;// 但新的做法是在订单里创建缩略图子文件夹;// 总而言之,都要去掉缩略图;
  324. nIndex = it->find_last_of(_T('\\'));
  325. if (nIndex != TString::npos)
  326. {
  327. strtmp = it->substr(nIndex + 1);
  328. if (strtmp[0] == _T('s') || strtmp[0] == _T('m'))
  329. {
  330. continue;
  331. }
  332. }
  333. #endif
  334. AryImgs.Add(CString(it->c_str()));
  335. break;
  336. }
  337. }
  338. }
  339. }
  340. void IMGCommon::SubgroupExt(IN STR_VEC &vtAllImgs, IN LPCTSTR lpSubgroupExt, IN STR_VEC &vtImgs)
  341. {
  342. if (vtAllImgs.size() == 0 || lpSubgroupExt == NULL) return;
  343. // 获取复制扩展名;
  344. int nIndex = 0;
  345. TString strtmp;
  346. TString strSubgroupExt(lpSubgroupExt);
  347. if (strSubgroupExt.find(_T("*.*")) != TString::npos)
  348. {
  349. for (STR_VEC::iterator it = vtAllImgs.begin(); it != vtAllImgs.end(); it++)
  350. vtImgs.push_back(*it);
  351. return;
  352. }
  353. strSubgroupExt.append(_T("|"));
  354. STR_VEC vtSubgroupExt;
  355. // 将所有扩展名解析到数组里;
  356. do
  357. {
  358. nIndex = strSubgroupExt.find(_T('|'));
  359. if (nIndex != TString::npos)
  360. {
  361. strtmp = strSubgroupExt.substr(0, nIndex);
  362. strSubgroupExt = strSubgroupExt.substr(nIndex + 1);
  363. if (strtmp.compare(_T("*.*")))
  364. vtSubgroupExt.push_back(strtmp);
  365. }
  366. } while (strSubgroupExt.find(_T('|')) != TString::npos);
  367. // 过滤非复制扩展名的文件,同时去掉缩略图,保留指定复制扩展名的文件;
  368. for (STR_VEC::iterator it = vtAllImgs.begin(); it != vtAllImgs.end(); it++)
  369. {
  370. BOOL bExsit = FALSE;
  371. for (STR_VEC::iterator itExt = vtSubgroupExt.begin(); itExt != vtSubgroupExt.end(); itExt++)
  372. {
  373. if (match(itExt->c_str(), it->c_str())) {
  374. #if 1// 这里应该去除"s"和"m"开头的缩略图;// 但新的做法是在订单里创建缩略图子文件夹;// 总而言之,都要去掉缩略图;
  375. nIndex = it->find_last_of(_T('\\'));
  376. if (nIndex != TString::npos) {
  377. strtmp = it->substr(nIndex + 1);
  378. if (strtmp[0] == _T('s') || strtmp[0] == _T('m')) {
  379. continue;
  380. }
  381. }
  382. #endif
  383. vtImgs.push_back(*it);
  384. break;
  385. }
  386. }
  387. }
  388. }
  389. //////////////////////////////////////////////////////////////////////////
  390. BOOL IMGCommon::LoadImgFromFile(IN Image** pImg, LPCTSTR lpPath)
  391. {
  392. if ( !PathFileExists(lpPath) )
  393. return FALSE;
  394. if ( *pImg )
  395. delete *pImg;
  396. *pImg = NULL;
  397. #ifdef UNICODE
  398. *pImg = Image::FromFile(lpPath);
  399. #else
  400. BSTR strtmp = _bstr_t(lpPath);
  401. *pImg = Image::FromFile(strtmp, TRUE);
  402. SysFreeString(strtmp);
  403. #endif
  404. return (*pImg ? TRUE : FALSE);
  405. }
  406. BOOL IMGCommon::LoadImgFromBuffer(IN Image** pImg, IN BYTE* pBuffer, IN CONST INT& nBufLen)
  407. {
  408. if ( pBuffer == NULL )
  409. return FALSE;
  410. if ( *pImg )
  411. delete *pImg;
  412. *pImg = NULL;
  413. HGLOBAL hMemery = GlobalAlloc(GMEM_MOVEABLE, nBufLen);
  414. if ( hMemery == NULL )
  415. return FALSE;
  416. BYTE *pMem = (BYTE*)GlobalLock(hMemery);
  417. memcpy(pMem, pBuffer, nBufLen);
  418. IStream *pstream = NULL;
  419. CreateStreamOnHGlobal(hMemery, TRUE, &pstream);
  420. *pImg = Image::FromStream(pstream);
  421. GlobalUnlock(hMemery);
  422. pstream->Release();
  423. return (*pImg ? TRUE : FALSE);
  424. }
  425. // 先以只读方式从文件中读取二进制流出来,可以做到不独占文件;
  426. BOOL IMGCommon::LoadImgFromBuffer(IN Image** pImg, LPCTSTR lpPath)
  427. {
  428. if ( !PathFileExists(lpPath) )
  429. return FALSE;
  430. if ( *pImg )
  431. delete *pImg;
  432. *pImg = NULL;
  433. return LoadImgFromFile(pImg, lpPath);
  434. CFile fp;
  435. CFileException e;
  436. BOOL bRet = FALSE;
  437. if ( fp.Open(lpPath, CFile::modeRead, &e))
  438. {
  439. DWORD dwLength = (DWORD)fp.GetLength();
  440. BYTE *pData = new BYTE[dwLength];
  441. fp.Read(pData,dwLength);
  442. fp.Close();
  443. bRet = LoadImgFromBuffer(pImg, pData, dwLength);
  444. if( pData )
  445. delete []pData;
  446. }
  447. return bRet;
  448. }
  449. /************************************************************************/
  450. /* 函数:LoadImgFromResource[9/21/2016 IT];
  451. /* 描述:从资源中加载;
  452. /* 参数:;
  453. /* [IN] :;
  454. /* [OUT] :;
  455. /* [IN/OUT] :;
  456. /* 返回:void;
  457. /* 注意:;
  458. /* 示例:;
  459. /*
  460. /* 修改:;
  461. /* 日期:;
  462. /* 内容:;
  463. /************************************************************************/
  464. Image* IMGCommon::LoadImgFromResource(IN HMODULE hModule, IN LPCTSTR lpName, IN LPCTSTR lpType)
  465. {
  466. HGLOBAL hGlobal = NULL;
  467. HRSRC hSource = NULL;
  468. LPVOID lpBuffer = NULL;
  469. DWORD dwSize = 0;
  470. // 1.定位我们的自定义资源,这里因为我们是从本模块定位资源,所以将句柄简单地置为NULL即可
  471. hSource = FindResource(hModule, lpName, lpType);
  472. if (hSource == NULL)
  473. {
  474. _tprintf(_T("载入资源失败:%s"), lpName);
  475. return NULL;
  476. }
  477. // 2.获取资源的大小;
  478. dwSize = (UINT)SizeofResource(NULL, hSource);
  479. // 3.加载资源;
  480. hGlobal = LoadResource(NULL, hSource);
  481. if (hGlobal == NULL)
  482. {
  483. _tprintf(_T("载入资源失败:%s"), lpName);
  484. return NULL;
  485. }
  486. // 4.锁定资源,获取buffer;
  487. lpBuffer = LockResource(hGlobal);
  488. if (lpBuffer == NULL)
  489. {
  490. _tprintf(_T("载入资源失败:%s"), lpName);
  491. return NULL;
  492. }
  493. // lpFileFullName需要先判断文件是否存在??不需要;
  494. Image *pImg = NULL;
  495. LoadImgFromBuffer(&pImg, (BYTE*)lpBuffer, dwSize);
  496. UnlockResource(hGlobal);
  497. FreeResource(hGlobal);
  498. return pImg;
  499. }
  500. BOOL IMGCommon::GetOrientation(IN Image *pImg)
  501. {
  502. if (pImg == NULL) return FALSE;
  503. UINT totalBufferSize;
  504. UINT numProperties;
  505. pImg->GetPropertySize(&totalBufferSize, &numProperties);
  506. // Allocate the buffer that will receive the property items.
  507. PropertyItem* pAllItems = (PropertyItem*)malloc(totalBufferSize);
  508. // Fill the buffer.
  509. pImg->GetAllPropertyItems(totalBufferSize, numProperties, pAllItems);
  510. // Print the id data member of each property item.
  511. for (UINT j = 0; j < numProperties; ++j)
  512. {
  513. if (PropertyTagOrientation == pAllItems[j].id)
  514. {
  515. short* ptrLong = (short*)(pAllItems[j].value);
  516. int ret = (int)*ptrLong;
  517. free(pAllItems);
  518. return ret;
  519. }
  520. }
  521. free(pAllItems);
  522. return TRUE;
  523. }
  524. /************************************************************************/
  525. /*
  526. 函数:GetEncoderClsid
  527. 描述:获取GDI+支持的图像格式编码器种类,以及所有种类编码器信息;
  528. 参数:
  529. IN: format 要获取的图像格式;
  530. OUT: pClsid 返回符合条件的图像编码器信息;
  531. 返回:成功返回编码器索引,否则返回-1;
  532. */
  533. /************************************************************************/
  534. int IMGCommon::GetEncoderClsid(IN CONST WCHAR* format, OUT CLSID* pClsid)
  535. {
  536. // GDI+支持的图像编码器数量;
  537. UINT numEncoders = 0;
  538. // GDI+所有图像格式编码器详细信息所需要的空间大小;
  539. UINT nSize = 0;
  540. ImageCodecInfo* pImageCodecInfo = NULL;
  541. // 2.获取GDI+支持的所有图像格式编码器详细信息所需要的空间大小;
  542. GetImageEncodersSize(&numEncoders, &nSize);
  543. if (nSize == 0)
  544. return -1;
  545. //3.为ImageCodecInfo数组分配足额空间;
  546. pImageCodecInfo = (ImageCodecInfo*)(malloc(nSize));
  547. if (pImageCodecInfo == NULL)
  548. return -1;
  549. //4.获取所有的图像编码器信息;
  550. GetImageEncoders(numEncoders, nSize, pImageCodecInfo);
  551. //5.查找符合的图像编码器的Clsid;
  552. for (UINT i = 0; i < numEncoders; ++i)
  553. {
  554. if (wcscmp(pImageCodecInfo[i].MimeType, format) == 0)
  555. {
  556. *pClsid = pImageCodecInfo[i].Clsid;
  557. free(pImageCodecInfo);
  558. return i; // Success
  559. }
  560. }
  561. //6.释放步骤3分配的内存;
  562. free(pImageCodecInfo);
  563. return -1;
  564. }
  565. /************************************************************************/
  566. /*
  567. 函数:SaveImg2newfile
  568. 描述:将Image对象另存为其他格式的文件;
  569. 参数:
  570. IN: pImg GDI+ Image对象指针,需要保存的图像对象;
  571. IN: lpnewfile 其他格式的新文件名(jpg,bmp,png,jpeg);
  572. IN: uQuality 另存为新文件时的图像质量值;
  573. 返回:
  574. 注意:Image::Save 若文件存在,会覆盖存在的文件;
  575. */
  576. /************************************************************************/
  577. BOOL IMGCommon::SaveImg2newfile(IN Image* pImg, IN CString strNewfile, IN ULONG uQuality)
  578. {
  579. if (pImg == NULL) return FALSE;
  580. // 需要判断路径是否存在,不存在创建目录;
  581. int nIndex = strNewfile.ReverseFind(_T('\\'));
  582. if (nIndex == -1)
  583. return FALSE;
  584. if (!PathFileExists(strNewfile.Left(nIndex)))
  585. {
  586. // 如果文件夹不存在,创建;
  587. SHCreateDirectoryEx(NULL, strNewfile.Left(nIndex), NULL);
  588. }
  589. nIndex = strNewfile.ReverseFind(_T('.'));
  590. if (nIndex == -1)
  591. return FALSE;
  592. Status stat = GenericError;
  593. CLSID encoderClsid = { 0 };
  594. BSTR newfile = strNewfile.AllocSysString();
  595. strNewfile = strNewfile.Mid(nIndex + 1);
  596. if (strNewfile.CompareNoCase(_T("bmp")) == 0)
  597. {
  598. GetEncoderClsid(L"image/bmp", &encoderClsid);
  599. stat = pImg->Save(newfile, &encoderClsid, NULL);
  600. }
  601. else if (strNewfile.CompareNoCase(_T("png")) == 0)
  602. {
  603. GetEncoderClsid(L"image/png", &encoderClsid);
  604. stat = pImg->Save(newfile, &encoderClsid, NULL);
  605. }
  606. else// if ( strNewfile.CompareNoCase(_T("jpeg")) == 0 )
  607. {
  608. GetEncoderClsid(L"image/jpeg", &encoderClsid);
  609. EncoderParameters encoderParameters;
  610. encoderParameters.Count = 1;
  611. encoderParameters.Parameter[0].Guid = EncoderQuality;
  612. encoderParameters.Parameter[0].Type = EncoderParameterValueTypeLong;
  613. encoderParameters.Parameter[0].NumberOfValues = 1;
  614. // Save the image as a JPEG with quality level 100.
  615. encoderParameters.Parameter[0].Value = &uQuality;
  616. stat = pImg->Save(newfile, &encoderClsid, &encoderParameters);
  617. }
  618. SysFreeString(newfile);
  619. return stat == Ok ? TRUE : FALSE;
  620. }
  621. BOOL IMGCommon::SaveImg2newfile(IN Image* pImg, IN TString strNewfile, IN ULONG uQuality)
  622. {
  623. if (pImg == NULL) return FALSE;
  624. // 需要判断路径是否存在,不存在创建目录;
  625. int nIndex = strNewfile.find_last_of(_T('\\'));
  626. if (nIndex == TString::npos)
  627. return FALSE;
  628. if (!PathFileExists(strNewfile.substr(0, nIndex).c_str()))
  629. {
  630. // 如果文件夹不存在,创建;
  631. SHCreateDirectoryEx(NULL, strNewfile.substr(0, nIndex).c_str(), NULL);
  632. }
  633. nIndex = strNewfile.find_last_of(_T('.'));
  634. if (nIndex == TString::npos)
  635. return FALSE;
  636. Status stat = GenericError;
  637. CLSID encoderClsid = { 0 };
  638. BSTR newfile = _bstr_t(strNewfile.c_str());
  639. strNewfile = strNewfile.substr(nIndex + 1);
  640. if (strNewfile.find(_T("bmp")) == 0)
  641. {
  642. GetEncoderClsid(L"image/bmp", &encoderClsid);
  643. stat = pImg->Save(newfile, &encoderClsid, NULL);
  644. }
  645. else if (strNewfile.find(_T("png")) == 0)
  646. {
  647. GetEncoderClsid(L"image/png", &encoderClsid);
  648. stat = pImg->Save(newfile, &encoderClsid, NULL);
  649. }
  650. else// if ( strNewfile.CompareNoCase(_T("jpeg")) == 0 )
  651. {
  652. GetEncoderClsid(L"image/jpeg", &encoderClsid);
  653. EncoderParameters encoderParameters;
  654. encoderParameters.Count = 1;
  655. encoderParameters.Parameter[0].Guid = EncoderQuality;
  656. encoderParameters.Parameter[0].Type = EncoderParameterValueTypeLong;
  657. encoderParameters.Parameter[0].NumberOfValues = 1;
  658. // Save the image as a JPEG with quality level 100.
  659. encoderParameters.Parameter[0].Value = &uQuality;
  660. stat = pImg->Save(newfile, &encoderClsid, &encoderParameters);
  661. }
  662. SysFreeString(newfile);
  663. return stat == Ok ? TRUE : FALSE;
  664. }
  665. /************************************************************************/
  666. /*
  667. 函数:ZoomImg
  668. 描述:缩放到指定大小的区域中,返回缩放后的尺寸;
  669. 参数:
  670. IN: Imgrc 图像的大小;
  671. IN: dwDisplayPix 图像要显示的区域的大小;
  672. OUT: dwZoomPix 图像在显示区域内缩放后的尺寸;
  673. 返回:
  674. 注意:dwZoomPix的尺寸必须在dwDisplayPix内;
  675. */
  676. /************************************************************************/
  677. int IMGCommon::ZoomImg(IN CRect &Imgrc, IN CONST DWORD &dwDisplayPix, OUT DWORD &dwZoomPix)
  678. {
  679. double fDisplayWidth = GET_XPIX(dwDisplayPix);
  680. double fDisplayHeight = GET_YPIX(dwDisplayPix);
  681. // 显示区域长宽比;
  682. double fDisplayAspectRatio = fDisplayWidth / fDisplayHeight;
  683. // 图片长宽比;
  684. double fImgAspectRatio = ((double)Imgrc.Width()) / ((double)Imgrc.Height());
  685. double fZoomWidth;
  686. double fZoomHeight;
  687. if (fDisplayAspectRatio > fImgAspectRatio)
  688. {
  689. fZoomWidth = fDisplayHeight*fImgAspectRatio;
  690. fZoomHeight = fDisplayHeight;
  691. }
  692. else
  693. {
  694. fZoomWidth = fDisplayWidth;
  695. fZoomHeight = fDisplayWidth / fImgAspectRatio;
  696. }
  697. dwZoomPix = SET_PIX((int)fZoomWidth, (int)fZoomHeight);
  698. //////////////////////////////////////////////////////////////////////////
  699. int nRetval = 0;
  700. if ((fDisplayWidth == Imgrc.Width()) && (fDisplayHeight == Imgrc.Height()))
  701. {
  702. nRetval = ZoomNull;
  703. }
  704. else if ((fDisplayWidth > Imgrc.Width()) && (fDisplayHeight > Imgrc.Height()))
  705. {
  706. nRetval = ZoomOut;
  707. }
  708. else
  709. {
  710. nRetval = ZoomIn;
  711. }
  712. return nRetval;
  713. }
  714. int IMGCommon::ZoomImg(IN CONST DWORD &dwImgPix, IN CONST DWORD &dwDisplayPix, OUT DWORD &dwZoomPix)
  715. {
  716. double fDisplayWidth = GET_XPIX(dwDisplayPix);
  717. double fDisplayHeight = GET_YPIX(dwDisplayPix);
  718. // 显示区域长宽比;
  719. double fDisplayAspectRatio = fDisplayWidth / fDisplayHeight;
  720. // 图片长宽比;
  721. double fImgAspectRatio = ((double)GET_XPIX(dwImgPix)) / ((double)GET_YPIX(dwImgPix));
  722. double fZoomWidth;
  723. double fZoomHeight;
  724. if (fDisplayAspectRatio > fImgAspectRatio)
  725. {
  726. fZoomWidth = fDisplayHeight*fImgAspectRatio;
  727. fZoomHeight = fDisplayHeight;
  728. }
  729. else
  730. {
  731. fZoomWidth = fDisplayWidth;
  732. fZoomHeight = fDisplayWidth / fImgAspectRatio;
  733. }
  734. dwZoomPix = SET_PIX((int)fZoomWidth, (int)fZoomHeight);
  735. //////////////////////////////////////////////////////////////////////////
  736. int nRetval = 0;
  737. if ((fDisplayWidth == GET_XPIX(dwImgPix)) && (fDisplayHeight == GET_YPIX(dwImgPix)))
  738. {
  739. nRetval = ZoomNull;
  740. }
  741. else if ((fDisplayWidth > GET_XPIX(dwImgPix)) && (fDisplayHeight > GET_YPIX(dwImgPix)))
  742. {
  743. nRetval = ZoomOut;
  744. }
  745. else
  746. {
  747. nRetval = ZoomIn;
  748. }
  749. return nRetval;
  750. }
  751. /************************************************************************/
  752. /*
  753. 函数:ImgThumbnail
  754. 描述:生成高质量的图像缩略图;
  755. 参数:
  756. 返回:
  757. 注意:为了健壮性,strNewfile的路径可以在函数内再次判断是否有效,防止在函数外没有进行经判断;
  758. */
  759. /************************************************************************/
  760. BOOL IMGCommon::ImgThumbnail(IN Image* pImg, IN CONST DWORD &dwDisplayPix, IN CString strNewfile, IN ULONG uQuality)
  761. {
  762. if (pImg == NULL) return FALSE;
  763. DWORD dwZoomPix;
  764. int retval = ZoomImg(SET_PIX(pImg->GetWidth(), pImg->GetHeight()), dwDisplayPix, dwZoomPix);
  765. if (retval == ZoomOut)
  766. {
  767. WriteTextLog(_T("复制时放大图像,退出复制!"));
  768. return FALSE;
  769. }
  770. else if (retval == ZoomNull)
  771. {
  772. return SaveImg2newfile(pImg, strNewfile, uQuality);
  773. }
  774. // 绘制缩略图;
  775. Bitmap bitmap(GET_XPIX(dwZoomPix), GET_YPIX(dwZoomPix));
  776. Graphics *graphic = Graphics::FromImage(&bitmap);
  777. graphic->Clear(Color(255, 255, 255, 255));
  778. // 设置缩放或旋转图片时的算法(图片质量);
  779. graphic->SetInterpolationMode(InterpolationModeHighQualityBicubic);
  780. // 设置渲染图形对象的质量;
  781. graphic->SetSmoothingMode(SmoothingModeHighQuality);//SmoothingModeHighSpeed
  782. // 设置图形对象的像素偏移模式;
  783. graphic->SetPixelOffsetMode(PixelOffsetModeHighQuality);
  784. // 设置图形对象的合成模式;
  785. graphic->SetCompositingMode(CompositingModeSourceOver);
  786. // 设置图形对象的合成质量;
  787. graphic->SetCompositingQuality(CompositingQualityHighQuality);
  788. // 设置图形对象的文本渲染模式;
  789. graphic->SetTextRenderingHint(TextRenderingHintAntiAliasGridFit);
  790. graphic->DrawImage(pImg, 0, 0, GET_XPIX(dwZoomPix), GET_YPIX(dwZoomPix));
  791. delete graphic;
  792. // 生成缩略图;
  793. return SaveImg2newfile(&bitmap, strNewfile, uQuality);
  794. }
  795. BOOL IMGCommon::ImgThumbnail(IN LPCTSTR lpImgpath, IN CONST DWORD &dwDisplayPix, IN CString strNewfile, IN ULONG uQuality)
  796. {
  797. if (!PathFileExists(lpImgpath))
  798. return FALSE;
  799. #ifdef UNICODE
  800. Image Img(lpImgpath);
  801. #else
  802. BSTR strtmp = _bstr_t(lpImgpath);
  803. Image Img(strtmp);
  804. SysFreeString(strtmp);
  805. #endif
  806. DWORD dwZoomPix;
  807. int retval = ZoomImg(SET_PIX(Img.GetWidth(), Img.GetHeight()), dwDisplayPix, dwZoomPix);
  808. if (retval == ZoomOut)
  809. {
  810. WriteTextLog(_T("复制时放大图像,退出复制!"));
  811. return FALSE;
  812. }
  813. else if (retval == ZoomNull)
  814. {
  815. return SaveImg2newfile(&Img, strNewfile, uQuality);
  816. }
  817. // 绘制缩略图;
  818. Bitmap bitmap(GET_XPIX(dwZoomPix), GET_YPIX(dwZoomPix));
  819. Graphics *graphic = Graphics::FromImage(&bitmap);
  820. graphic->Clear(Color(255, 255, 255, 255));
  821. // 设置缩放或旋转图片时的算法(图片质量);
  822. graphic->SetInterpolationMode(InterpolationModeHighQualityBicubic);
  823. // 设置渲染图形对象的质量;
  824. graphic->SetSmoothingMode(SmoothingModeHighQuality);//SmoothingModeHighSpeed
  825. // 设置图形对象的像素偏移模式;
  826. graphic->SetPixelOffsetMode(PixelOffsetModeHighQuality);
  827. // 设置图形对象的合成模式;
  828. graphic->SetCompositingMode(CompositingModeSourceOver);
  829. // 设置图形对象的合成质量;
  830. graphic->SetCompositingQuality(CompositingQualityHighQuality);
  831. // 设置图形对象的文本渲染模式;
  832. graphic->SetTextRenderingHint(TextRenderingHintAntiAliasGridFit);
  833. Status st = graphic->DrawImage(&Img, 0, 0, GET_XPIX(dwZoomPix), GET_YPIX(dwZoomPix));
  834. if ( st != Ok )
  835. {
  836. delete graphic;
  837. return FALSE;
  838. }
  839. delete graphic;
  840. // 生成缩略图;
  841. return SaveImg2newfile(&bitmap, strNewfile, uQuality);
  842. }
  843. BOOL IMGCommon::ImgThumbnail(IN CONST TString &strImgpath, IN CONST DWORD &dwDisplayPix, IN TString strNewfile, IN ULONG uQuality)
  844. {
  845. if (!PathFileExists(strImgpath.c_str()))
  846. {
  847. SetLastError(GetLastError());
  848. return FALSE;
  849. }
  850. #ifdef UNICODE
  851. BSTR strtmp = _bstr_t(strImgpath.c_str());
  852. Image Img(strtmp);
  853. SysFreeString(strtmp);
  854. //Image Img(strImgpath);
  855. #else
  856. BSTR strtmp = _bstr_t(strImgpath.c_str());
  857. Image Img(strtmp);
  858. SysFreeString(strtmp);
  859. #endif
  860. DWORD dwZoomPix;
  861. int retval = ZoomImg(SET_PIX(Img.GetWidth(), Img.GetHeight()), dwDisplayPix, dwZoomPix);
  862. if (retval == ZoomOut)
  863. {
  864. WriteTextLog(_T("复制时放大图像,退出复制!"));
  865. return FALSE;
  866. }
  867. else if (retval == ZoomNull)
  868. {
  869. return SaveImg2newfile(&Img, strNewfile, uQuality);
  870. }
  871. // 绘制缩略图;
  872. Bitmap bitmap(GET_XPIX(dwZoomPix), GET_YPIX(dwZoomPix));
  873. Graphics *graphic = Graphics::FromImage(&bitmap);
  874. graphic->Clear(Color(255, 255, 255, 255));
  875. // 设置缩放或旋转图片时的算法(图片质量);
  876. graphic->SetInterpolationMode(InterpolationModeHighQualityBicubic);
  877. // 设置渲染图形对象的质量;
  878. graphic->SetSmoothingMode(SmoothingModeHighQuality);//SmoothingModeHighSpeed
  879. // 设置图形对象的像素偏移模式;
  880. graphic->SetPixelOffsetMode(PixelOffsetModeHighQuality);
  881. // 设置图形对象的合成模式;
  882. graphic->SetCompositingMode(CompositingModeSourceOver);
  883. // 设置图形对象的合成质量;
  884. graphic->SetCompositingQuality(CompositingQualityHighQuality);
  885. // 设置图形对象的文本渲染模式;
  886. graphic->SetTextRenderingHint(TextRenderingHintAntiAliasGridFit);
  887. Status st = graphic->DrawImage(&Img, 0, 0, GET_XPIX(dwZoomPix), GET_YPIX(dwZoomPix));
  888. if ( st != Ok )
  889. {
  890. delete graphic;
  891. return FALSE;
  892. }
  893. delete graphic;
  894. // 生成缩略图;
  895. return SaveImg2newfile(&bitmap, strNewfile, uQuality);
  896. }
  897. BOOL IMGCommon::SimpleImgThumbnail(IN LPCTSTR lpImgpath, IN CONST DWORD &dwDisplayPix, IN CString strNewfile, IN ULONG uQuality)
  898. {
  899. if (!PathFileExists(lpImgpath))
  900. return FALSE;
  901. DWORD dwZoomPix;
  902. Image *pImg = NULL;
  903. LoadImgFromFile(&pImg, lpImgpath);
  904. if ( pImg == NULL )
  905. {
  906. return FALSE;
  907. }
  908. int retval = ZoomImg(SET_PIX(pImg->GetWidth(), pImg->GetHeight()), dwDisplayPix, dwZoomPix);
  909. if (retval == ZoomOut)
  910. {
  911. WriteTextLog(_T("复制时放大图像,退出复制!"));
  912. if (pImg) delete pImg;
  913. return FALSE;
  914. }
  915. else if (retval == ZoomNull)
  916. {
  917. BOOL bRet = SaveImg2newfile(pImg, strNewfile, uQuality);
  918. if (pImg) delete pImg;
  919. return bRet;
  920. }
  921. Image *pThumbnailImg = pImg->GetThumbnailImage(GET_XPIX(dwZoomPix), GET_YPIX(dwZoomPix));
  922. if (pThumbnailImg)
  923. {
  924. // 提高缩略图质量;
  925. Graphics graphic(pThumbnailImg);
  926. graphic.DrawImage(pImg, 0, 0, pThumbnailImg->GetWidth(), pThumbnailImg->GetHeight());
  927. INT nRientation = GetOrientation(pImg);
  928. if ( nRientation == 8 )
  929. {
  930. pThumbnailImg->RotateFlip(Rotate270FlipNone);
  931. }
  932. else if (nRientation == 6)
  933. {
  934. pThumbnailImg->RotateFlip(Rotate90FlipNone);
  935. }
  936. SaveImg2newfile(pThumbnailImg, strNewfile, uQuality);
  937. }
  938. else
  939. {
  940. if (pImg) delete pImg;
  941. return FALSE;
  942. }
  943. if (pImg) delete pImg;
  944. if (pThumbnailImg) delete pThumbnailImg;
  945. return TRUE;
  946. }