If Image contains specific color then

ぐ巨炮叔叔 提交于 2019-12-10 18:43:16

问题


Exists a easy way to check if a image contains certain rgb color?

For example:

Dim img As Image = Image.FromFile("C:\image.png")

If img.contains color.red(.toRGB) then... 

I think the hard way to check this is by getting the color value of each pixel of the image, excluding the transparency of the pngs... also that technique I think will take a longer amount of time to process the entire image... but anyway I don't know how to do that too.

UPDATE

Tried this, but it sayd i'm trying to read from protected memory access:

' Lock the bitmap's data.
Public Function LockBitmap(ByVal bm As Bitmap) As Byte()
    Dim bmp = New Bitmap("C:\Users\Administrador\Desktop\PrtScr capture_3.jpg")
    Dim bmdata As BitmapData = bmp.LockBits(New Rectangle(0, 0, bmp.Width, bmp.Height), ImageLockMode.ReadOnly, PixelFormat.Format32bppArgb)
    Dim buffer(bmdata.Stride * bmdata.Height - 1) As Byte
    Marshal.Copy(bmdata.Scan0, buffer, 0, buffer.Length)
    Dim stride As Integer = bmdata.Stride
    bmp.UnlockBits(bmdata)

    'loop over the pixels
    For y As Integer = 0 To bmp.Height - 1
        For x As Integer = 0 To bmp.Width - 1
            Dim valB As Integer = 255 - Marshal.ReadByte(bmdata.Scan0, (stride * y) + (x * 4))
            Dim valG As Integer = 255 - Marshal.ReadByte(bmdata.Scan0, (stride * y) + (x * 4) + 1)
            Dim valR As Integer = 255 - Marshal.ReadByte(bmdata.Scan0, (stride * y) + (x * 4) + 2)

            MsgBox(valR & "," & valG & "," & valB)
        Next
    Next

End Function

Tried this but can't set the value: Can't set the correct value in this while loop

Imports System.Drawing.Imaging

Public Class Form1

Private Shared Function ImageHasColor(ByVal image As Image, ByVal R As Int32, ByVal G As Int32, ByVal B As Int32) As Boolean

    Using bmp = New Bitmap(image.Width, image.Height, PixelFormat.Format32bppArgb)

        Using graph = Graphics.FromImage(bmp)
            graph.DrawImage(image, 0, 0)
        End Using

        Dim data = bmp.LockBits(New Rectangle(0, 0, bmp.Width, bmp.Height), ImageLockMode.[ReadOnly], bmp.PixelFormat)

        Dim pt = CType(data.Scan0, Integer)
        MsgBox(pt)
        Dim res As Boolean

        Dim i = 0
        While i < data.Height * data.Width
            Dim color__1 = Color.FromArgb(pt(i))

            If color__1.A <> 0 AndAlso color__1.R = R AndAlso color__1.G = G AndAlso color__1.B = B Then
                res = True
                Exit While
            End If

            System.Math.Max(System.Threading.Interlocked.Increment(i), i - 1)
        End While

        bmp.UnlockBits(data)

        Return res
    End Using
End Function


Private Sub Form1_Load(sender As Object, e As EventArgs) Handles MyBase.Load
    Dim bmp = New Bitmap("C:\Users\Administrador\Desktop\PrtScr capture_3.jpg")
    MsgBox(ImageHasColor(bmp, 240, 240, 240))
End Sub

End Class

Tried taking an example of this making my modifications: http://social.msdn.microsoft.com/Forums/en-US/vblanguage/thread/1d04ef14-cffb-4ae1-83d3-3a0cd8255c95

    'get a BitmapDataObject for fast processing and copy the Data of the image to an array
    Dim bmdata As BitmapData = bmp.LockBits(New Rectangle(0, 0, bmp.Width, bmp.Height), ImageLockMode.ReadOnly, PixelFormat.Format32bppArgb)
    Dim buffer(bmdata.Stride * bmdata.Height - 1) As Byte
    Marshal.Copy(bmdata.Scan0, buffer, 0, buffer.Length)
    Dim stride As Integer = bmdata.Stride
    bmp.UnlockBits(bmdata)

    'loop over the pixels
    For y As Integer = 0 To bmp.Height - 1
        For x As Integer = 0 To bmp.Width - 1
            'Colors/Bytes stored in sequence BGRA
            'test access of alpha channel
            If buffer((stride * y) + (x * 4) + 3) <> 0 Then
               'do something
            End If
        Next
    Next

' use Marshal.Copy again to put the processed data back to the image.

More time spent trying more examples from here: http://www.vb-helper.com/howto_net_lockbits_image.html

' Title Manipulate image pixels very quickly using LockBits in VB .NET

Public g_RowSizeBytes As Integer
Public g_PixBytes() As Byte

Private m_BitmapData As BitmapData

' Lock the bitmap's data.
Public Sub LockBitmap(ByVal bm As Bitmap)
    ' Lock the bitmap data.
    Dim bounds As Rectangle = New Rectangle( _
        0, 0, bm.Width, bm.Height)
    m_BitmapData = bm.LockBits(bounds, _
        Imaging.ImageLockMode.ReadWrite, _
        Imaging.PixelFormat.Format24bppRgb)
    g_RowSizeBytes = m_BitmapData.Stride

    ' Allocate room for the data.
    Dim total_size As Integer = m_BitmapData.Stride * _
        m_BitmapData.Height
    ReDim g_PixBytes(total_size)

    ' Copy the data into the g_PixBytes array.
    Marshal.Copy(m_BitmapData.Scan0, g_PixBytes, _
        0, total_size)
End Sub

Also tried also an alternative way : https://stackoverflow.com/questions/16598567/transparencykey-and-displayed-images-issue

I don't have the necessary knowledges to make it.

PLEASE, SOMEONE CAN HELP ME?


回答1:


Well, you can do it easily through the "hard way" as you identified, checking every pixel until a specific color is found. Worst case scenario? Loop the entire image.

Private Function ColorInBitmap(Path As String, Target As Color) As Boolean
    Dim B As New Bitmap(Path)
    For x As Integer = 1 To B.Width
        For y As Integer = 1 To B.Height
            Dim tmpC As Color = B.GetPixel(x - 1, y - 1)
            If tmpC.Equals(Target) Then Return True
        Next
    Next
    Return False
End Function

It took 5,6 seconds to return False on 2048x1536 image, measuring before the Bitmap class loads the JPG

You can use it simply by passing the path to the image and the target color.

ColorInBitmap("c:\test.jpg", Color.FromArgb(109, 109, 109))

Note that this Fromargb overload sets alpha value to the max.



来源:https://stackoverflow.com/questions/16598400/if-image-contains-specific-color-then

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!