-
Notifications
You must be signed in to change notification settings - Fork 1.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Oriented Bounding Box Tracker #1774
Comments
Hi @LilBabines! "Straightening" up the bounding boxes will give you worse association with any type of IoU. Think about the case of a top down view where two cars are side to side. With OBB there would be zero overlap but there may be if they are straightened. If your use-case is simple, it may be possible to simplify the problem by a |
Thank you very much for your quick response. As you mentioned, "straightening" up the bounding boxes is indeed risky, and I fully understand the challenges related to oriented versus non-oriented bounding boxes. I’ve attached an image to illustrate my use case (quite similar to #1725), which I believe is very well-suited for oriented bounding boxes. I’m considering starting with OCSort for a more or less complete adaptation to handle oriented bounding boxes. I will also take a closer look later to consider feature extraction if needed. Do you think this could be a valuable enhancement and interest the community? |
Absolutely. OBB tracking would be a natural next step for this repo as well given the wider adoption of these object detection methods. I recommend you to trigger the load of a specific Kalman Filter designed for OBB when the detection input is one of (
|
OBB straightening: import cv2
import math
import numpy as np
def extract_and_straighten_crops(image, cxs, cys, ws, hs, rs):
"""
Extract and straighten multiple crops from an image given arrays of OBB parameters.
Parameters
----------
image : numpy.ndarray
Input image (BGR or RGB).
cxs, cys : numpy.ndarray
Arrays of center coordinates for each OBB.
ws, hs : numpy.ndarray
Arrays of widths and heights for each OBB.
rs : numpy.ndarray
Array of rotation angles in radians for each OBB.
Returns
-------
crops : list of numpy.ndarray
Extracted, straightened crops for each OBB.
"""
angles_degrees = np.degrees(rs)
rows, cols = image.shape[:2]
# Precompute rotation matrices (still a loop, but it's small and fast)
rotation_matrices = [
cv2.getRotationMatrix2D((float(cx), float(cy)), float(-angle), 1.0)
for cx, cy, angle in zip(cxs, cys, angles_degrees)
]
# Vectorized bounding box calculation
half_ws = ws / 2
half_hs = hs / 2
x_mins = np.int32(cxs - half_ws)
y_mins = np.int32(cys - half_hs)
x_maxs = np.int32(cxs + half_ws)
y_maxs = np.int32(cys + half_hs)
# Clip coordinates to image boundaries
x_mins = np.clip(x_mins, 0, cols)
y_mins = np.clip(y_mins, 0, rows)
x_maxs = np.clip(x_maxs, 0, cols)
y_maxs = np.clip(y_maxs, 0, rows)
crops = []
for (M, x_min, y_min, x_max, y_max) in zip(rotation_matrices, x_mins, y_mins, x_maxs, y_maxs):
rotated = cv2.warpAffine(
image, M, (cols, rows),
flags=cv2.INTER_LINEAR,
borderMode=cv2.BORDER_CONSTANT, borderValue=(122, 122, 122)
)
crop = rotated[y_min:y_max, x_min:x_max]
crops.append(crop)
return crops
if __name__ == "__main__":
# Example usage:
image = cv2.imread("bus.jpg")
cxs = np.array([740, 500, 300])
cys = np.array([636, 400, 200])
ws = np.array([138, 80, 100])
hs = np.array([483, 60, 120])
r_degrees = np.array([45, 30, -20])
rs = np.radians(r_degrees)
crops = extract_and_straighten_crops(image, cxs, cys, ws, hs, rs)
for i, crop in enumerate(crops):
cv2.imshow(f"Straightened Crop {i}", crop)
cv2.waitKey(0)
cv2.destroyAllWindows() Could be used for feature extraction later on 🚀 |
👋 Hello, this issue has been automatically marked as stale because it has not had recent activity. Please note it will be closed if no further activity occurs. |
Hello, |
Awesome @LilBabines ! Looking forward to this 😄 |
Will find some time next week for feedback of what is implemented |
🚀 #1791 |
Here are the main script and annotation files for testing. I'll provide images if the owners allow me to share samples with you. import cv2
from pathlib import Path
from boxmot import OcSort
import math
import numpy as np
WIDTH = 3840
HEIGHT = 2160
def obbox_to_cxcywha(obbox):
cls, x1, y1, x2, y2, x3, y3, x4, y4, conf = obbox
points = np.array([[x1, y1], [x2, y2], [x3, y3], [x4, y4]])
#points = points[np.argsort(points[:, 0])] #[[0,2,1,3],:]
points = np.multiply(points, [WIDTH, HEIGHT])
x1, x2, x3, x4 = points[:, 0]
y1, y2, y3, y4 = points[:, 1]
cx = (x1 + x2 + x3 + x4) / 4
cy = (y1 + y2 + y3 + y4) / 4
# Calculate width and height
w = math.sqrt((x2 - x1)**2 + (y2 - y1)**2)
h = math.sqrt((x3 - x2)**2 + (y3 - y2)**2)
angle = math.atan2(y2 - y1, x2 - x1)
return cx, cy, w, h, angle, conf, cls
def lines_to_cxcyywha(lines):
obboxes = []
for line in lines:
line = line.strip().split(' ')
obboxes.append(obbox_to_cxcywha([float(i) for i in line]))
return obboxes
if __name__ == "__main__":
# Initialize the tracker
tracker = OcSort(
is_obb=True,
asso_func="centroid",
min_hits=10,
asso_threshold=0.98,
det_thresh = 0.7,
max_age=20,
use_byte=True,
Q_xy_scaling = 0.01,
Q_s_scaling = 0.0001,
)
annot_dir = Path('data_test/labels_predict') # Path to the annotations directory
#img_dir = Path('data_test/clip_2') # Path to the images directory
for file in sorted(annot_dir.glob('*.txt')):
# Read the annotation file
with open(file, 'r') as f:
lines = f.readlines()
bboxes = lines_to_cxcyywha(lines)
detections = np.array(bboxes)
#frame = cv2.imread(str(img_dir / (file.stem + '.png')))
frame = np.zeros((HEIGHT,WIDTH,3))
# Update the tracker
res = tracker.update(detections, frame) # --> M X (x, y, x, y, id, conf, cls, ind)
# Plot tracking results on the image
tracker.plot_results(frame, show_trajectories=True, fontscale=2, thickness=4)
# Display the frame
cv2.imshow('BoXMOT', cv2.resize(frame, (WIDTH//3, HEIGHT//3)))
print(len(tracker.active_tracks),end='\r')
key = cv2.waitKey(1) & 0xFF
if key == ord('q') or key ==27:
break
cv2.destroyAllWindows() |
👋 Hello, this issue has been automatically marked as stale because it has not had recent activity. Please note it will be closed if no further activity occurs. |
Search before asking
Question
Hello,
I’m currently working on a project that requires object tracking with Oriented Bounding Boxes (OBB). Despite thorough research, I haven’t found any convincing implementation for a tracker specifically handling OBB, either in this repository or elsewhere. For instance, the
yolo track model=yolov11m-obb
command and the code from ultralytics/trackers don’t seem to clearly integrate OBB.In this regard, I’d like to ask:
update(dets = obb_to_x1y1x2y2(boxes))
Thank you in advance for your support and for the excellent work on this package! I’m looking forward to your feedback.
Best regards,
The text was updated successfully, but these errors were encountered: