Picture.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590
  1. //-----------------------------------------------------------------------------
  2. // Picture (Implementations) Version 1.00
  3. //
  4. // Routins 4 Showing Picture Files... (.BMP .DIB .EMF .GIF .ICO .JPG .WMF)
  5. //
  6. // Author: Dr. Yovav Gad, EMail: Sources@SuperMain.com ,Web: www.SuperMain.com
  7. //=============================================================================
  8. //
  9. // Full Story:
  10. // ~~~~~~~~~~~
  11. // There R Many Libraries To Handle Image Files, Anyway Most Of Them Do Not
  12. // Include Source Files Or Just Very Complicated To Implement / Understand,
  13. //
  14. // After Many Days Of Searching (And Not Finding) a Way To Load a JPG From a
  15. // Resource And Show It On a *Dialog Based* Application, I Decided 2 Take Steps
  16. //
  17. // So I Created What I Call a Very *Simple* & Useful Class,
  18. // It Can Easily Implemented By Adding It To a Project, And U Do Not Have To
  19. // Be a Real JPEG Freak - And Invent All Header Reading From The Beginning
  20. // (It Uses The IPicture Interface - Same Way As Internet Explorer Does)
  21. //
  22. // I Would Like To Thank Mr.Peter Hendrix For His Wonderful Work
  23. // That I Found On: http://www.thecodeproject.com/bitmap/cpicture.asp
  24. // Which I Was Inspired And Got The IPicture Interface Idea From...
  25. //
  26. // Guess U Will Find It Useful,
  27. // Appreciate If U Can Mention My Name On Your Final Code,
  28. // Please Feel Free To Send Me Any Improvements Or SaveAsJPG() Functions:
  29. //
  30. // (This Program Has No Bugs - Only Undocumented Solutions)
  31. //
  32. // Author: Dr. Yovav Gad, EMail: Sources@SuperMain.com ,Web: www.SuperMain.com
  33. //
  34. //--------------------------Example & Usage 4 Dummies--------------------------
  35. //
  36. // U Need 2 Add "CPicture.CPP" and "CPicture.H" Into Your Project (From FileView)
  37. // So U Will Get Control Over The Functions In This Class,
  38. // Then U Can Create a Picture Object And Show It On a Device Context
  39. //
  40. // CPicture m_Picture; // Create a Picture Object (An Instance Of This Class)
  41. // #include "Picture.h" // Make Sure U Include This Where U Gonna Create The Object...
  42. //
  43. // Load Picture Data Into The IPicture Interface (.BMP .DIB .EMF .GIF .ICO .JPG .WMF)
  44. // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  45. // m_Picture.Load("Test.JPG"); // Load From a File - Just Load It (Show Later)
  46. // m_Picture.Load(IDR_TEST, "JPG"); // Load From a Resource - Just Load It (Show Later)
  47. // (U Must Include IDR_TEST In Your Resources Under a Custom Name, 4 Example - "JPG")
  48. //
  49. // When Using DC Object On a *Dialog Based* Application (CPaintDC dc(this);)
  50. // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  51. // m_Picture.UpdateSizeOnDC(&dc); // Get Picture Dimentions In Pixels
  52. // m_Picture.Show(&dc, CPoint(0,0), CPoint(m_Picture.m_Width, m_Picture.m_Height), 0,0);
  53. // m_Picture.Show(&dc, CRect(0,0,100,100)); // Change Original Dimentions
  54. // m_Picture.ShowBitmapResource(&dc, IDB_TEST, CPoint(0,0)); // Show Bitmap Resource
  55. //
  56. // OR When Using a Pointer On a "Regular" MFC Application (CDC* pDC)
  57. // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  58. // m_Picture.UpdateSizeOnDC(pDC); // Get Picture Dimentions In Pixels
  59. // m_Picture.Show(pDC, CPoint(0,0), CPoint(m_Picture.m_Width, m_Picture.m_Height), 0,0);
  60. // m_Picture.Show(pDC, CRect(0,0,100,100)); // Change Original Dimentions
  61. // m_Picture.ShowBitmapResource(pDC, IDB_TEST, CPoint(0,0)); // Show Bitmap Resource
  62. //
  63. // Show Picture Information
  64. // ~~~~~~~~~~~~~~~~~~~~~~~~
  65. // CString S;
  66. // S.Format("Size = %4d\nWidth = %4d\nHeight = %4d\nWeight = %4d\n",
  67. // m_Picture.m_Weight, m_Picture.m_Width, m_Picture.m_Height, m_Picture.m_Weight);
  68. // AfxMessageBox(S);
  69. //
  70. //----------------------------Cut The Bullshit Here----------------------------
  71. #include "stdafx.h"
  72. #include "Picture.h"
  73. #ifdef _DEBUG
  74. #define new DEBUG_NEW
  75. #undef THIS_FILE
  76. static char THIS_FILE[] = __FILE__;
  77. #endif
  78. #define HIMETRIC_INCH 2540
  79. #define ERROR_TITLE "CPicture Error" // Error Title (Related To This Class)...
  80. //-----------------------------------------------------------------------------
  81. // Does: Constructor - Create a New CPicture Object To Hold Picture Data
  82. // ~~~~
  83. //
  84. //-----------------------------------------------------------------------------
  85. CPicture::CPicture()
  86. //=============================================================================
  87. {
  88. m_IPicture = NULL;
  89. m_Height = 0;
  90. m_Weight = 0;
  91. m_Width = 0;
  92. }
  93. //-----------------------------------------------------------------------------
  94. // Does: Destructor - Free Data And Information From The CPicture Object
  95. // ~~~~
  96. //
  97. //-----------------------------------------------------------------------------
  98. CPicture::~CPicture()
  99. //=============================================================================
  100. {
  101. if(m_IPicture != NULL) FreePictureData(); // Important - Avoid Leaks...
  102. }
  103. //-----------------------------------------------------------------------------
  104. // Does: Free The Allocated Memory That Holdes The IPicture Interface Data
  105. // ~~~~ And Clear Picture Information
  106. //
  107. // Note: This Might Also Be Useful If U Only Need To Show The Picture Once
  108. // ~~~~~ Or If U Copy The Picture To The Device Context, So It Can Still
  109. // Remain On Screen - But IPicture Data Is Not Needed No More
  110. //
  111. //-----------------------------------------------------------------------------
  112. void CPicture::FreePictureData()
  113. //=============================================================================
  114. {
  115. if(m_IPicture != NULL)
  116. {
  117. m_IPicture->Release();
  118. m_IPicture = NULL;
  119. m_Height = 0;
  120. m_Weight = 0;
  121. m_Width = 0;
  122. }
  123. }
  124. //-----------------------------------------------------------------------------
  125. // Does: Open a Resource And Load It Into IPicture (Interface)
  126. // ~~~~ (.BMP .DIB .EMF .GIF .ICO .JPG .WMF)
  127. //
  128. // Note: When Adding a Bitmap Resource It Would Automatically Show On "Bitmap"
  129. // ~~~~ This NOT Good Coz We Need To Load It From a Custom Resource "BMP"
  130. // To Add a Custom Rresource: Import Resource -> Open As -> Custom
  131. // (Both .BMP And .DIB Should Be Found Under "BMP")
  132. //
  133. // InPut: ResourceName - As a UINT Defined (Example: IDR_PICTURE_RESOURCE)
  134. // ~~~~~ ResourceType - Type Name (Example: "JPG")
  135. //
  136. // OutPut: TRUE If Succeeded...
  137. // ~~~~~~
  138. //-----------------------------------------------------------------------------
  139. BOOL CPicture::Load(UINT ResourceName, LPCSTR ResourceType)
  140. //=============================================================================
  141. {
  142. BOOL bResult = FALSE;
  143. HGLOBAL hGlobal = NULL;
  144. HRSRC hSource = NULL;
  145. LPVOID lpVoid = NULL;
  146. int nSize = 0;
  147. if(m_IPicture != NULL) FreePictureData(); // Important - Avoid Leaks...
  148. hSource = FindResource(AfxGetResourceHandle(), MAKEINTRESOURCE(ResourceName), ResourceType);
  149. if(hSource == NULL)
  150. {
  151. HWND hWnd = AfxGetApp()->GetMainWnd()->m_hWnd;
  152. MessageBoxEx(hWnd, "FindResource() Failed\t", ERROR_TITLE, MB_OK | MB_ICONSTOP, LANG_ENGLISH);
  153. return(FALSE);
  154. }
  155. hGlobal = LoadResource(AfxGetResourceHandle(), hSource);
  156. if(hGlobal == NULL)
  157. {
  158. HWND hWnd = AfxGetApp()->GetMainWnd()->m_hWnd;
  159. MessageBoxEx(hWnd, "LoadResource() Failed\t", ERROR_TITLE, MB_OK | MB_ICONSTOP, LANG_ENGLISH);
  160. return(FALSE);
  161. }
  162. lpVoid = LockResource(hGlobal);
  163. if(lpVoid == NULL)
  164. {
  165. HWND hWnd = AfxGetApp()->GetMainWnd()->m_hWnd;
  166. MessageBoxEx(hWnd, "LockResource() Failed\t", ERROR_TITLE, MB_OK | MB_ICONSTOP, LANG_ENGLISH);
  167. return(FALSE);
  168. }
  169. nSize = (UINT)SizeofResource(AfxGetResourceHandle(), hSource);
  170. if(LoadPictureData((BYTE*)hGlobal, nSize)) bResult = TRUE;
  171. UnlockResource(hGlobal); // 16Bit Windows Needs This
  172. FreeResource(hGlobal); // 16Bit Windows Needs This (32Bit - Automatic Release)
  173. m_Weight = nSize; // Update Picture Size Info...
  174. if(m_IPicture != NULL) // Do Not Try To Read From Memory That Is Not Exist...
  175. {
  176. m_IPicture->get_Height(&m_Height);
  177. m_IPicture->get_Width(&m_Width);
  178. // Calculate Its Size On a "Standard" (96 DPI) Device Context
  179. m_Height = MulDiv(m_Height, 96, HIMETRIC_INCH);
  180. m_Width = MulDiv(m_Width, 96, HIMETRIC_INCH);
  181. }
  182. else // Picture Data Is Not a Known Picture Type
  183. {
  184. m_Height = 0;
  185. m_Width = 0;
  186. bResult = FALSE;
  187. }
  188. return(bResult);
  189. }
  190. //-----------------------------------------------------------------------------
  191. // Does: Open a File And Load It Into IPicture (Interface)
  192. // ~~~~ (.BMP .DIB .EMF .GIF .ICO .JPG .WMF)
  193. //
  194. // InPut: sFilePathName - Path And FileName Target To Save
  195. // ~~~~~
  196. //
  197. // OutPut: TRUE If Succeeded...
  198. // ~~~~~~
  199. //-----------------------------------------------------------------------------
  200. BOOL CPicture::Load(CString sFilePathName)
  201. //=============================================================================
  202. {
  203. BOOL bResult = FALSE;
  204. CFile PictureFile;
  205. CFileException e;
  206. int nSize = 0;
  207. if(m_IPicture != NULL) FreePictureData(); // Important - Avoid Leaks...
  208. if(PictureFile.Open(sFilePathName, CFile::modeRead | CFile::typeBinary, &e))
  209. {
  210. nSize = PictureFile.GetLength();
  211. BYTE* pBuffer = new BYTE[nSize];
  212. if (PictureFile.Read(pBuffer, nSize) > 0 ) //从文件读到pBuffer
  213. { if(LoadPictureData(pBuffer, nSize)) bResult = TRUE; }//接作调用函数读pBuffer
  214. PictureFile.Close();
  215. delete [] pBuffer;
  216. }
  217. else // Open Failed...
  218. {
  219. TCHAR szCause[255];
  220. e.GetErrorMessage(szCause, 255, NULL);
  221. HWND hWnd = AfxGetApp()->GetMainWnd()->m_hWnd;
  222. MessageBoxEx(hWnd, szCause, ERROR_TITLE, MB_OK | MB_ICONSTOP, LANG_ENGLISH);
  223. bResult = FALSE;
  224. }
  225. m_Weight = nSize; // Update Picture Size Info...
  226. if(m_IPicture != NULL) // Do Not Try To Read From Memory That Is Not Exist...
  227. {
  228. m_IPicture->get_Height(&m_Height);
  229. m_IPicture->get_Width(&m_Width);
  230. // Calculate Its Size On a "Standard" (96 DPI) Device Context
  231. m_Height = MulDiv(m_Height, 96, HIMETRIC_INCH);
  232. m_Width = MulDiv(m_Width, 96, HIMETRIC_INCH);
  233. }
  234. else // Picture Data Is Not a Known Picture Type
  235. {
  236. m_Height = 0;
  237. m_Width = 0;
  238. bResult = FALSE;
  239. }
  240. return(bResult);
  241. }
  242. //-----------------------------------------------------------------------------
  243. // Does: Read The Picture Data From a Source (File / Resource)
  244. // ~~~~ And Load It Into The Current IPicture Object In Use
  245. //
  246. // InPut: Buffer Of Data Source (File / Resource) And Its Size
  247. // ~~~~~
  248. //
  249. // OutPut: Feed The IPicture Object With The Picture Data
  250. // ~~~~~~ (Use Draw Functions To Show It On a Device Context)
  251. // TRUE If Succeeded...
  252. //-----------------------------------------------------------------------------
  253. BOOL CPicture::LoadPictureData(BYTE *pBuffer, int nSize)
  254. //=============================================================================
  255. {
  256. BOOL bResult = FALSE;
  257. HGLOBAL hGlobal = GlobalAlloc(GMEM_MOVEABLE, nSize);
  258. if(hGlobal == NULL)
  259. {
  260. HWND hWnd = AfxGetApp()->GetMainWnd()->m_hWnd;
  261. MessageBoxEx(hWnd, "Can not allocate enough memory\t", ERROR_TITLE, MB_OK | MB_ICONSTOP, LANG_ENGLISH);
  262. return(FALSE);
  263. }
  264. void* pData = GlobalLock(hGlobal);
  265. memcpy(pData, pBuffer, nSize);
  266. GlobalUnlock(hGlobal);
  267. IStream* pStream = NULL;
  268. if(CreateStreamOnHGlobal(hGlobal, TRUE, &pStream) == S_OK)
  269. {
  270. HRESULT hr;
  271. if((hr = OleLoadPicture(pStream, nSize, FALSE, IID_IPicture, (LPVOID *)&m_IPicture)) == E_NOINTERFACE)
  272. {
  273. HWND hWnd = AfxGetApp()->GetMainWnd()->m_hWnd;
  274. MessageBoxEx(hWnd, "IPicture interface is not supported\t", ERROR_TITLE, MB_OK | MB_ICONSTOP, LANG_ENGLISH);
  275. return(FALSE);
  276. }
  277. else // S_OK
  278. {
  279. pStream->Release();
  280. pStream = NULL;
  281. bResult = TRUE;
  282. }
  283. }
  284. FreeResource(hGlobal); // 16Bit Windows Needs This (32Bit - Automatic Release)
  285. return(bResult);
  286. }
  287. //-----------------------------------------------------------------------------
  288. // Does: Draw The Loaded Picture Direct To The Client DC
  289. // ~~~~
  290. //
  291. // Note: Bigger OR Smaller Dimentions Than The Original Picture Size
  292. // ~~~~ Will Draw The Picture Streached To Its New Given NEW Dimentions...
  293. //
  294. // InPut: pDC - Given DC To Draw On
  295. // ~~~~~ DrawRect - Dimentions Of The Picture To Draw (As a Rectangle)
  296. //
  297. // OutPut: TRUE If Succeeded...
  298. // ~~~~~~
  299. //-----------------------------------------------------------------------------
  300. BOOL CPicture::Show(CDC *pDC, CRect DrawRect)
  301. //=============================================================================
  302. {
  303. if (pDC == NULL || m_IPicture == NULL) return FALSE;
  304. long Width = 0;
  305. long Height = 0;
  306. m_IPicture->get_Width(&Width);
  307. m_IPicture->get_Height(&Height);
  308. HRESULT hrP = NULL;
  309. hrP = m_IPicture->Render(pDC->m_hDC,
  310. DrawRect.left, // Left
  311. DrawRect.top, // Top
  312. DrawRect.right - DrawRect.left, // Right
  313. DrawRect.bottom - DrawRect.top, // Bottom
  314. 0,
  315. Height,
  316. Width,
  317. -Height,
  318. &DrawRect);
  319. if (SUCCEEDED(hrP)) return(TRUE);
  320. HWND hWnd = AfxGetApp()->GetMainWnd()->m_hWnd;
  321. MessageBoxEx(hWnd, "Can not allocate enough memory\t", ERROR_TITLE, MB_OK | MB_ICONSTOP, LANG_ENGLISH);
  322. return(FALSE);
  323. }
  324. //-----------------------------------------------------------------------------
  325. // Does: Draw The Loaded Picture Direct To The Client DC
  326. // ~~~~
  327. //
  328. // Note: Bigger OR Smaller Dimentions Than The Original Picture Size
  329. // ~~~~ Will Draw The Picture Streached To Its New Given Dimentions...
  330. //
  331. // InPut: pDC - Given DC To Draw On
  332. // ~~~~~ LeftTop - Opening Point To Start Drawing (Left,Top)
  333. // WidthHeight - Dimentions Of The Picture To Draw (Width,Height)
  334. // MagnifyX - Magnify Pixel Width, 0 = Default (No Magnify)
  335. // MagnifyY - Magnify Pixel Height, 0 = Default (No Magnify)
  336. //
  337. // OutPut: TRUE If Succeeded...
  338. // ~~~~~~
  339. //-----------------------------------------------------------------------------
  340. BOOL CPicture::Show(CDC *pDC, CPoint LeftTop, CPoint WidthHeight, int MagnifyX, int MagnifyY)
  341. //=============================================================================
  342. {
  343. if (pDC == NULL || m_IPicture == NULL) return FALSE;
  344. long Width = 0;
  345. long Height = 0;
  346. m_IPicture->get_Width(&Width);
  347. m_IPicture->get_Height(&Height);
  348. if(MagnifyX == NULL) MagnifyX = 0;
  349. if(MagnifyY == NULL) MagnifyY = 0;
  350. MagnifyX = int(MulDiv(Width, pDC->GetDeviceCaps(LOGPIXELSX), HIMETRIC_INCH) * MagnifyX);
  351. MagnifyY = int(MulDiv(Height,pDC->GetDeviceCaps(LOGPIXELSY), HIMETRIC_INCH) * MagnifyY);
  352. CRect DrawRect(LeftTop.x, LeftTop.y, MagnifyX, MagnifyY);
  353. HRESULT hrP = NULL;
  354. hrP = m_IPicture->Render(pDC->m_hDC,
  355. LeftTop.x, // Left
  356. LeftTop.y, // Top
  357. WidthHeight.x +MagnifyX, // Width
  358. WidthHeight.y +MagnifyY, // Height
  359. 0,
  360. Height,
  361. Width,
  362. -Height,
  363. &DrawRect);
  364. if(SUCCEEDED(hrP)) return(TRUE);
  365. HWND hWnd = AfxGetApp()->GetMainWnd()->m_hWnd;
  366. MessageBoxEx(hWnd, "Can not allocate enough memory\t", ERROR_TITLE, MB_OK | MB_ICONSTOP, LANG_ENGLISH);
  367. return(FALSE);
  368. }
  369. //-----------------------------------------------------------------------------
  370. // Does: Saves The Picture That Is Stored In The IPicture Object As a Bitmap
  371. // ~~~~ (Converts From Any Known Picture Type To a Bitmap / Icon File)
  372. //
  373. // InPut: sFilePathName - Path And FileName Target To Save
  374. // ~~~~~
  375. //
  376. // OutPut: TRUE If Succeeded...
  377. // ~~~~~~
  378. //-----------------------------------------------------------------------------
  379. BOOL CPicture::SaveAsBitmap(CString sFilePathName)
  380. //=============================================================================
  381. {
  382. BOOL bResult = FALSE;
  383. ILockBytes *Buffer = 0;
  384. IStorage *pStorage = 0;
  385. IStream *FileStream = 0;
  386. BYTE *BufferBytes;
  387. STATSTG BytesStatistics;
  388. DWORD OutData;
  389. long OutStream;
  390. CFile BitmapFile; CFileException e;
  391. double SkipFloat = 0;
  392. DWORD ByteSkip = 0;
  393. _ULARGE_INTEGER RealData;
  394. CreateILockBytesOnHGlobal(NULL, TRUE, &Buffer); // Create ILockBytes Buffer
  395. HRESULT hr = ::StgCreateDocfileOnILockBytes(Buffer,
  396. STGM_SHARE_EXCLUSIVE | STGM_CREATE | STGM_READWRITE, 0, &pStorage);
  397. hr = pStorage->CreateStream(L"PICTURE",
  398. STGM_SHARE_EXCLUSIVE | STGM_CREATE | STGM_READWRITE, 0, 0, &FileStream);
  399. m_IPicture->SaveAsFile(FileStream, TRUE, &OutStream); // Copy Data Stream
  400. FileStream->Release();
  401. pStorage->Release();
  402. Buffer->Flush();
  403. // Get Statistics For Final Size Of Byte Array
  404. Buffer->Stat(&BytesStatistics, STATFLAG_NONAME);
  405. // Cut UnNeeded Data Coming From SaveAsFile() (Leave Only "Pure" Picture Data)
  406. SkipFloat = (double(OutStream) / 512); // Must Be In a 512 Blocks...
  407. if(SkipFloat > DWORD(SkipFloat)) ByteSkip = (DWORD)SkipFloat + 1;
  408. else ByteSkip = (DWORD)SkipFloat;
  409. ByteSkip = ByteSkip * 512; // Must Be In a 512 Blocks...
  410. // Find Difference Between The Two Values
  411. ByteSkip = (DWORD)(BytesStatistics.cbSize.QuadPart - ByteSkip);
  412. // Allocate Only The "Pure" Picture Data
  413. RealData.LowPart = 0;
  414. RealData.HighPart = 0;
  415. RealData.QuadPart = ByteSkip;
  416. BufferBytes = (BYTE*)malloc(OutStream);
  417. if(BufferBytes == NULL)
  418. {
  419. Buffer->Release();
  420. HWND hWnd = AfxGetApp()->GetMainWnd()->m_hWnd;
  421. MessageBoxEx(hWnd, "Can not allocate enough memory\t", ERROR_TITLE, MB_OK | MB_ICONSTOP, LANG_ENGLISH);
  422. }
  423. Buffer->ReadAt(RealData, BufferBytes, OutStream, &OutData);
  424. if(BitmapFile.Open(sFilePathName, CFile::typeBinary | CFile::modeCreate | CFile::modeWrite, &e))
  425. {
  426. BitmapFile.Write(BufferBytes, OutData);
  427. BitmapFile.Close();
  428. bResult = TRUE;
  429. }
  430. else // Write File Failed...
  431. {
  432. TCHAR szCause[255];
  433. e.GetErrorMessage(szCause, 255, NULL);
  434. HWND hWnd = AfxGetApp()->GetMainWnd()->m_hWnd;
  435. MessageBoxEx(hWnd, szCause, ERROR_TITLE, MB_OK | MB_ICONSTOP, LANG_ENGLISH);
  436. bResult = FALSE;
  437. }
  438. Buffer->Release();
  439. free(BufferBytes);
  440. return(bResult);
  441. }
  442. //-----------------------------------------------------------------------------
  443. // Does: Draw a Bitmap Resource To The Client DC (Using Bitblt())
  444. // ~~~~ It Will Use The Bitmap Resource Original Size (Width And Height)
  445. // (.BMP .DIB)
  446. //
  447. // Note: This Function Is Just Another Simple Way Of Displaying a Bitmap Resource,
  448. // ~~~~ It Is Not Connected With The IPicture Interface And Can Be Used
  449. // As a StandAlone On Any Device Context (Directly)
  450. //
  451. // InPut: BMPResource - Resource Name As Defined In The Resources
  452. // ~~~~~ pDC - Given DC To Draw On
  453. // LeftTop - Opening Point To Start Drawing (Left,Top)
  454. //
  455. // OutPut: TRUE If Succeeded...
  456. // ~~~~~~
  457. //-----------------------------------------------------------------------------
  458. BOOL CPicture::ShowBitmapResource(CDC *pDC, const int BMPResource, CPoint LeftTop)
  459. //=============================================================================
  460. {
  461. if (pDC == NULL) return(FALSE);
  462. CBitmap BMP;
  463. if(BMP.LoadBitmap(BMPResource))
  464. {
  465. // Get Bitmap Details
  466. BITMAP BMPInfo;
  467. BMP.GetBitmap(&BMPInfo);
  468. // Create An In-Memory DC Compatible With The Display DC We R Gonna Paint On
  469. CDC DCMemory;
  470. DCMemory.CreateCompatibleDC(pDC);
  471. // Select The Bitmap Into The In-Memory DC
  472. CBitmap* pOldBitmap = DCMemory.SelectObject(&BMP);
  473. // Copy Bits From The In-Memory DC Into The On-Screen DC
  474. pDC->BitBlt(LeftTop.x, LeftTop.y, BMPInfo.bmWidth, BMPInfo.bmHeight, &DCMemory, 0, 0, SRCCOPY);
  475. DCMemory.SelectObject(pOldBitmap); // (As Shown In MSDN Example...)
  476. }
  477. else
  478. {
  479. TRACE0("ERROR: Can Not Find The Bitmap Resource\n");
  480. return(FALSE);
  481. }
  482. return(TRUE);
  483. }
  484. //-----------------------------------------------------------------------------
  485. // Does: Get The Original Picture Pixel Size (Ignor What Current DC Is Using)
  486. // ~~~~ Pointer To a Device Context Is Needed For Pixel Calculation,
  487. //
  488. // Also Updates The Class's Height And Width Properties,
  489. // (Coz Till Now We Had No Device Context To Work With...96 DPI Assumed)
  490. //
  491. // InPut: The Client DC (Needed To Check The Size Of The Pixels)
  492. // ~~~~~
  493. //
  494. // OutPut: TRUE If Succeeded...
  495. // ~~~~~~
  496. //-----------------------------------------------------------------------------
  497. BOOL CPicture::UpdateSizeOnDC(CDC *pDC)
  498. //=============================================================================
  499. {
  500. if(pDC == NULL || m_IPicture == NULL) { m_Height = 0; m_Width = 0; return(FALSE); };
  501. m_IPicture->get_Height(&m_Height);
  502. m_IPicture->get_Width(&m_Width);
  503. // Get Current DPI - Dot Per Inch
  504. int CurrentDPI_X = pDC->GetDeviceCaps(LOGPIXELSX);
  505. int CurrentDPI_Y = pDC->GetDeviceCaps(LOGPIXELSY);
  506. // Use a "Standard" Print (When Printing)
  507. if(pDC->IsPrinting())
  508. {
  509. CurrentDPI_X = 96;
  510. CurrentDPI_Y = 96;
  511. }
  512. m_Height = MulDiv(m_Height, CurrentDPI_Y, HIMETRIC_INCH);
  513. m_Width = MulDiv(m_Width, CurrentDPI_X, HIMETRIC_INCH);
  514. return(TRUE);
  515. }