Quantcast
Channel: VBForums - CodeBank - Visual Basic 6 and earlier
Viewing all articles
Browse latest Browse all 1448

How to convert StdPicture into pixel array

$
0
0
Here's some code you can put in a module, that will let you convert any StdPicture object into an ordinary byte array that contains the pixel data in 32bit-per-pixel format. In addition to it being 32bit, it makes sure that the value for field Height in the BitmapInfoHeader structure used in the conversion is a negative number, so that the first row of pixels (y=0) is always at the top (like with most image formats) rather than at the bottom (like it usually is for BMP files). The below code is fully commented, so you can see how it works.
Code:

Private Declare Function GetDIBits Lib "gdi32.dll" (ByVal hDC As Long, ByVal hBitmap As Long, ByVal nStartScan As Long, ByVal nNumScans As Long, ByRef lpBits As Any, ByRef lpBI As BITMAPINFO, ByVal wUsage As Long) As Long
Private Declare Function CreateCompatibleDC Lib "gdi32.dll" (ByVal hDC As Long) As Long
Private Declare Function DeleteDC Lib "gdi32.dll" (ByVal hDC As Long) As Long

Private Type BITMAPINFOHEADER
    biSize As Long
    biWidth As Long
    biHeight As Long
    biPlanes As Integer
    biBitCount As Integer
    biCompression As Long
    biSizeImage As Long
    biXPelsPerMeter As Long
    biYPelsPerMeter As Long
    biClrUsed As Long
    biClrImportant As Long
End Type

Private Type RGBQUAD
    rgbBlue As Byte
    rgbGreen As Byte
    rgbRed As Byte
    rgbReserved As Byte
End Type

Private Type BITMAPINFO
    bmiHeader As BITMAPINFOHEADER
    bmiColors As RGBQUAD
End Type



Public Function StdPictureToPix(ByRef Picture As StdPicture) As Byte()
    Dim Pix() As Byte
    Dim BMI As BITMAPINFO
    Dim Width As Long
    Dim Height As Long
    Dim hDC As Long
   
   
    hDC = CreateCompatibleDC(0) 'Create a temporary in-memory device context
    BMI.bmiHeader.biSize = Len(BMI.bmiHeader) 'Initialize BitmapInfoHeader with header size
    GetDIBits hDC, Picture.Handle, 0, 0, ByVal 0&, BMI, 0 'Get Information about the image
   
    'Set up BitmapInfoHeader for getting the pixel data in 32bit format, and with y=0 on the top
    With BMI.bmiHeader
        .biBitCount = 32
        .biClrUsed = 0
        .biClrImportant = 0
        .biSizeImage = 0
        .biCompression = 0
        Width = .biWidth
        Height = Abs(.biHeight)
        .biHeight = -Height
    End With
    '32bit format has, for each pixel, 3 color channels in the order B, G, R, and an unused channel
    'This always satisfies the condition that each image row must have a multiple-of-4 byte count
   
    ReDim Pix(3, Width - 1, Height - 1) 'Initialize array for holding pixel data
    GetDIBits hDC, Picture.Handle, 0, Height, Pix(0, 0, 0), BMI, 0 'Get pixel data
    DeleteDC hDC 'Get rid of temporary in-memory device context
   
    StdPictureToPix = Pix()
End Function

After calling this to get the pixels in an array, to get the width you just need to do UBound(arrayname,2)+1 to get the width in the calling function, and UBound(arrayname,3)+1 to get the height. UBound(arrayname,1)+1 always is 4, because there are 4 bytes per pixel, and the first dimension in the array is the color channel selector (0=B, 1=G, 2=R, 3=unused). Here's some sample code for sample code for how to use it to load a picture file directly into a byte array, and then displaying it to the form (albeit using the inefficient PSet statement).
Code:

Private Sub Form_Load()
    Dim x As Long
    Dim y As Long
    Dim Pix() As Byte
    Dim Img As New ImageFile
   
    Pix() = StdPictureToPix(LoadPicture("picturefile.jpg"))
    Show
    For y = 0 To UBound(Pix, 3)
        For x = 0 To UBound(Pix, 2)
            PSet (x, y), RGB(Pix(2, x, y), Pix(1, x, y), Pix(0, x, y))
        Next x
        DoEvents
    Next y
End Sub

Note that LoadPicture only works with BMP, JPG, and GIF formats. If you want to support TIF and PNG, you will need to use WIA 2.0, which comes with all versions of Windows since Vista, but not XP (though it might be available in Windows XP with SP3).

Viewing all articles
Browse latest Browse all 1448

Trending Articles



<script src="https://jsc.adskeeper.com/r/s/rssing.com.1596347.js" async> </script>