#include #include #include #include #include #include #include #define CMDLINEMAX 10 #define ASSESS_TILL INT_MAX using namespace std; using namespace cv; /* TODO: do normalization ala Kalal's assessment protocol for TLD */ static Mat image; static bool paused; vector palette; void print_table(char* videos[],int videoNum,char* algorithms[],int algNum,const vector >& results); static void listTrackers(){ vector algorithms; Algorithm::getList(algorithms); cout << "\nAvailable tracker algorithms:\n"; for (size_t i=0; i < algorithms.size(); i++){ const char* algoname=algorithms[i].c_str(); char *pos=NULL; if((pos=strstr((char*)algoname,"TRACKER."))!=NULL){ printf("%s\n",pos+8); } } } static int lineToRect(char* line,Rect2d& res){ char * ptr=line,*pos=ptr; if(line==NULL || line[0]=='\0'){ return -1; } if(strcmp(line,"NaN,NaN,NaN,NaN\n")==0){ res.height=res.width=-1.0; return 0; } double nums[4]={0}; for(int i=0; i<4 && (ptr=strpbrk(ptr,"0123456789-"))!= NULL;i++,ptr=pos){ nums[i]=strtod(ptr,&pos); if(pos==ptr){ printf("lineToRect had problems with decoding line %s\n",line); return -1; } } res.x=cv::min(nums[0],nums[2]); res.y=cv::min(nums[1],nums[3]); res.width=cv::abs(nums[0]-nums[2]); res.height=cv::abs(nums[1]-nums[3]); return 0; } static inline double overlap(Rect2d r1,Rect2d r2){ if(r1.width<0 || r2.width<0 || r1.height<0 || r1.width<0)return -1.0; double a1=r1.area(), a2=r2.area(), a0=(r1&r2).area(); return a0/(a1+a2-a0); } static void help(){ cout << "\nThis example shows the functionality of \"Long-term optical tracking API\"" "-- pause video [p] and draw a bounding box around the target to start the tracker\n" "Example of is in opencv_extra/testdata/cv/tracking/\n" "Call:\n" "./tracker [] ...\n" << endl; cout << "\n\nHot keys: \n" "\tq - quit the program\n" "\tp - pause video\n"; listTrackers(); exit(EXIT_SUCCESS); } static void parseCommandLineArgs(int argc, char** argv,char* videos[],char* gts[], int* vc,char* algorithms[],char* initBoxes[][CMDLINEMAX],int* ac){ vector trackers; Algorithm::getList(trackers); for(int i=0;i<(int)trackers.size();i++){ if(strstr(trackers[i].c_str(),"TRACKER.")!=trackers[i].c_str()){ trackers.erase(trackers.begin()+i); i--; } } *ac=*vc=0; for(int i=1;i >& results){ vector grid(1+algNum,0); char spaces[100];memset(spaces,' ',100); for(int i=0;i > >results; }; class CorrectFrames : public AssessmentRes::Assessment{ public: CorrectFrames(double tol):tol_(tol),len_(1),correctFrames_(1){} int printf(char* buf){return sprintf(buf,"%d/%d",correctFrames_,len_);} void printName(){printf((char*)"Num of correct frames\n");} void assess(const Rect2d& ethalon,const Rect2d& res){len_++;if(overlap(ethalon,res)>tol_)correctFrames_++;} private: double tol_; int len_; int correctFrames_; }; class AvgTime : public AssessmentRes::Assessment{ public: AvgTime(double res):res_(res){} int printf(char* buf){return sprintf(buf,"%gms",res_);} void printName(){printf((char*)"Average frame tracking time\n");} void assess(const Rect2d& /*ethalon*/,const Rect2d&/* res*/){}; private: double res_; }; class PRF : public AssessmentRes::Assessment{ public: PRF():occurences_(0),responses_(0),true_responses_(0){}; void printName(){printf((char*)"PRF\n");} int printf(char* buf){return sprintf(buf,"%g/%g/%g",(1.0*true_responses_)/responses_,(1.0*true_responses_)/occurences_, (2.0*true_responses_)/(responses_+occurences_));} void assess(const Rect2d& ethalon,const Rect2d& res){ if(res.height>=0)responses_++; if(ethalon.height>=0)occurences_++; if(ethalon.height>=0 && res.height>=0)true_responses_++; } private: int occurences_,responses_,true_responses_; }; AssessmentRes::AssessmentRes(int algnum):len(0),results(algnum){ for(int i=0;i<(int)results.size();i++){ #if !0 results[i].push_back(Ptr(new CorrectFrames(0.0))); #else results[i].push_back(Ptr(new CorrectFrames(0.5))); #endif results[i].push_back(Ptr(new PRF())); } } static AssessmentRes assessment(char* video,char* gt_str, char* algorithms[],char* initBoxes_str[],int algnum){ char buf[200]; int start_frame=0; int linecount=0; Rect2d boundingBox; vector averageMillisPerFrame(algnum,0.0); FILE* gt=fopen(gt_str,"r"); if(gt==NULL){ printf("cannot open the ground truth file %s\n",gt_str); exit(EXIT_FAILURE); } for(linecount=0;fgets(buf,sizeof(buf),gt)!=NULL;linecount++); if(linecount==0){ printf("ground truth file %s has no lines\n",gt_str); exit(EXIT_FAILURE); } fseek(gt,0,SEEK_SET); if(fgets(buf,sizeof(buf),gt)==NULL){ printf("ground truth file %s has no lines\n",gt_str); exit(EXIT_FAILURE); } std::vector initBoxes(algnum); for(int i=0;i >trackers(algnum); for(int i=0;i> frame; frame.copyTo( image ); if(lineToRect(buf,boundingBox)<0){ if(gt!=NULL){ fclose(gt); } exit(EXIT_FAILURE); } rectangle( image, boundingBox,palette[0], 2, 1 ); for(int i=0;i<(int)trackers.size();i++){ rectangle(image,initBoxes[i],palette[i+1], 2, 1 ); if( !trackers[i]->init( frame, initBoxes[i] ) ){ printf("could not initialize tracker %s with box %s at video %s\n",algorithms[i],initBoxes_str[i],video); if(gt!=NULL){ fclose(gt); } exit(EXIT_FAILURE); } } imshow( "Tracking API", image ); int frameCounter = 0; AssessmentRes res((int)trackers.size()); for ( ;; ){ if( !paused ){ cap >> frame; if(frame.empty()){ break; } frame.copyTo( image ); if(fgets(buf,sizeof(buf),gt)==NULL){ printf("ground truth is over\n"); break; } if(lineToRect(buf,boundingBox)<0){ if(gt!=NULL){ fclose(gt); } exit(EXIT_FAILURE); } rectangle( image, boundingBox,palette[0], 2, 1 ); frameCounter++; for(int i=0;i<(int)trackers.size();i++){ bool trackerRes=true; clock_t start;start=clock(); trackerRes=trackers[i]->update( frame, initBoxes[i] ); start=clock()-start; averageMillisPerFrame[i]+=1000.0*start/CLOCKS_PER_SEC; if(trackerRes==false){ initBoxes[i].height=initBoxes[i].width=-1.0; }else{ rectangle( image, initBoxes[i], palette[i+1], 2, 1 ); } #if !1 if(i==1){ printf("TLD\n"); printf("boundingBox=[%f,%f,%f,%f]\n",boundingBox.x,boundingBox.y,boundingBox.width,boundingBox.height); printf("initBoxes[i]=[%f,%f,%f,%f]\n",initBoxes[i].x,initBoxes[i].y,initBoxes[i].width,initBoxes[i].height); printf("overlap=%f\n",overlap(initBoxes[i],boundingBox)); exit(0); } #endif for(int j=0;j<(int)res.results[i].size();j++) res.results[i][j]->assess(boundingBox,initBoxes[i]); } imshow( "Tracking API", image ); if(frameCounter>=ASSESS_TILL){ break; } char c = (char) waitKey( 2 ); if( c == 'q' ) break; if( c == 'p' ) paused = !paused; } } if(gt!=NULL){ fclose(gt); } destroyWindow( "Tracking API"); res.len=linecount; res.videoName=video; for(int i=0;i<(int)res.results.size();i++) res.results[i].push_back(Ptr(new AvgTime(averageMillisPerFrame[i]/res.len))); return res; } int main( int argc, char** argv ){ palette.push_back(Scalar(255,0,0));//BGR palette.push_back(Scalar(0,0,255)); palette.push_back(Scalar(0,255,255)); int vcount=0,acount=0; char* videos[CMDLINEMAX],*gts[CMDLINEMAX],*algorithms[CMDLINEMAX],*initBoxes[CMDLINEMAX][CMDLINEMAX]; parseCommandLineArgs(argc,argv,videos,gts,&vcount,algorithms,initBoxes,&acount); CV_Assert(acount results; for(int i=0;i > resultStrings(vcount); for(int i=0;iprintf(resultStrings[videoCount][algoCount]); } print_table(videos,vcount,algorithms,acount,resultStrings); } return 0; }