Link to this sectionReference for ultralytics/trackers/byte_tracker.py#
This page is sourced from https://github.com/ultralytics/ultralytics/blob/main/ultralytics/trackers/byte_tracker.py. Have an improvement or example to add? Open a Pull Request — thank you! 🙏
Link to this section ultralytics.trackers.byte_tracker.STrack#
STrack(self, xywh: np.ndarray, score: float, cls: Any)Bases: BaseTrack
Single object tracking representation that uses Kalman filtering for state estimation.
This class is responsible for storing all the information regarding individual tracklets and performs state updates and predictions based on Kalman filter.
Args
| Name | Type | Description | Default |
|---|---|---|---|
xywh | np.ndarray | Bounding box in (x, y, w, h, idx) or (x, y, w, h, angle, idx) format, where (x, y) isthe center, (w, h) are width and height, and idx is the detection index. | required |
score | float | Confidence score of the detection. | required |
cls | Any | Class label for the detected object. | required |
Attributes
| Name | Type | Description |
|---|---|---|
shared_kalman | KalmanFilterXYAH | Shared Kalman filter used across all STrack instances for prediction. |
_tlwh | np.ndarray | Private attribute to store top-left corner coordinates and width and height of bounding box. |
kalman_filter | KalmanFilterXYAH | Instance of Kalman filter used for this particular object track. |
mean | np.ndarray | Mean state estimate vector. |
covariance | np.ndarray | Covariance of state estimate. |
is_activated | bool | Boolean flag indicating if the track has been activated. |
score | float | Confidence score of the track. |
tracklet_len | int | Length of the tracklet. |
cls | Any | Class label for the object. |
idx | int | Index or identifier for the object. |
frame_id | int | Current frame ID. |
start_frame | int | Frame where the object was first detected. |
angle | `float | None` |
Methods
| Name | Description |
|---|---|
tlwh | Get the bounding box in top-left-width-height format from the current state estimate. |
xyxy | Convert bounding box from (top left x, top left y, width, height) to (min x, min y, max x, max y) format. |
xywh | Get the current position of the bounding box in (center x, center y, width, height) format. |
xywha | Get position in (center x, center y, width, height, angle) format, warning if angle is missing. |
result | Get the current tracking results in the appropriate bounding box format. |
__repr__ | Return a string representation of the STrack object including start frame, end frame, and track ID. |
activate | Activate a new tracklet using the provided Kalman filter and initialize its state and covariance. |
convert_coords | Convert a bounding box's top-left-width-height format to its x-y-aspect-height equivalent. |
multi_predict | Perform multi-object predictive tracking using Kalman filter for the provided list of STrack instances. |
predict | Predict the next state (mean and covariance) of the object using the Kalman filter. |
re_activate | Reactivate a previously lost track using new detection data and update its state and attributes. |
tlwh_to_xyah | Convert bounding box from tlwh format to center-x-center-y-aspect-height (xyah) format. |
update | Update the state of a matched track. |
Examples
Initialize and activate a new track
>>> track = STrack(xywh=[100, 200, 50, 80, 0], score=0.9, cls="person")
>>> track.activate(kalman_filter=KalmanFilterXYAH(), frame_id=1)Source code in ultralytics/trackers/byte_tracker.py
class STrack(BaseTrack):
"""Single object tracking representation that uses Kalman filtering for state estimation.
This class is responsible for storing all the information regarding individual tracklets and performs state updates
and predictions based on Kalman filter.
Attributes:
shared_kalman (KalmanFilterXYAH): Shared Kalman filter used across all STrack instances for prediction.
_tlwh (np.ndarray): Private attribute to store top-left corner coordinates and width and height of bounding box.
kalman_filter (KalmanFilterXYAH): Instance of Kalman filter used for this particular object track.
mean (np.ndarray): Mean state estimate vector.
covariance (np.ndarray): Covariance of state estimate.
is_activated (bool): Boolean flag indicating if the track has been activated.
score (float): Confidence score of the track.
tracklet_len (int): Length of the tracklet.
cls (Any): Class label for the object.
idx (int): Index or identifier for the object.
frame_id (int): Current frame ID.
start_frame (int): Frame where the object was first detected.
angle (float | None): Optional angle information for oriented bounding boxes.
Methods:
predict: Predict the next state of the object using Kalman filter.
multi_predict: Predict the next states for multiple tracks.
activate: Activate a new tracklet.
re_activate: Reactivate a previously lost tracklet.
update: Update the state of a matched track.
convert_coords: Convert bounding box to x-y-aspect-height format.
tlwh_to_xyah: Convert tlwh bounding box to xyah format.
Examples:
Initialize and activate a new track
>>> track = STrack(xywh=[100, 200, 50, 80, 0], score=0.9, cls="person")
>>> track.activate(kalman_filter=KalmanFilterXYAH(), frame_id=1)
"""
shared_kalman = KalmanFilterXYAH()
def __init__(self, xywh: np.ndarray, score: float, cls: Any):
"""Initialize a new STrack instance.
Args:
xywh (np.ndarray): Bounding box in `(x, y, w, h, idx)` or `(x, y, w, h, angle, idx)` format, where (x, y) is
the center, (w, h) are width and height, and `idx` is the detection index.
score (float): Confidence score of the detection.
cls (Any): Class label for the detected object.
"""
super().__init__()
# xywh+idx or xywha+idx
assert len(xywh) in {5, 6}, f"expected 5 or 6 values but got {len(xywh)}"
self._tlwh = np.asarray(xywh2ltwh(xywh[:4]), dtype=np.float32)
self.kalman_filter = None
self.mean, self.covariance = None, None
self.is_activated = False
self.score = score
self.tracklet_len = 0
self.cls = cls
self.idx = xywh[-1]
self.angle = xywh[4] if len(xywh) == 6 else NoneLink to this section ultralytics.trackers.byte_tracker.STrack.tlwh#
def tlwh(self) -> np.ndarrayGet the bounding box in top-left-width-height format from the current state estimate.
Source code in ultralytics/trackers/byte_tracker.py
@property
def tlwh(self) -> np.ndarray:
"""Get the bounding box in top-left-width-height format from the current state estimate."""
if self.mean is None:
return self._tlwh.copy()
ret = self.mean[:4].copy()
ret[2] *= ret[3]
ret[:2] -= ret[2:] / 2
return retLink to this section ultralytics.trackers.byte_tracker.STrack.xyxy#
def xyxy(self) -> np.ndarrayConvert bounding box from (top left x, top left y, width, height) to (min x, min y, max x, max y) format.
Source code in ultralytics/trackers/byte_tracker.py
@property
def xyxy(self) -> np.ndarray:
"""Convert bounding box from (top left x, top left y, width, height) to (min x, min y, max x, max y) format."""
ret = self.tlwh.copy()
ret[2:] += ret[:2]
return retLink to this section ultralytics.trackers.byte_tracker.STrack.xywh#
def xywh(self) -> np.ndarrayGet the current position of the bounding box in (center x, center y, width, height) format.
Source code in ultralytics/trackers/byte_tracker.py
@property
def xywh(self) -> np.ndarray:
"""Get the current position of the bounding box in (center x, center y, width, height) format."""
ret = np.asarray(self.tlwh).copy()
ret[:2] += ret[2:] / 2
return retLink to this section ultralytics.trackers.byte_tracker.STrack.xywha#
def xywha(self) -> np.ndarrayGet position in (center x, center y, width, height, angle) format, warning if angle is missing.
Source code in ultralytics/trackers/byte_tracker.py
@property
def xywha(self) -> np.ndarray:
"""Get position in (center x, center y, width, height, angle) format, warning if angle is missing."""
if self.angle is None:
LOGGER.warning("`angle` attr not found, returning `xywh` instead.")
return self.xywh
return np.concatenate([self.xywh, self.angle[None]])Link to this section ultralytics.trackers.byte_tracker.STrack.result#
def result(self) -> list[float]Get the current tracking results in the appropriate bounding box format.
Source code in ultralytics/trackers/byte_tracker.py
@property
def result(self) -> list[float]:
"""Get the current tracking results in the appropriate bounding box format."""
coords = self.xyxy if self.angle is None else self.xywha
return [*coords.tolist(), self.track_id, self.score, self.cls, self.idx]Link to this section ultralytics.trackers.byte_tracker.STrack.__repr__#
def __repr__(self) -> strReturn a string representation of the STrack object including start frame, end frame, and track ID.
Source code in ultralytics/trackers/byte_tracker.py
def __repr__(self) -> str:
"""Return a string representation of the STrack object including start frame, end frame, and track ID."""
return f"OT_{self.track_id}_({self.start_frame}-{self.end_frame})"Link to this section ultralytics.trackers.byte_tracker.STrack.activate#
def activate(self, kalman_filter: KalmanFilterXYAH, frame_id: int)Activate a new tracklet using the provided Kalman filter and initialize its state and covariance.
Args
| Name | Type | Description | Default |
|---|---|---|---|
kalman_filter | KalmanFilterXYAH | required | |
frame_id | int | required |
Source code in ultralytics/trackers/byte_tracker.py
def activate(self, kalman_filter: KalmanFilterXYAH, frame_id: int):
"""Activate a new tracklet using the provided Kalman filter and initialize its state and covariance."""
self.kalman_filter = kalman_filter
self.track_id = self.next_id()
self.mean, self.covariance = self.kalman_filter.initiate(self.convert_coords(self._tlwh))
self.tracklet_len = 0
self.state = TrackState.Tracked
if frame_id == 1:
self.is_activated = True
self.frame_id = frame_id
self.start_frame = frame_idLink to this section ultralytics.trackers.byte_tracker.STrack.convert_coords#
def convert_coords(self, tlwh: np.ndarray) -> np.ndarrayConvert a bounding box's top-left-width-height format to its x-y-aspect-height equivalent.
Args
| Name | Type | Description | Default |
|---|---|---|---|
tlwh | np.ndarray | required |
Source code in ultralytics/trackers/byte_tracker.py
def convert_coords(self, tlwh: np.ndarray) -> np.ndarray:
"""Convert a bounding box's top-left-width-height format to its x-y-aspect-height equivalent."""
return self.tlwh_to_xyah(tlwh)Link to this section ultralytics.trackers.byte_tracker.STrack.multi_predict#
def multi_predict(stracks: list[STrack])Perform multi-object predictive tracking using Kalman filter for the provided list of STrack instances.
Args
| Name | Type | Description | Default |
|---|---|---|---|
stracks | list[STrack] | required |
Source code in ultralytics/trackers/byte_tracker.py
@staticmethod
def multi_predict(stracks: list[STrack]):
"""Perform multi-object predictive tracking using Kalman filter for the provided list of STrack instances."""
if not stracks:
return
multi_mean = np.asarray([st.mean.copy() for st in stracks])
multi_covariance = np.asarray([st.covariance for st in stracks])
for i, st in enumerate(stracks):
if st.state != TrackState.Tracked:
multi_mean[i][7] = 0
multi_mean, multi_covariance = STrack.shared_kalman.multi_predict(multi_mean, multi_covariance)
for i, (mean, cov) in enumerate(zip(multi_mean, multi_covariance)):
stracks[i].mean = mean
stracks[i].covariance = covLink to this section ultralytics.trackers.byte_tracker.STrack.predict#
def predict(self)Predict the next state (mean and covariance) of the object using the Kalman filter.
Source code in ultralytics/trackers/byte_tracker.py
def predict(self):
"""Predict the next state (mean and covariance) of the object using the Kalman filter."""
mean_state = self.mean.copy()
if self.state != TrackState.Tracked:
mean_state[7] = 0
self.mean, self.covariance = self.kalman_filter.predict(mean_state, self.covariance)Link to this section ultralytics.trackers.byte_tracker.STrack.re_activate#
def re_activate(self, new_track: STrack, frame_id: int, new_id: bool = False)Reactivate a previously lost track using new detection data and update its state and attributes.
Args
| Name | Type | Description | Default |
|---|---|---|---|
new_track | STrack | required | |
frame_id | int | required | |
new_id | bool | False |
Source code in ultralytics/trackers/byte_tracker.py
def re_activate(self, new_track: STrack, frame_id: int, new_id: bool = False):
"""Reactivate a previously lost track using new detection data and update its state and attributes."""
self.mean, self.covariance = self.kalman_filter.update(
self.mean, self.covariance, self.convert_coords(new_track.tlwh)
)
self.tracklet_len = 0
self.state = TrackState.Tracked
self.is_activated = True
self.frame_id = frame_id
if new_id:
self.track_id = self.next_id()
self.score = new_track.score
self.cls = new_track.cls
self.angle = new_track.angle
self.idx = new_track.idxLink to this section ultralytics.trackers.byte_tracker.STrack.tlwh_to_xyah#
def tlwh_to_xyah(tlwh: np.ndarray) -> np.ndarrayConvert bounding box from tlwh format to center-x-center-y-aspect-height (xyah) format.
Args
| Name | Type | Description | Default |
|---|---|---|---|
tlwh | np.ndarray | required |
Source code in ultralytics/trackers/byte_tracker.py
@staticmethod
def tlwh_to_xyah(tlwh: np.ndarray) -> np.ndarray:
"""Convert bounding box from tlwh format to center-x-center-y-aspect-height (xyah) format."""
ret = np.asarray(tlwh).copy()
ret[:2] += ret[2:] / 2
ret[2] /= ret[3]
return retLink to this section ultralytics.trackers.byte_tracker.STrack.update#
def update(self, new_track: STrack, frame_id: int)Update the state of a matched track.
Args
| Name | Type | Description | Default |
|---|---|---|---|
new_track | STrack | The new track containing updated information. | required |
frame_id | int | The ID of the current frame. | required |
Examples
Update the state of a track with new detection information
>>> track = STrack([100, 200, 50, 80, 0], score=0.9, cls=0)
>>> new_track = STrack([105, 205, 55, 85, 0], score=0.95, cls=0)
>>> track.update(new_track, 2)Source code in ultralytics/trackers/byte_tracker.py
def update(self, new_track: STrack, frame_id: int):
"""Update the state of a matched track.
Args:
new_track (STrack): The new track containing updated information.
frame_id (int): The ID of the current frame.
Examples:
Update the state of a track with new detection information
>>> track = STrack([100, 200, 50, 80, 0], score=0.9, cls=0)
>>> new_track = STrack([105, 205, 55, 85, 0], score=0.95, cls=0)
>>> track.update(new_track, 2)
"""
self.frame_id = frame_id
self.tracklet_len += 1
new_tlwh = new_track.tlwh
self.mean, self.covariance = self.kalman_filter.update(
self.mean, self.covariance, self.convert_coords(new_tlwh)
)
self.state = TrackState.Tracked
self.is_activated = True
self.score = new_track.score
self.cls = new_track.cls
self.angle = new_track.angle
self.idx = new_track.idxLink to this section ultralytics.trackers.byte_tracker.BYTETracker#
BYTETracker(self, args)BYTETracker: A tracking algorithm built on top of YOLO for object detection and tracking.
This class encapsulates the functionality for initializing, updating, and managing the tracks for detected objects in a video sequence. It maintains the state of tracked, lost, and removed tracks over frames, utilizes Kalman filtering for predicting the new object locations, and performs data association.
Args
| Name | Type | Description | Default |
|---|---|---|---|
args | Namespace | Command-line arguments containing tracking parameters. | required |
Attributes
| Name | Type | Description |
|---|---|---|
tracked_stracks | list[STrack] | List of successfully activated tracks. |
lost_stracks | list[STrack] | List of lost tracks. |
removed_stracks | list[STrack] | List of removed tracks. |
frame_id | int | The current frame ID. |
args | Namespace | Command-line arguments. |
max_frames_lost | int | The maximum frames for a track to be considered as 'lost'. |
kalman_filter | KalmanFilterXYAH | Kalman Filter object. |
Methods
| Name | Description |
|---|---|
_apply_match | Update or re-activate a single track with its matched detection. |
_apply_matches | Apply a list of matched (track, detection) pairs from an association stage. |
_first_association | First-stage association between track pool and high-score detections. |
_format_output | Format the current tracked objects into the output array. |
_init_new_tracks | Activate new tracks from detections that survived all association stages. |
_input_for | Return the per-detection auxiliary input for init_track. |
_post_first_association | Hook executed after the first association stage and before the second. |
_pre_first_associate | Hook called after Kalman predict, before first-stage assignment. Default: GMC if available. |
_remove_stale_lost | Remove lost tracks that have exceeded the maximum allowed frames. |
_second_association | Second-stage association between remaining tracked tracks and low-score detections. |
_split_detections | Split detections into high-confidence and low-confidence subsets. |
_split_tracked | Separate self.tracked_stracks into confirmed and unconfirmed lists. |
_unconfirmed_association | Associate unconfirmed tracks with leftover high-score detections. |
get_dists | Calculate the distance between tracks and detections using IoU and optionally fuse scores. |
get_kalmanfilter | Return a Kalman filter object for tracking bounding boxes using KalmanFilterXYAH. |
init_track | Initialize object tracking with given detections, scores, and class labels as STrack instances. |
multi_predict | Predict the next states for multiple tracks using Kalman filter. |
reset | Reset the tracker by clearing all tracked, lost, and removed tracks and reinitializing the Kalman filter. |
reset_id | Reset the ID counter for STrack instances to ensure unique track IDs across tracking sessions. |
update | Update the tracker with new detections and return the current list of tracked objects. |
Examples
Initialize BYTETracker and update with detection results
>>> tracker = BYTETracker(args)
>>> results = yolo_model.detect(image)
>>> tracked_objects = tracker.update(results)Source code in ultralytics/trackers/byte_tracker.py
class BYTETracker:
"""BYTETracker: A tracking algorithm built on top of YOLO for object detection and tracking.
This class encapsulates the functionality for initializing, updating, and managing the tracks for detected objects
in a video sequence. It maintains the state of tracked, lost, and removed tracks over frames, utilizes Kalman
filtering for predicting the new object locations, and performs data association.
Attributes:
tracked_stracks (list[STrack]): List of successfully activated tracks.
lost_stracks (list[STrack]): List of lost tracks.
removed_stracks (list[STrack]): List of removed tracks.
frame_id (int): The current frame ID.
args (Namespace): Command-line arguments.
max_frames_lost (int): The maximum frames for a track to be considered as 'lost'.
kalman_filter (KalmanFilterXYAH): Kalman Filter object.
Methods:
update: Update object tracker with new detections.
get_kalmanfilter: Return a Kalman filter object for tracking bounding boxes.
init_track: Initialize object tracking with detections.
get_dists: Calculate the distance between tracks and detections.
multi_predict: Predict the location of tracks.
reset_id: Reset the ID counter of STrack.
reset: Reset the tracker by clearing all tracks.
joint_stracks: Combine two lists of stracks.
sub_stracks: Filter out the stracks present in the second list from the first list.
remove_duplicate_stracks: Remove duplicate stracks based on IoU.
Examples:
Initialize BYTETracker and update with detection results
>>> tracker = BYTETracker(args)
>>> results = yolo_model.detect(image)
>>> tracked_objects = tracker.update(results)
"""
track_class = STrack
def __init__(self, args):
"""Initialize a BYTETracker instance for object tracking.
Args:
args (Namespace): Command-line arguments containing tracking parameters.
"""
self.tracked_stracks: list[STrack] = []
self.lost_stracks: list[STrack] = []
self.removed_stracks: list[STrack] = []
self.frame_id = 0
self.args = args
self.max_frames_lost = args.track_buffer
self.kalman_filter = self.get_kalmanfilter()
self.reset_id()Link to this section ultralytics.trackers.byte_tracker.BYTETracker._apply_match#
def _apply_match(self, track: STrack, det: STrack, activated: list[STrack], refind: list[STrack]) -> NoneUpdate or re-activate a single track with its matched detection.
Args
| Name | Type | Description | Default |
|---|---|---|---|
track | STrack | required | |
det | STrack | required | |
activated | list[STrack] | required | |
refind | list[STrack] | required |
Source code in ultralytics/trackers/byte_tracker.py
def _apply_match(self, track: STrack, det: STrack, activated: list[STrack], refind: list[STrack]) -> None:
"""Update or re-activate a single track with its matched detection."""
if track.state == TrackState.Tracked:
track.update(det, self.frame_id)
activated.append(track)
else:
track.re_activate(det, self.frame_id, new_id=False)
refind.append(track)Link to this section ultralytics.trackers.byte_tracker.BYTETracker._apply_matches#
def _apply_matches(
self,
matches: list[list[int]] | np.ndarray,
pool: list[STrack],
detections: list[STrack],
activated: list[STrack],
refind: list[STrack],
) -> NoneApply a list of matched (track, detection) pairs from an association stage.
Args
| Name | Type | Description | Default |
|---|---|---|---|
matches | `list[list[int]] | np.ndarray` | |
pool | list[STrack] | required | |
detections | list[STrack] | required | |
activated | list[STrack] | required | |
refind | list[STrack] | required |
Source code in ultralytics/trackers/byte_tracker.py
def _apply_matches(
self,
matches: list[list[int]] | np.ndarray,
pool: list[STrack],
detections: list[STrack],
activated: list[STrack],
refind: list[STrack],
) -> None:
"""Apply a list of matched (track, detection) pairs from an association stage."""
for itracked, idet in matches:
self._apply_match(pool[itracked], detections[idet], activated, refind)Link to this section ultralytics.trackers.byte_tracker.BYTETracker._first_association#
def _first_association(
self, strack_pool: list[STrack], detections: list[STrack], activated: list[STrack], refind: list[STrack]
) -> tuple[list[int], list[int]]First-stage association between track pool and high-score detections.
Args
| Name | Type | Description | Default |
|---|---|---|---|
strack_pool | list[STrack] | required | |
detections | list[STrack] | required | |
activated | list[STrack] | required | |
refind | list[STrack] | required |
Returns
| Type | Description |
|---|---|
tuple[list[int], list[int]] | Unmatched track indices and unmatched detection indices. |
Source code in ultralytics/trackers/byte_tracker.py
def _first_association(
self, strack_pool: list[STrack], detections: list[STrack], activated: list[STrack], refind: list[STrack]
) -> tuple[list[int], list[int]]:
"""First-stage association between track pool and high-score detections.
Returns:
(tuple[list[int], list[int]]): Unmatched track indices and unmatched detection indices.
"""
dists = self.get_dists(strack_pool, detections)
matches, u_track, u_detection = matching.linear_assignment(dists, thresh=self.args.match_thresh)
self._apply_matches(matches, strack_pool, detections, activated, refind)
return u_track, u_detectionLink to this section ultralytics.trackers.byte_tracker.BYTETracker._format_output#
def _format_output(self) -> np.ndarrayFormat the current tracked objects into the output array.
Source code in ultralytics/trackers/byte_tracker.py
def _format_output(self) -> np.ndarray:
"""Format the current tracked objects into the output array."""
return np.asarray([x.result for x in self.tracked_stracks if x.is_activated], dtype=np.float32)Link to this section ultralytics.trackers.byte_tracker.BYTETracker._init_new_tracks#
def _init_new_tracks(
self,
u_detection: list[int],
detections: list[STrack],
activated: list[STrack],
refind: list[STrack] | None = None,
) -> NoneActivate new tracks from detections that survived all association stages.
Args
| Name | Type | Description | Default |
|---|---|---|---|
u_detection | list[int] | required | |
detections | list[STrack] | required | |
activated | list[STrack] | required | |
refind | `list[STrack] | None` |
Source code in ultralytics/trackers/byte_tracker.py
def _init_new_tracks(
self,
u_detection: list[int],
detections: list[STrack],
activated: list[STrack],
refind: list[STrack] | None = None,
) -> None:
"""Activate new tracks from detections that survived all association stages."""
for inew in u_detection:
track = detections[inew]
if track.score < self.args.new_track_thresh:
continue
track.activate(self.kalman_filter, self.frame_id)
activated.append(track)Link to this section ultralytics.trackers.byte_tracker.BYTETracker._input_for#
def _input_for(self, img: np.ndarray | None, feats: np.ndarray | None, mask: np.ndarray) -> AnyReturn the per-detection auxiliary input for init_track.
Default behavior preserves the legacy flow: when feats is provided it is sliced by the detection mask, otherwise the raw frame img is passed through.
Args
| Name | Type | Description | Default |
|---|---|---|---|
img | `np.ndarray | None` | Current BGR frame. |
feats | `np.ndarray | None` | Optional per-detection features. |
mask | np.ndarray | Boolean mask used to slice feats. | required |
Returns
| Type | Description |
|---|---|
Any | The auxiliary payload (features or image) to hand to init_track. |
Source code in ultralytics/trackers/byte_tracker.py
def _input_for(self, img: np.ndarray | None, feats: np.ndarray | None, mask: np.ndarray) -> Any:
"""Return the per-detection auxiliary input for ``init_track``.
Default behavior preserves the legacy flow: when ``feats`` is provided it is sliced by the
detection mask, otherwise the raw frame ``img`` is passed through.
Args:
img (np.ndarray | None): Current BGR frame.
feats (np.ndarray | None): Optional per-detection features.
mask (np.ndarray): Boolean mask used to slice ``feats``.
Returns:
(Any): The auxiliary payload (features or image) to hand to ``init_track``.
"""
if feats is not None and len(feats):
return feats[mask]
return imgLink to this section ultralytics.trackers.byte_tracker.BYTETracker._post_first_association#
def _post_first_association(
self,
strack_pool: list[STrack],
detections: list[STrack],
u_track: list[int],
u_detection: list[int],
activated: list[STrack],
refind: list[STrack],
) -> tuple[list[int], list[int]]Hook executed after the first association stage and before the second.
Args
| Name | Type | Description | Default |
|---|---|---|---|
strack_pool | list[STrack] | required | |
detections | list[STrack] | required | |
u_track | list[int] | required | |
u_detection | list[int] | required | |
activated | list[STrack] | required | |
refind | list[STrack] | required |
Returns
| Type | Description |
|---|---|
tuple[list[int], list[int]] | Potentially modified unmatched track and detection indices. |
Source code in ultralytics/trackers/byte_tracker.py
def _post_first_association(
self,
strack_pool: list[STrack],
detections: list[STrack],
u_track: list[int],
u_detection: list[int],
activated: list[STrack],
refind: list[STrack],
) -> tuple[list[int], list[int]]:
"""Hook executed after the first association stage and before the second.
Returns:
(tuple[list[int], list[int]]): Potentially modified unmatched track and detection indices.
"""
return u_track, u_detectionLink to this section ultralytics.trackers.byte_tracker.BYTETracker._pre_first_associate#
def _pre_first_associate(
self, strack_pool: list[STrack], unconfirmed: list[STrack], img: np.ndarray | None, results_high: Any
) -> NoneHook called after Kalman predict, before first-stage assignment. Default: GMC if available.
Args
| Name | Type | Description | Default |
|---|---|---|---|
strack_pool | list[STrack] | required | |
unconfirmed | list[STrack] | required | |
img | `np.ndarray | None` | |
results_high | Any | required |
Source code in ultralytics/trackers/byte_tracker.py
def _pre_first_associate(
self, strack_pool: list[STrack], unconfirmed: list[STrack], img: np.ndarray | None, results_high: Any
) -> None:
"""Hook called after Kalman predict, before first-stage assignment. Default: GMC if available."""
if hasattr(self, "gmc") and img is not None:
try:
warp = self.gmc.apply(img, results_high.xyxy)
except Exception as e:
LOGGER.warning(f"GMC failed, falling back to identity: {e}")
warp = np.eye(2, 3)
multi_gmc(strack_pool, warp)
multi_gmc(unconfirmed, warp)Link to this section ultralytics.trackers.byte_tracker.BYTETracker._remove_stale_lost#
def _remove_stale_lost(self, removed: list[STrack]) -> NoneRemove lost tracks that have exceeded the maximum allowed frames.
Args
| Name | Type | Description | Default |
|---|---|---|---|
removed | list[STrack] | required |
Source code in ultralytics/trackers/byte_tracker.py
def _remove_stale_lost(self, removed: list[STrack]) -> None:
"""Remove lost tracks that have exceeded the maximum allowed frames."""
for track in self.lost_stracks:
if self.frame_id - track.end_frame > self.max_frames_lost:
track.mark_removed()
removed.append(track)Link to this section ultralytics.trackers.byte_tracker.BYTETracker._second_association#
def _second_association(
self,
strack_pool: list[STrack],
u_track: list[int],
detections_second: list[STrack],
activated: list[STrack],
refind: list[STrack],
lost: list[STrack],
) -> NoneSecond-stage association between remaining tracked tracks and low-score detections.
Args
| Name | Type | Description | Default |
|---|---|---|---|
strack_pool | list[STrack] | required | |
u_track | list[int] | required | |
detections_second | list[STrack] | required | |
activated | list[STrack] | required | |
refind | list[STrack] | required | |
lost | list[STrack] | required |
Source code in ultralytics/trackers/byte_tracker.py
def _second_association(
self,
strack_pool: list[STrack],
u_track: list[int],
detections_second: list[STrack],
activated: list[STrack],
refind: list[STrack],
lost: list[STrack],
) -> None:
"""Second-stage association between remaining tracked tracks and low-score detections."""
r_tracked_stracks = [strack_pool[i] for i in u_track if strack_pool[i].state == TrackState.Tracked]
if r_tracked_stracks and detections_second:
dists = matching.iou_distance(r_tracked_stracks, detections_second)
if self.args.fuse_score:
dists = matching.fuse_score(dists, detections_second)
matches, u_track, _ = matching.linear_assignment(dists, thresh=0.5)
self._apply_matches(matches, r_tracked_stracks, detections_second, activated, refind)
else:
u_track = list(range(len(r_tracked_stracks)))
for it in u_track:
track = r_tracked_stracks[it]
if track.state != TrackState.Lost:
track.mark_lost()
lost.append(track)Link to this section ultralytics.trackers.byte_tracker.BYTETracker._split_detections#
def _split_detections(self, results: Any) -> tuple[Any, Any, np.ndarray, np.ndarray]Split detections into high-confidence and low-confidence subsets.
Args
| Name | Type | Description | Default |
|---|---|---|---|
results | Any | Results-like object with conf attribute supporting boolean indexing. | required |
Returns
| Type | Description |
|---|---|
tuple[Any, Any, np.ndarray, np.ndarray] | High-confidence results, low-confidence results, high mask, and |
Source code in ultralytics/trackers/byte_tracker.py
def _split_detections(self, results: Any) -> tuple[Any, Any, np.ndarray, np.ndarray]:
"""Split detections into high-confidence and low-confidence subsets.
Args:
results (Any): Results-like object with ``conf`` attribute supporting boolean indexing.
Returns:
(tuple[Any, Any, np.ndarray, np.ndarray]): High-confidence results, low-confidence results, high mask, and
low mask.
"""
scores = results.conf
remain_inds = scores >= self.args.track_high_thresh
inds_low = scores > self.args.track_low_thresh
inds_below_high = scores < self.args.track_high_thresh
return results[remain_inds], results[inds_low & inds_below_high], remain_inds, inds_low & inds_below_highLink to this section ultralytics.trackers.byte_tracker.BYTETracker._split_tracked#
def _split_tracked(self) -> tuple[list[STrack], list[STrack]]Separate self.tracked_stracks into confirmed and unconfirmed lists.
Returns
| Type | Description |
|---|---|
tuple[list[STrack], list[STrack]] | (unconfirmed, tracked) where unconfirmed holds tracks whose |
Source code in ultralytics/trackers/byte_tracker.py
def _split_tracked(self) -> tuple[list[STrack], list[STrack]]:
"""Separate ``self.tracked_stracks`` into confirmed and unconfirmed lists.
Returns:
(tuple[list[STrack], list[STrack]]): ``(unconfirmed, tracked)`` where ``unconfirmed`` holds tracks whose
``is_activated`` flag is False.
"""
unconfirmed, tracked = [], []
for track in self.tracked_stracks:
(unconfirmed if not track.is_activated else tracked).append(track)
return unconfirmed, trackedLink to this section ultralytics.trackers.byte_tracker.BYTETracker._unconfirmed_association#
def _unconfirmed_association(
self,
unconfirmed: list[STrack],
u_detection: list[int],
detections: list[STrack],
activated: list[STrack],
removed: list[STrack],
) -> tuple[list[int], list[STrack]]Associate unconfirmed tracks with leftover high-score detections.
Args
| Name | Type | Description | Default |
|---|---|---|---|
unconfirmed | list[STrack] | required | |
u_detection | list[int] | required | |
detections | list[STrack] | required | |
activated | list[STrack] | required | |
removed | list[STrack] | required |
Returns
| Type | Description |
|---|---|
tuple[list[int], list[STrack]] | Unmatched detection indices after association, and the filtered detection |
Source code in ultralytics/trackers/byte_tracker.py
def _unconfirmed_association(
self,
unconfirmed: list[STrack],
u_detection: list[int],
detections: list[STrack],
activated: list[STrack],
removed: list[STrack],
) -> tuple[list[int], list[STrack]]:
"""Associate unconfirmed tracks with leftover high-score detections.
Returns:
(tuple[list[int], list[STrack]]): Unmatched detection indices after association, and the filtered detection
list those indices refer to.
"""
detections = [detections[i] for i in u_detection]
if not unconfirmed:
return list(range(len(detections))), detections
dists = self.get_dists(unconfirmed, detections)
matches, u_unconfirmed, u_detection = matching.linear_assignment(dists, thresh=0.7)
for itracked, idet in matches:
unconfirmed[itracked].update(detections[idet], self.frame_id)
activated.append(unconfirmed[itracked])
for it in u_unconfirmed:
track = unconfirmed[it]
track.mark_removed()
removed.append(track)
return u_detection, detectionsLink to this section ultralytics.trackers.byte_tracker.BYTETracker.get_dists#
def get_dists(self, tracks: list[STrack], detections: list[STrack]) -> np.ndarrayCalculate the distance between tracks and detections using IoU and optionally fuse scores.
Args
| Name | Type | Description | Default |
|---|---|---|---|
tracks | list[STrack] | required | |
detections | list[STrack] | required |
Source code in ultralytics/trackers/byte_tracker.py
def get_dists(self, tracks: list[STrack], detections: list[STrack]) -> np.ndarray:
"""Calculate the distance between tracks and detections using IoU and optionally fuse scores."""
dists = matching.iou_distance(tracks, detections)
if self.args.fuse_score:
dists = matching.fuse_score(dists, detections)
return distsLink to this section ultralytics.trackers.byte_tracker.BYTETracker.get_kalmanfilter#
def get_kalmanfilter(self) -> KalmanFilterXYAHReturn a Kalman filter object for tracking bounding boxes using KalmanFilterXYAH.
Source code in ultralytics/trackers/byte_tracker.py
def get_kalmanfilter(self) -> KalmanFilterXYAH:
"""Return a Kalman filter object for tracking bounding boxes using KalmanFilterXYAH."""
return KalmanFilterXYAH()Link to this section ultralytics.trackers.byte_tracker.BYTETracker.init_track#
def init_track(self, results, img: np.ndarray | None = None) -> list[STrack]Initialize object tracking with given detections, scores, and class labels as STrack instances.
Args
| Name | Type | Description | Default |
|---|---|---|---|
results | required | ||
img | `np.ndarray | None` |
Source code in ultralytics/trackers/byte_tracker.py
def init_track(self, results, img: np.ndarray | None = None) -> list[STrack]:
"""Initialize object tracking with given detections, scores, and class labels as STrack instances."""
if len(results) == 0:
return []
bboxes = parse_bboxes(results)
return [self.track_class(xywh, s, c) for (xywh, s, c) in zip(bboxes, results.conf, results.cls)]Link to this section ultralytics.trackers.byte_tracker.BYTETracker.multi_predict#
def multi_predict(self, tracks: list[STrack])Predict the next states for multiple tracks using Kalman filter.
Args
| Name | Type | Description | Default |
|---|---|---|---|
tracks | list[STrack] | required |
Source code in ultralytics/trackers/byte_tracker.py
def multi_predict(self, tracks: list[STrack]):
"""Predict the next states for multiple tracks using Kalman filter."""
STrack.multi_predict(tracks)Link to this section ultralytics.trackers.byte_tracker.BYTETracker.reset#
def reset(self)Reset the tracker by clearing all tracked, lost, and removed tracks and reinitializing the Kalman filter.
Source code in ultralytics/trackers/byte_tracker.py
def reset(self):
"""Reset the tracker by clearing all tracked, lost, and removed tracks and reinitializing the Kalman filter."""
self.tracked_stracks = []
self.lost_stracks = []
self.removed_stracks = []
self.frame_id = 0
self.kalman_filter = self.get_kalmanfilter()
self.reset_id()Link to this section ultralytics.trackers.byte_tracker.BYTETracker.reset_id#
def reset_id()Reset the ID counter for STrack instances to ensure unique track IDs across tracking sessions.
Source code in ultralytics/trackers/byte_tracker.py
@staticmethod
def reset_id():
"""Reset the ID counter for STrack instances to ensure unique track IDs across tracking sessions."""
STrack.reset_id()Link to this section ultralytics.trackers.byte_tracker.BYTETracker.update#
def update(self, results, img: np.ndarray | None = None, feats: np.ndarray | None = None, **kwargs) -> np.ndarrayUpdate the tracker with new detections and return the current list of tracked objects.
Args
| Name | Type | Description | Default |
|---|---|---|---|
results | required | ||
img | `np.ndarray | None` | |
feats | `np.ndarray | None` | |
**kwargs | required |
Source code in ultralytics/trackers/byte_tracker.py
def update(self, results, img: np.ndarray | None = None, feats: np.ndarray | None = None, **kwargs) -> np.ndarray:
"""Update the tracker with new detections and return the current list of tracked objects."""
self.frame_id += 1
activated_stracks = []
refind_stracks = []
lost_stracks = []
removed_stracks = []
results_high, results_low, mask_high, mask_low = self._split_detections(results)
detections = self.init_track(results_high, self._input_for(img, feats, mask_high))
detections_second = self.init_track(results_low, self._input_for(img, feats, mask_low))
unconfirmed, tracked_stracks = self._split_tracked()
strack_pool = joint_stracks(tracked_stracks, self.lost_stracks)
self.multi_predict(strack_pool)
self._pre_first_associate(strack_pool, unconfirmed, img, results_high)
u_track, u_detection = self._first_association(strack_pool, detections, activated_stracks, refind_stracks)
u_track, u_detection = self._post_first_association(
strack_pool, detections, u_track, u_detection, activated_stracks, refind_stracks
)
self._second_association(
strack_pool, u_track, detections_second, activated_stracks, refind_stracks, lost_stracks
)
u_detection, detections = self._unconfirmed_association(
unconfirmed, u_detection, detections, activated_stracks, removed_stracks
)
self._init_new_tracks(u_detection, detections, activated_stracks, refind_stracks)
self._remove_stale_lost(removed_stracks)
merge_track_pools(self, activated_stracks, refind_stracks, lost_stracks, removed_stracks)
return self._format_output()