Load, Display, and Save Images - Windows C DLL

This tutorial shows how to load, display, and save images using the LEADTOOLS SDK in a C/C++ Windows API Application.

Overview  
Summary This tutorial covers how to load, display, and save images in a Windows C DLL Application.
Completion Time 30 minutes
Visual Studio Project Download tutorial project (18 KB)
Platform Windows C DLL Application
IDE Visual Studio 2017, 2019, 2022
Development License Download LEADTOOLS

Required Knowledge

Before working on the Load, Display, and Save Images - Windows C DLL tutorial, get familiar with the basic steps of creating a project by reviewing the Add References and Set a License tutorial.

Create the Project and Add LEADTOOLS References

Start with a copy of the 64-bit Windows API project created in the Add References and Set a License tutorial. If the project is not available, create it by following the steps in that tutorial.

In order to load, display, and save images, LEADTOOLS requires additional DLLs. Add the required libraries by opening the pre-compiled header file, either pch.h or stdafx.h depending on the Visual Studio version used, and add the following lines:

#pragma comment (lib, "C:\\LEADTOOLS23\\Lib\\CDLL\\x64\\Ltfil_x.lib") // file loading and saving 
#pragma comment (lib, "C:\\LEADTOOLS23\\Lib\\CDLL\\x64\\Ltdis_x.lib") // image display 

Note: For a complete list of which DLL files are required for your application, refer to Files to be Included with your Application.

Set the License File

The License unlocks the features needed for the project. It must be set before any toolkit function is called. For details, including tutorials for different platforms, refer to Setting a Runtime License.

There are two types of runtime licenses:

Note: Adding LEADTOOLS references and setting a license are covered in more detail in the Add References and Set a License tutorial.

Add the Load Bitmap Code

Now that the LEADTOOLS references have been added and the license has been set, coding can begin.

The steps below are for Visual Studio 2019; they could be different for other versions of Visual Studio.

Go to the Solution Explorer and double-click the resources file (.rc)

Resources File

Expand the menu in the resources tree and double-click the menu resource to open it in the designer interface.

Menu resource to open it in the designer

In the empty item below the Exit item, click and type &Open. Drag the new item above Exit. This should cause the item's ID to become ID_FILE_OPEN.

Drag the new item to set its ID

Go to the main CPP file of the project, which contains the WndProc() function for the main window. Add the following declaration to the global variables near the top:

BITMAPHANDLE LEADBmp; 

Navigate to the WndProc function and under the switch (wmId) statement, that is below the WM_COMMAND case, add a new case:

switch (wmId) 
{ 
   case ID_FILE_OPEN: 
      { 
         TCHAR szFileName[260] = TEXT(""); // File name 
         if (SUCCESS != GetBitmapLoadingName(hWnd, szFileName, ARRAYSIZE(szFileName))) 
            break; 
         if (LEADBmp.Flags.Allocated) 
            L_FreeBitmap(&LEADBmp); 
         L_LoadBitmap(szFileName, &LEADBmp, sizeof(LEADBmp), 24, ORDER_BGR, NULL, NULL); 
         InvalidateRect(hWnd, NULL, TRUE); 
      } 
      break; 
   // Keep rest of the code as is 

The GetBitmapLoadingName() function can be any function that fills the szFileName variable with a valid image file name (including full path, if needed). To display a File Open dialog to obtain the file name, add the following code:

L_INT GetBitmapLoadingName(HWND hwnd, TCHAR* pszFileName, DWORD nLen) 
{ 
   OPENFILENAME OpenFileName = { 0 }; 
   OpenFileName.lStructSize = sizeof(OpenFileName); 
   OpenFileName.hwndOwner = hwnd; 
   OpenFileName.lpstrFile = pszFileName; 
   OpenFileName.nMaxFile = nLen; 
   OpenFileName.lpstrFilter = TEXT("All\0*.*\0Jpeg\0*.jpg\0"); 
   OpenFileName.nFilterIndex = 1; 
   OpenFileName.lpstrFileTitle = NULL; 
   OpenFileName.nMaxFileTitle = 0; 
   OpenFileName.lpstrInitialDir = NULL; 
   OpenFileName.Flags = OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST; 
 
   // Show the File Open dialog box 
   if (!GetOpenFileName(&OpenFileName)) 
      return FAILURE; 
   return SUCCESS; 
} 

Whenever an image is allocated in a BITMAPHANDLE structure, the programmer must call L_FreeBitmap to free it when it is no longer needed. Since loading a bitmap allocates such an image, call LEADBmp.Flags.Allocated to check before exiting the application.

Modify the case that handles WM_DESTROY in the WndProc() function to become:

case WM_DESTROY: 
   if (LEADBmp.Flags.Allocated) 
      L_FreeBitmap(&LEADBmp); 
   PostQuitMessage(0); 
   break; 

Add the Image Display Code

Modify the case that handles WM_PAINT in the WndProc() function to become:

case WM_PAINT: 
{ 
   PAINTSTRUCT ps; 
   HDC hdc = BeginPaint(hWnd, &ps); 
   // TODO: Add any drawing code that uses hdc here... 
   RECT rc = { 0, 0, BITMAPWIDTH(&LEADBmp), BITMAPHEIGHT(&LEADBmp) }; 
   if (LEADBmp.Flags.Allocated) 
      L_PaintDC(hdc, &LEADBmp, NULL, NULL, &rc, NULL, SRCCOPY); 
   EndPaint(hWnd, &ps); 
} 
break; 

Important Note: The loading code in this tutorial forces the image to load at 24 bits per pixel, which means it will not have a palette after it gets loaded. If it is required to display images that are between 1 and 8 bits per pixel, the palette must be handled in the painting code as shown in the code example for L_PaintDC.

Add the Save Bitmap Code

Add a Save menu item under the File menu. To do that, use steps similar to those in the Add the Bitmap Loading Code section above, but set the menu item text to &Save instead of &Open, which should cause the new item's ID to become ID_FILE_SAVE.

Go to the WndProc function, and under the switch (wmId) statement that is below the WM_COMMAND case, add a new case:

switch (wmId) 
{ 
case ID_FILE_SAVE: 
{ 
   if (!LEADBmp.Flags.Allocated) 
   { 
      MessageBox(hWnd, TEXT("Cannot save. No image loaded"), TEXT("LEADTOOLS Demo"), MB_ICONERROR); 
      break; 
   } 
   TCHAR szFileName[260] = TEXT(""); // File name 
   if (SUCCESS != GetBitmapSavingName(hWnd, szFileName, ARRAYSIZE(szFileName))) 
      break; 
   L_SaveBitmap(szFileName, &LEADBmp, FILE_JPEG_411, 24, 15, NULL); 
} 
break; 
// Keep rest of the code as is 

The GetBitmapSavingName() function can be any function that fills the szFileName variable with a valid image file name (including full path, if needed). To display a File Save dialog to obtain the file name, add the following code:

L_INT GetBitmapSavingName(HWND hwnd, TCHAR* pszFileName, DWORD nLen) 
{ 
   OPENFILENAME OpenFileName = { 0 }; 
   OpenFileName.lStructSize = sizeof(OPENFILENAME); 
   OpenFileName.hwndOwner = hwnd; 
   OpenFileName.lpstrFilter = TEXT("Jpeg\0*.jpg\0"); 
   OpenFileName.lpstrFile = pszFileName; 
   OpenFileName.nMaxFile = nLen; 
   OpenFileName.lpstrDefExt = TEXT("jpg"); 
   OpenFileName.lpstrFileTitle = NULL; 
   OpenFileName.nMaxFileTitle = 0; 
   OpenFileName.lpstrInitialDir = NULL; 
   OpenFileName.Flags = OFN_OVERWRITEPROMPT; 
 
   // Show the File Save dialog box 
   if (!GetSaveFileName(&OpenFileName)) 
      return FAILURE; 
   return SUCCESS; 
} 

Important Note: The code above saves the image as 24-bit JPEG. To save other types, pass a different value instead of FILE_JPEG_411. More details are in the LEADTOOLS Image Formats help topic.

Run the Project

Run the project by pressing F5, or by selecting Debug -> Start Debugging.

If the steps are followed correctly, the application should run and enable the user to load any image on the machine that is supported by the LEADTOOLS codec filters. The image will display in the main window. Press the Save menu item to save the displayed image as a JPEG in the output location specified using the Save Dialog.

Wrap-up

This tutorial covered how to add the necessary DLLs to load, display, and save images. It also covered how to use the BITMAPHANDLE structure with the L_LoadBitmap, L_SaveBitmap, L_PaintDC and L_FreeBitmap functions.

See Also

Help Version 23.0.2025.1.8
Products | Support | Contact Us | Intellectual Property Notices
© 1991-2024 LEAD Technologies, Inc. All Rights Reserved.

Products | Support | Contact Us | Intellectual Property Notices
© 1991-2023 LEAD Technologies, Inc. All Rights Reserved.