15c6c1daeSBarry Smith 2999739cfSJacob Faibussowitsch #include <petsc/private/drawimpl.h> /*I "petscdraw.h" I*/ 35c6c1daeSBarry Smith 45c6c1daeSBarry Smith /* 55c6c1daeSBarry Smith val is the label value. sep is the separation to the next (or previous) 65c6c1daeSBarry Smith label; this is useful in determining how many significant figures to 75c6c1daeSBarry Smith keep. 85c6c1daeSBarry Smith */ 95c6c1daeSBarry Smith PetscErrorCode PetscADefLabel(PetscReal val,PetscReal sep,char **p) 105c6c1daeSBarry Smith { 115c6c1daeSBarry Smith static char buf[40]; 125c6c1daeSBarry Smith 135c6c1daeSBarry Smith PetscFunctionBegin; 145c6c1daeSBarry Smith /* Find the string */ 15ba129bcfSBarry Smith if (PetscAbsReal(val)/sep < 1.e-4) { 165c6c1daeSBarry Smith buf[0] = '0'; buf[1] = 0; 1736c9bc0dSBarry Smith } else { 184d25b1e8SKarl Rupp sprintf(buf,"%0.1e",(double)val); 19*9566063dSJacob Faibussowitsch PetscCall(PetscStripZerosPlus(buf)); 20*9566063dSJacob Faibussowitsch PetscCall(PetscStripe0(buf)); 21*9566063dSJacob Faibussowitsch PetscCall(PetscStripInitialZero(buf)); 22*9566063dSJacob Faibussowitsch PetscCall(PetscStripAllZeros(buf)); 23*9566063dSJacob Faibussowitsch PetscCall(PetscStripTrailingZeros(buf)); 245c6c1daeSBarry Smith } 255c6c1daeSBarry Smith *p = buf; 265c6c1daeSBarry Smith PetscFunctionReturn(0); 275c6c1daeSBarry Smith } 285c6c1daeSBarry Smith 295c6c1daeSBarry Smith /* Finds "nice" locations for the ticks */ 305c6c1daeSBarry Smith PetscErrorCode PetscADefTicks(PetscReal low,PetscReal high,int num,int *ntick,PetscReal *tickloc,int maxtick) 315c6c1daeSBarry Smith { 325c6c1daeSBarry Smith int i,power; 33e5ab1681SLisandro Dalcin PetscReal x = 0.0,base=0.0,eps; 345c6c1daeSBarry Smith 355c6c1daeSBarry Smith PetscFunctionBegin; 36*9566063dSJacob Faibussowitsch PetscCall(PetscAGetBase(low,high,num,&base,&power)); 37*9566063dSJacob Faibussowitsch PetscCall(PetscAGetNice(low,base,-1,&x)); 385c6c1daeSBarry Smith 395c6c1daeSBarry Smith /* Values are of the form j * base */ 405c6c1daeSBarry Smith /* Find the starting value */ 415c6c1daeSBarry Smith if (x < low) x += base; 425c6c1daeSBarry Smith 43e5ab1681SLisandro Dalcin i = 0; eps = base/10; 44e5ab1681SLisandro Dalcin while (i < maxtick && x <= high+eps) { 45e5ab1681SLisandro Dalcin tickloc[i++] = x; x += base; 465c6c1daeSBarry Smith } 475c6c1daeSBarry Smith *ntick = i; 48e5ab1681SLisandro Dalcin tickloc[i-1] = PetscMin(tickloc[i-1],high); 495c6c1daeSBarry Smith 505c6c1daeSBarry Smith if (i < 2 && num < 10) { 51*9566063dSJacob Faibussowitsch PetscCall(PetscADefTicks(low,high,num+1,ntick,tickloc,maxtick)); 525c6c1daeSBarry Smith } 535c6c1daeSBarry Smith PetscFunctionReturn(0); 545c6c1daeSBarry Smith } 555c6c1daeSBarry Smith 565c6c1daeSBarry Smith #define EPS 1.e-6 575c6c1daeSBarry Smith 585c6c1daeSBarry Smith PetscErrorCode PetscExp10(PetscReal d,PetscReal *result) 595c6c1daeSBarry Smith { 605c6c1daeSBarry Smith PetscFunctionBegin; 6185ec1a3cSBarry Smith *result = PetscPowReal((PetscReal)10.0,d); 625c6c1daeSBarry Smith PetscFunctionReturn(0); 635c6c1daeSBarry Smith } 645c6c1daeSBarry Smith 655c6c1daeSBarry Smith PetscErrorCode PetscMod(PetscReal x,PetscReal y,PetscReal *result) 665c6c1daeSBarry Smith { 675c6c1daeSBarry Smith int i; 685c6c1daeSBarry Smith 695c6c1daeSBarry Smith PetscFunctionBegin; 705c6c1daeSBarry Smith if (y == 1) { 715c6c1daeSBarry Smith *result = 0.0; 725c6c1daeSBarry Smith PetscFunctionReturn(0); 735c6c1daeSBarry Smith } 745c6c1daeSBarry Smith i = ((int)x) / ((int)y); 755c6c1daeSBarry Smith x = x - i * y; 765c6c1daeSBarry Smith while (x > y) x -= y; 775c6c1daeSBarry Smith *result = x; 785c6c1daeSBarry Smith PetscFunctionReturn(0); 795c6c1daeSBarry Smith } 805c6c1daeSBarry Smith 815c6c1daeSBarry Smith PetscErrorCode PetscCopysign(PetscReal a,PetscReal b,PetscReal *result) 825c6c1daeSBarry Smith { 835c6c1daeSBarry Smith PetscFunctionBegin; 845c6c1daeSBarry Smith if (b >= 0) *result = a; 855c6c1daeSBarry Smith else *result = -a; 865c6c1daeSBarry Smith PetscFunctionReturn(0); 875c6c1daeSBarry Smith } 885c6c1daeSBarry Smith 895c6c1daeSBarry Smith /* 905c6c1daeSBarry Smith Given a value "in" and a "base", return a nice value. 915c6c1daeSBarry Smith based on "sign", extend up (+1) or down (-1) 925c6c1daeSBarry Smith */ 935c6c1daeSBarry Smith PetscErrorCode PetscAGetNice(PetscReal in,PetscReal base,int sign,PetscReal *result) 945c6c1daeSBarry Smith { 955c6c1daeSBarry Smith PetscReal etmp,s,s2,m; 965c6c1daeSBarry Smith 975c6c1daeSBarry Smith PetscFunctionBegin; 98*9566063dSJacob Faibussowitsch PetscCall(PetscCopysign (0.5,(double)sign,&s)); 995c6c1daeSBarry Smith etmp = in / base + 0.5 + s; 100*9566063dSJacob Faibussowitsch PetscCall(PetscCopysign (0.5,etmp,&s)); 101*9566063dSJacob Faibussowitsch PetscCall(PetscCopysign (EPS * etmp,(double)sign,&s2)); 1025c6c1daeSBarry Smith etmp = etmp - 0.5 + s - s2; 103*9566063dSJacob Faibussowitsch PetscCall(PetscMod(etmp,1.0,&m)); 1045c6c1daeSBarry Smith etmp = base * (etmp - m); 1055c6c1daeSBarry Smith *result = etmp; 1065c6c1daeSBarry Smith PetscFunctionReturn(0); 1075c6c1daeSBarry Smith } 1085c6c1daeSBarry Smith 1095c6c1daeSBarry Smith PetscErrorCode PetscAGetBase(PetscReal vmin,PetscReal vmax,int num,PetscReal *Base,int *power) 1105c6c1daeSBarry Smith { 1115c6c1daeSBarry Smith PetscReal base,ftemp,e10; 1125c6c1daeSBarry Smith static PetscReal base_try[5] = {10.0,5.0,2.0,1.0,0.5}; 1135c6c1daeSBarry Smith int i; 1145c6c1daeSBarry Smith 1155c6c1daeSBarry Smith PetscFunctionBegin; 1165c6c1daeSBarry Smith /* labels of the form n * BASE */ 1175c6c1daeSBarry Smith /* get an approximate value for BASE */ 1185c6c1daeSBarry Smith base = (vmax - vmin) / (double)(num + 1); 1195c6c1daeSBarry Smith 1205c6c1daeSBarry Smith /* make it of form m x 10^power, m in [1.0, 10) */ 1215c6c1daeSBarry Smith if (base <= 0.0) { 1225c6c1daeSBarry Smith base = PetscAbsReal(vmin); 1235c6c1daeSBarry Smith if (base < 1.0) base = 1.0; 1245c6c1daeSBarry Smith } 12577b4d14cSPeter Brune ftemp = PetscLog10Real((1.0 + EPS) * base); 1265c6c1daeSBarry Smith if (ftemp < 0.0) ftemp -= 1.0; 1275c6c1daeSBarry Smith *power = (int)ftemp; 128*9566063dSJacob Faibussowitsch PetscCall(PetscExp10((double)-*power,&e10)); 1295c6c1daeSBarry Smith base = base * e10; 1305c6c1daeSBarry Smith if (base < 1.0) base = 1.0; 1315c6c1daeSBarry Smith /* now reduce it to one of 1, 2, or 5 */ 1325c6c1daeSBarry Smith for (i=1; i<5; i++) { 1335c6c1daeSBarry Smith if (base >= base_try[i]) { 134*9566063dSJacob Faibussowitsch PetscCall(PetscExp10((double)*power,&e10)); 1355c6c1daeSBarry Smith base = base_try[i-1] * e10; 1365c6c1daeSBarry Smith if (i == 1) *power = *power + 1; 1375c6c1daeSBarry Smith break; 1385c6c1daeSBarry Smith } 1395c6c1daeSBarry Smith } 1405c6c1daeSBarry Smith *Base = base; 1415c6c1daeSBarry Smith PetscFunctionReturn(0); 1425c6c1daeSBarry Smith } 143