/*M/////////////////////////////////////////////////////////////////////////////////////// // // IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING. // // By downloading, copying, installing or using the software you agree to this license. // If you do not agree to this license, do not download, install, // copy or use the software. // // // License Agreement // For Open Source Computer Vision Library // // Copyright (C) 2012, Willow Garage Inc., all rights reserved. // Third party copyrights are property of their respective owners. // // Redistribution and use in source and binary forms, with or without modification, // are permitted provided that the following conditions are met: // // * Redistribution's of source code must retain the above copyright notice, // this list of conditions and the following disclaimer. // // * Redistribution's in binary form must reproduce the above copyright notice, // this list of conditions and the following disclaimer in the documentation // and/or other materials provided with the distribution. // // * The names of the copyright holders may not be used to endorse or promote products // derived from this software without specific prior written permission. // // This software is provided by the copyright holders and contributors "as is" and // any express or implied warranties, including, but not limited to, the implied // warranties of merchantability and fitness for a particular purpose are disclaimed. // In no event shall the Intel Corporation or contributors be liable for any direct, // indirect, incidental, special, exemplary, or consequential damages // (including, but not limited to, procurement of substitute goods or services; // loss of use, data, or profits; or business interruption) however caused // and on any theory of liability, whether in contract, strict liability, // or tort (including negligence or otherwise) arising in any way out of // the use of this software, even if advised of the possibility of such damage. // //M*/ /******************************************************************************************* The LogPolar Blind Spot Model code has been contributed by Fabio Solari and Manuela Chessa. More details can be found in: M. Chessa, S. P. Sabatini, F. Solari and F. Tatti (2011) A Quantitative Comparison of Speed and Reliability for Log-Polar Mapping Techniques, Computer Vision Systems - 8th International Conference, ICVS 2011, Sophia Antipolis, France, September 20-22, 2011 (http://dx.doi.org/10.1007/978-3-642-23968-7_5) ********************************************************************************************/ #include "precomp.hpp" #include #include namespace cv { //------------------------------------interp------------------------------------------- LogPolar_Interp::LogPolar_Interp(int w, int h, Point2i center, int R, double ro0, int interp, int full, int S, int sp) { if ( (center.x!=w/2 || center.y!=h/2) && full==0) full=1; if (center.x<0) center.x=0; if (center.y<0) center.y=0; if (center.x>=w) center.x=w-1; if (center.y>=h) center.y=h-1; if (full){ int rtmp; if (center.x<=w/2 && center.y>=h/2) rtmp=(int)sqrt((float)center.y*center.y + (float)(w-center.x)*(w-center.x)); else if (center.x>=w/2 && center.y>=h/2) rtmp=(int)sqrt((float)center.y*center.y + (float)center.x*center.x); else if (center.x>=w/2 && center.y<=h/2) rtmp=(int)sqrt((float)(h-center.y)*(h-center.y) + (float)center.x*center.x); else //if (center.x<=w/2 && center.y<=h/2) rtmp=(int)sqrt((float)(h-center.y)*(h-center.y) + (float)(w-center.x)*(w-center.x)); M=2*rtmp; N=2*rtmp; top = M/2 - center.y; bottom = M/2 - (h-center.y); left = M/2 - center.x; right = M/2 - (w - center.x); }else{ top=bottom=left=right=0; M=w; N=h; } if (sp){ int jc=M/2-1, ic=N/2-1; int romax=min(ic, jc); double a=exp(log((double)(romax/2-1)/(double)ro0)/(double)R); S=(int) floor(2*M_PI/(a-1)+0.5); } this->interp=interp; create_map(M, N, R, S, ro0); } void LogPolar_Interp::create_map(int M, int N, int R, int S, double ro0) { this->M=M; this->N=N; this->R=R; this->S=S; this->ro0=ro0; int jc=N/2-1, ic=M/2-1; romax=min(ic, jc); a=exp(log((double)romax/(double)ro0)/(double)R); q=((double)S)/(2*M_PI); Rsri = Mat::zeros(S,R,CV_32FC1); Csri = Mat::zeros(S,R,CV_32FC1); ETAyx = Mat::zeros(N,M,CV_32FC1); CSIyx = Mat::zeros(N,M,CV_32FC1); for(int v=0; v(v,u)=(float)(ro0*pow(a,u)*sin(v/q)+jc); Csri.at(v,u)=(float)(ro0*pow(a,u)*cos(v/q)+ic); } } for(int j=0; j=ic) theta=atan((double)(j-jc)/(double)(i-ic)); else theta=atan((double)(j-jc)/(double)(i-ic))+M_PI; if(theta<0) theta+=2*M_PI; ETAyx.at(j,i)=(float)(q*theta); double ro2=(j-jc)*(j-jc)+(i-ic)*(i-ic); CSIyx.at(j,i)=(float)(0.5*log(ro2/(ro0*ro0))/log(a)); } } } const Mat LogPolar_Interp::to_cortical(const Mat &source) { Mat out(S,R,CV_8UC1,Scalar(0)); Mat source_border; copyMakeBorder(source,source_border,top,bottom,left,right,BORDER_CONSTANT,Scalar(0)); remap(source_border,out,Csri,Rsri,interp); return out; } const Mat LogPolar_Interp::to_cartesian(const Mat &source) { Mat out(N,M,CV_8UC1,Scalar(0)); Mat source_border; if (interp==INTER_NEAREST || interp==INTER_LINEAR){ copyMakeBorder(source,source_border,0,1,0,0,BORDER_CONSTANT,Scalar(0)); Mat rowS0 = source_border.row(S); source_border.row(0).copyTo(rowS0); } else if (interp==INTER_CUBIC){ copyMakeBorder(source,source_border,0,2,0,0,BORDER_CONSTANT,Scalar(0)); Mat rowS0 = source_border.row(S); Mat rowS1 = source_border.row(S+1); source_border.row(0).copyTo(rowS0); source_border.row(1).copyTo(rowS1); } else if (interp==INTER_LANCZOS4){ copyMakeBorder(source,source_border,0,4,0,0,BORDER_CONSTANT,Scalar(0)); Mat rowS0 = source_border.row(S); Mat rowS1 = source_border.row(S+1); Mat rowS2 = source_border.row(S+2); Mat rowS3 = source_border.row(S+3); source_border.row(0).copyTo(rowS0); source_border.row(1).copyTo(rowS1); source_border.row(2).copyTo(rowS2); source_border.row(3).copyTo(rowS3); } remap(source_border,out,CSIyx,ETAyx,interp); Mat out_cropped=out(Range(top,N-1-bottom),Range(left,M-1-right)); return out_cropped; } LogPolar_Interp::~LogPolar_Interp() { } //------------------------------------overlapping---------------------------------- LogPolar_Overlapping::LogPolar_Overlapping(int w, int h, Point2i center, int R, double ro0, int full, int S, int sp) { if ( (center.x!=w/2 || center.y!=h/2) && full==0) full=1; if (center.x<0) center.x=0; if (center.y<0) center.y=0; if (center.x>=w) center.x=w-1; if (center.y>=h) center.y=h-1; if (full){ int rtmp; if (center.x<=w/2 && center.y>=h/2) rtmp=(int)sqrt((float)center.y*center.y + (float)(w-center.x)*(w-center.x)); else if (center.x>=w/2 && center.y>=h/2) rtmp=(int)sqrt((float)center.y*center.y + (float)center.x*center.x); else if (center.x>=w/2 && center.y<=h/2) rtmp=(int)sqrt((float)(h-center.y)*(h-center.y) + (float)center.x*center.x); else //if (center.x<=w/2 && center.y<=h/2) rtmp=(int)sqrt((float)(h-center.y)*(h-center.y) + (float)(w-center.x)*(w-center.x)); M=2*rtmp; N=2*rtmp; top = M/2 - center.y; bottom = M/2 - (h-center.y); left = M/2 - center.x; right = M/2 - (w - center.x); }else{ top=bottom=left=right=0; M=w; N=h; } if (sp){ int jc=M/2-1, ic=N/2-1; int romax=min(ic, jc); double a=exp(log((double)(romax/2-1)/(double)ro0)/(double)R); S=(int) floor(2*M_PI/(a-1)+0.5); } create_map(M, N, R, S, ro0); } void LogPolar_Overlapping::create_map(int M, int N, int R, int S, double ro0) { this->M=M; this->N=N; this->R=R; this->S=S; this->ro0=ro0; int jc=N/2-1, ic=M/2-1; romax=min(ic, jc); a=exp(log((double)romax/(double)ro0)/(double)R); q=((double)S)/(2*M_PI); ind1=0; Rsri=Mat::zeros(S,R,CV_32FC1); Csri=Mat::zeros(S,R,CV_32FC1); ETAyx=Mat::zeros(N,M,CV_32FC1); CSIyx=Mat::zeros(N,M,CV_32FC1); Rsr.resize(R*S); Csr.resize(R*S); Wsr.resize(R); w_ker_2D.resize(R*S); for(int v=0; v(v,u)=(float)(ro0*pow(a,u)*sin(v/q)+jc); Csri.at(v,u)=(float)(ro0*pow(a,u)*cos(v/q)+ic); Rsr[v*R+u]=(int)floor(Rsri.at(v,u)); Csr[v*R+u]=(int)floor(Csri.at(v,u)); } } bool done=false; for(int i=0; i1)&&(done==false)) { ind1=i; done =true; } } for(int j=0; j=ic) theta=atan((double)(j-jc)/(double)(i-ic)); else theta=atan((double)(j-jc)/(double)(i-ic))+M_PI; if(theta<0) theta+=2*M_PI; ETAyx.at(j,i)=(float)(q*theta); double ro2=(j-jc)*(j-jc)+(i-ic)*(i-ic); CSIyx.at(j,i)=(float)(0.5*log(ro2/(ro0*ro0))/log(a)); } } for(int v=0; v(v,u)-Csr[v*R+u]; double dy=Rsri.at(v,u)-Rsr[v*R+u]; double tot=0; for(int j=0; j<2*w+1; j++) for(int i=0; i<2*w+1; i++) { (w_ker_2D[v*R+u].weights)[j*(2*w+1)+i]=exp(-(pow(i-w-dx, 2)+pow(j-w-dy, 2))/(2*sigma*sigma)); tot+=(w_ker_2D[v*R+u].weights)[j*(2*w+1)+i]; } for(int j=0; j<(2*w+1); j++) for(int i=0; i<(2*w+1); i++) (w_ker_2D[v*R+u].weights)[j*(2*w+1)+i]/=tot; } } const Mat LogPolar_Overlapping::to_cortical(const Mat &source) { Mat out(S,R,CV_8UC1,Scalar(0)); Mat source_border; copyMakeBorder(source,source_border,top,bottom,left,right,BORDER_CONSTANT,Scalar(0)); remap(source_border,out,Csri,Rsri,INTER_LINEAR); int wm=w_ker_2D[R-1].w; vector IMG((M+2*wm+1)*(N+2*wm+1), 0); for(int j=0; j(j,i); for(int v=0; v(v,u)=(uchar) floor(tmp+0.5); } return out; } const Mat LogPolar_Overlapping::to_cartesian(const Mat &source) { Mat out(N,M,CV_8UC1,Scalar(0)); Mat source_border; copyMakeBorder(source,source_border,0,1,0,0,BORDER_CONSTANT,Scalar(0)); Mat rowS = source_border.row(S); source_border.row(0).copyTo(rowS); remap(source_border,out,CSIyx,ETAyx,INTER_LINEAR); int wm=w_ker_2D[R-1].w; vector IMG((N+2*wm+1)*(M+2*wm+1), 0.); vector NOR((N+2*wm+1)*(M+2*wm+1), 0.); for(int v=0; v(v, u); NOR[ind]+=((w_ker_2D[v*R+u]).weights[j*(2*w+1)+i]); } } } for(int i=0; i<((N+2*wm+1)*(M+2*wm+1)); i++) IMG[i]/=NOR[i]; //int xc=M/2-1, yc=N/2-1; for(int j=wm; j0) ret[M*(j-wm)+i-wm]=(int) floor(IMG[(M+2*wm+1)*j+i]+0.5);*/ //int ro=(int)floor(sqrt((double)((j-wm-yc)*(j-wm-yc)+(i-wm-xc)*(i-wm-xc)))); int csi=(int) floor(CSIyx.at(j-wm,i-wm)); if((csi>=(ind1-(w_ker_2D[ind1]).w))&&(csi(j-wm,i-wm)=(uchar) floor(IMG[(M+2*wm+1)*j+i]+0.5); } Mat out_cropped=out(Range(top,N-1-bottom),Range(left,M-1-right)); return out_cropped; } LogPolar_Overlapping::~LogPolar_Overlapping() { } //----------------------------------------adjacent--------------------------------------- LogPolar_Adjacent::LogPolar_Adjacent(int w, int h, Point2i center, int R, double ro0, double smin, int full, int S, int sp) { if ( (center.x!=w/2 || center.y!=h/2) && full==0) full=1; if (center.x<0) center.x=0; if (center.y<0) center.y=0; if (center.x>=w) center.x=w-1; if (center.y>=h) center.y=h-1; if (full){ int rtmp; if (center.x<=w/2 && center.y>=h/2) rtmp=(int)sqrt((float)center.y*center.y + (float)(w-center.x)*(w-center.x)); else if (center.x>=w/2 && center.y>=h/2) rtmp=(int)sqrt((float)center.y*center.y + (float)center.x*center.x); else if (center.x>=w/2 && center.y<=h/2) rtmp=(int)sqrt((float)(h-center.y)*(h-center.y) + (float)center.x*center.x); else //if (center.x<=w/2 && center.y<=h/2) rtmp=(int)sqrt((float)(h-center.y)*(h-center.y) + (float)(w-center.x)*(w-center.x)); M=2*rtmp; N=2*rtmp; top = M/2 - center.y; bottom = M/2 - (h-center.y); left = M/2 - center.x; right = M/2 - (w - center.x); }else{ top=bottom=left=right=0; M=w; N=h; } if (sp){ int jc=M/2-1, ic=N/2-1; int romax=min(ic, jc); double a=exp(log((double)(romax/2-1)/(double)ro0)/(double)R); S=(int) floor(2*M_PI/(a-1)+0.5); } create_map(M, N, R, S, ro0, smin); } void LogPolar_Adjacent::create_map(int M, int N, int R, int S, double ro0, double smin) { LogPolar_Adjacent::M=M; LogPolar_Adjacent::N=N; LogPolar_Adjacent::R=R; LogPolar_Adjacent::S=S; LogPolar_Adjacent::ro0=ro0; romax=min(M/2.0, N/2.0); a=exp(log(romax/ro0)/(double)R); q=S/(2*M_PI); A.resize(R*S); L.resize(M*N); for(int i=0; ismin) { double xs[4], ys[4]; int us[4], vs[4]; xs[0]=xs[3]=x+length/4.0; xs[1]=xs[2]=x-length/4.0; ys[1]=ys[0]=y+length/4.0; ys[2]=ys[3]=y-length/4.0; for(int z=0; z<4; z++) get_uv(xs[z], ys[z], us[z], vs[z]); bool c=true; for(int w=1; w<4; w++) { if(us[w]!=us[w-1]) c=false; if(vs[w]!=vs[w-1]) c=false; } if(c) { if(us[0]!=-1) { pixel p; p.u=us[0]; p.v=vs[0]; p.a=length*length; L[M*j+i].push_back(p); A[vs[0]*R+us[0]]+=length*length; } } else { for(int z=0; z<4; z++) if(us[z]!=-1) subdivide_recursively(xs[z], ys[z], i, j, length/2.0, smin); } } } const Mat LogPolar_Adjacent::to_cortical(const Mat &source) { Mat source_border; copyMakeBorder(source,source_border,top,bottom,left,right,BORDER_CONSTANT,Scalar(0)); vector map(R*S, 0.); for(int j=0; j(j,i)); } } for(int i=0; i(i,j)=(uchar) floor(map[i*R+j]+0.5); return out; } const Mat LogPolar_Adjacent::to_cartesian(const Mat &source) { vector map(M*N, 0.); for(int j=0; j((L[M*j+i])[z].v,(L[M*j+i])[z].u); } } Mat out(N,M,CV_8UC1,Scalar(0)); for(int i=0; i(i,j)=(uchar) floor(map[i*M+j]+0.5); Mat out_cropped=out(Range(top,N-1-bottom),Range(left,M-1-right)); return out_cropped; } bool LogPolar_Adjacent::get_uv(double x, double y, int&u, int&v) { double ro=sqrt(x*x+y*y), theta; if(x>0) theta=atan(y/x); else theta=atan(y/x)+M_PI; if(roromax) { u=-1; v=-1; return false; } else { u= (int) floor(log(ro/ro0)/log(a)); if(theta>=0) v= (int) floor(q*theta); else v= (int) floor(q*(theta+2*M_PI)); return true; } } LogPolar_Adjacent::~LogPolar_Adjacent() { } }