Interactive GUI Application for Color-Based Object Detection using OpenCV

siromer
The Deep Hub
Published in
5 min readMar 12, 2024

Think about an image; it is all about different color values in different pixels. Color values of these pixels might be a very useful way for detecting and tracking objects. Think about it, when you use OpenCV, you work with different images and color spaces. Every image is a bunch of pixels with different color values. Color of these pixels might easily filtered for every frame, and objects might be tracked by choosing specific color values.

  • In this article, I will create an object tracker GUI using the color values of pixels
Detecting fishes

Colors can be represented in different formats. There are various ways to represent colors :

  • RGB (Red, Green, Blue)
  • BGR (Blue, Green, Red)
  • HSV (Hue, Saturation, Value)

HSV COLOR SPACE

HSV stands for Hue, Saturation, and Value. It is a color space representation that is often used in image processing and computer vision tasks.

The advantages of using HSV color space for color selection are that it allows for easy manipulation of hue, saturation, and value. However, a disadvantage is that it may not accurately represent all colors.(source)

If you look at this image carefully you will notice that you can not obtain all the colors

HSV

How to use colors for object detection?

Using colors for object detection involves identifying objects in an image based on their color properties. There are 5 main steps:

  1. Choose a Color Space: In general, HSV is a good choice.
  2. Thresholding: Set thresholds in the chosen color space to isolate regions of the image that match the color of the object you want to detect. For example, if you choose HSV color space, define a range for hue, saturation, and value channels. If you want to detect blue objects, you need to define specific lower and upper limits for blue.
  3. Generate a Mask: Create a binary mask where pixels within the specified color range are set to 1 (white) and pixels outside the range are set to 0 (black). This mask is going to separate regions of interest in the image, in this case, it will isolate the desired color.
  4. Contour Detection: After finding mask , finding contours is a piece of cake. OpenCV provides cv2.findContours() function for finding contours.
  5. Draw Rectangles: cv2.findContours() function is going to return a list of contours. Iterate over that list and use cv2.boundingRect(contour) function to find the coordinates of bounding rectangles for each contour. After that, draw rectangles with these coordinates.
Detecting Blue Balls

Interactive GUI Application / CODE

I explained the main algorithm above with 5 steps. In the code section, I explained all the lines with comments.

The program is quite simple. The user chooses a color using a color bar, and the program takes that color, processes it, and extracts objects with that color.

import cv2
import numpy as np
import tkinter as tk
from tkinter import ttk
from PIL import Image, ImageTk

class ColorPickerApp:
def __init__(self, master):
self.master = master
self.master.title("Color Picker")
self.master.geometry("800x600") # Adjust the size of the window

# Create a frame to hold the color bar and color image
self.color_bar_frame = tk.Frame(master)
self.color_bar_frame.pack(side="top", fill="x", padx=5, pady=5)

self.hue_label = ttk.Label(self.color_bar_frame, text="Select Hue Value (10-179):")
self.hue_label.pack(side="left", padx=5, pady=5)

self.hue_scale = ttk.Scale(self.color_bar_frame, from_=10, to=179, orient="horizontal", command=self.update_color)
self.hue_scale.pack(side="left", padx=5, pady=5)

# Create a canvas for the color image
self.canvas_color = tk.Canvas(master, width=100, height=320)
self.canvas_color.pack(side="left", padx=5, pady=75)

# Create a canvas for the image
self.canvas_image = tk.Canvas(master, width=800, height=400)
self.canvas_image.pack(side="top", padx=5, pady=50)

self.detect_button = ttk.Button(master, text="Detect Objects", command=self.detect_objects)
self.detect_button.pack(side="top", padx=5, pady=5)

self.image = None
self.image_rgb = None
self.image_hsv = None

# Video capture
self.cap = cv2.VideoCapture("fish.mp4") # Change to 0 for webcam, or provide path for video file

# Load the initial frame
self.load_frame()

def load_frame(self):
ret, frame = self.cap.read()
if ret:
self.image = frame
self.image_rgb = cv2.cvtColor(frame, cv2.COLOR_BGR2RGB)
self.image_hsv = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV)

# Display the frame with detected regions
self.display_frame(self.image_rgb)
self.master.after(100, self.load_frame) # Continue to load frames

def update_color(self, value):
hue_value = int(float(value))
color_image = np.zeros((400, 100, 3), dtype=np.uint8)
color_image[:, :] = (hue_value, 255, 255)
color_image_rgb = cv2.cvtColor(color_image, cv2.COLOR_HSV2RGB)
color_image_rgb = Image.fromarray(color_image_rgb)

# Display the color image
color_image_tk = ImageTk.PhotoImage(image=color_image_rgb)
self.canvas_color.create_image(0, 0, anchor="nw", image=color_image_tk)
self.canvas_color.image = color_image_tk



def display_frame(self, frame):
img = Image.fromarray(frame)

# Get the original frame dimensions
frame_width, frame_height = img.size


# Define maximum width and height
max_width = 600
max_height = 300

# Calculate target width and height
target_width = min(frame_width, max_width)
target_height = min(frame_height, max_height)

# Calculate aspect ratio
aspect_ratio = frame_width / frame_height

# Adjust dimensions if necessary to fit within limits
if aspect_ratio > max_width / max_height:
target_width = max_width
target_height = int(target_width / aspect_ratio)
else:
target_height = max_height
target_width = int(target_height * aspect_ratio)


# Resize the frame while maintaining the aspect ratio
img = img.resize((target_width, target_height), Image.LANCZOS)

# Convert the resized frame to PhotoImage
img = ImageTk.PhotoImage(image=img)



# Clear previous frame and display the resized frame
self.canvas_image.delete("all")
self.canvas_image.create_image(0, 0, anchor="nw", image=img)
self.canvas_image.image = img

def detect_objects(self):
if self.image is None:
return

print("detecting objects")
# Define the hue range based on the current value of the hue scale
hue_value = int(self.hue_scale.get())
lower_limit = np.array([hue_value - 8, 100, 100])
upper_limit = np.array([hue_value + 8, 255, 255])

# Create a mask to detect objects within the specified hue range
mask = cv2.inRange(self.image_hsv, lower_limit, upper_limit)
contours, _ = cv2.findContours(mask, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)

# Draw rectangles around the detected objects
for contour in contours:
print("contour found")
#if cv2.contourArea(contour) > 50:
x, y, w, h = cv2.boundingRect(contour)
cv2.rectangle(self.image_rgb, (x, y), (x + w, y + h), (255, 255, 0), 5)

# Display the updated frame with detected objects
self.display_frame(self.image_rgb)

# Call detect_objects again after a delay
self.master.after(50, self.detect_objects)


def main():
root = tk.Tk()
app = ColorPickerApp(root)
root.mainloop()


if __name__ == "__main__":
main()
Result
Red Car

--

--