boofcv.abst.tracker.DetectDescribeAssociateTwoPass Maven / Gradle / Ivy
/*
* Copyright (c) 2011-2020, Peter Abeles. All Rights Reserved.
*
* This file is part of BoofCV (http://boofcv.org).
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package boofcv.abst.tracker;
import boofcv.abst.feature.associate.AssociateDescription2D;
import boofcv.struct.feature.AssociatedIndex;
import boofcv.struct.feature.TupleDesc;
import boofcv.struct.image.ImageGray;
import georegression.struct.point.Point2D_F64;
import org.ddogleg.struct.FastAccess;
/**
* Changes behavior of {@link DetectDescribeAssociate} so that it conforms to the {@link PointTrackerTwoPass} interface.
* It can now take hints for where tracks might appear in the image. If possible
* {@link AssociateDescription2D#setSource} will only be called once
* on the second pass.
*
* @author Peter Abeles
*/
public class DetectDescribeAssociateTwoPass, Desc extends TupleDesc>
extends DetectDescribeAssociate implements PointTrackerTwoPass
{
// associate used in the second pass
AssociateDescription2D associate2;
// has source been set in associate for the second pass
boolean sourceSet2;
/**
* Configure the tracker. The parameters associate and associate2 can be the same instance.
*
* @param manager Feature manager
* @param associate Association algorithm for the first pass
* @param associate2 Association algorithm for the second pass
* @param config Configures behavior
*/
public DetectDescribeAssociateTwoPass(DdaFeatureManager manager,
AssociateDescription2D associate,
AssociateDescription2D associate2,
ConfigTrackerDda config )
{
super(manager, associate, config);
this.associate2 = associate2;
}
@Override
public void process( I input ) {
frameID++;
sourceSet2 = false;
tracksActive.clear();
tracksInactive.clear();
tracksDropped.clear();
tracksNew.clear();
manager.detectFeatures(input);
for (int setIndex = 0; setIndex < sets.length; setIndex++) {
SetTrackInfo info = sets[setIndex];
info.featDst.reset();
info.locDst.reset();
manager.getFeatures(setIndex,info.locDst,info.featDst);
// skip if there are no features
if( !tracksAll.isEmpty() ) {
putIntoSrcList(info);
associate.setSource(info.locSrc, info.featSrc);
associate.setDestination(info.locDst, info.featDst);
associate.associate();
updateTrackLocation(info,associate.getMatches());
}
}
}
@Override
public void performSecondPass() {
if( tracksAll.isEmpty() )
return;
boolean setSource = associate2 != associate && !sourceSet2 && sets.length==1;
if( setSource ) {
sourceSet2 = true;
}
for (int setIndex = 0; setIndex < sets.length; setIndex++) {
SetTrackInfo info = sets[setIndex];
// minimize the number of times set source is called. In some implementations of associate this is an
// expensive operation
if (setSource) {
associate.setSource(info.locSrc, info.featSrc);
}
associate2.setDestination(info.locDst, info.featDst);
associate2.associate();
updateTrackLocation(info,associate2.getMatches());
}
}
@Override
public void finishTracking() {
if( tracksAll.isEmpty() )
return;
// Update the track state using association information
tracksActive.clear();
for (int setIndex = 0; setIndex < sets.length; setIndex++) {
SetTrackInfo info = sets[setIndex];
updateTrackState(info);
// add unassociated to the list
for (int i = 0; i < info.tracks.size(); i++) {
if (!info.isAssociated[i])
tracksInactive.add(info.tracks.get(i));
}
}
}
/**
* Update each track's location only and not its description. Update the active list too
*/
protected void updateTrackLocation( SetTrackInfo info, FastAccess matches) {
info.matches.resize(matches.size);
for (int i = 0; i < matches.size; i++) {
info.matches.get(i).set(matches.get(i));
}
tracksActive.clear();
for( int i = 0; i < info.matches.size; i++ ) {
AssociatedIndex indexes = info.matches.data[i];
PointTrack track = info.tracks.get(indexes.src);
Point2D_F64 loc = info.locDst.data[indexes.dst];
track.pixel.set(loc.x, loc.y);
tracksActive.add(track);
}
}
@Override
public void setHint( double pixelX , double pixelY , PointTrack track ) {
track.pixel.set(pixelX,pixelY);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy