xref: /petsc/src/sys/classes/draw/utils/axis.c (revision 999739cf12618b36ac48ee4250964a76f927d80d)
15c6c1daeSBarry Smith 
2*999739cfSJacob 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   PetscErrorCode ierr;
135c6c1daeSBarry Smith 
145c6c1daeSBarry Smith   PetscFunctionBegin;
155c6c1daeSBarry Smith   /* Find the string */
16ba129bcfSBarry Smith   if (PetscAbsReal(val)/sep <  1.e-4) {
175c6c1daeSBarry Smith     buf[0] = '0'; buf[1] = 0;
1836c9bc0dSBarry Smith   } else {
194d25b1e8SKarl Rupp     sprintf(buf,"%0.1e",(double)val);
207b8d257dSBarry Smith     ierr = PetscStripZerosPlus(buf);CHKERRQ(ierr);
2136c9bc0dSBarry Smith     ierr = PetscStripe0(buf);CHKERRQ(ierr);
225c6c1daeSBarry Smith     ierr = PetscStripInitialZero(buf);CHKERRQ(ierr);
235c6c1daeSBarry Smith     ierr = PetscStripAllZeros(buf);CHKERRQ(ierr);
245c6c1daeSBarry Smith     ierr = PetscStripTrailingZeros(buf);CHKERRQ(ierr);
255c6c1daeSBarry Smith   }
265c6c1daeSBarry Smith   *p = buf;
275c6c1daeSBarry Smith   PetscFunctionReturn(0);
285c6c1daeSBarry Smith }
295c6c1daeSBarry Smith 
305c6c1daeSBarry Smith /* Finds "nice" locations for the ticks */
315c6c1daeSBarry Smith PetscErrorCode PetscADefTicks(PetscReal low,PetscReal high,int num,int *ntick,PetscReal *tickloc,int maxtick)
325c6c1daeSBarry Smith {
335c6c1daeSBarry Smith   PetscErrorCode ierr;
345c6c1daeSBarry Smith   int            i,power;
35e5ab1681SLisandro Dalcin   PetscReal      x = 0.0,base=0.0,eps;
365c6c1daeSBarry Smith 
375c6c1daeSBarry Smith   PetscFunctionBegin;
385c6c1daeSBarry Smith   ierr = PetscAGetBase(low,high,num,&base,&power);CHKERRQ(ierr);
395c6c1daeSBarry Smith   ierr = PetscAGetNice(low,base,-1,&x);CHKERRQ(ierr);
405c6c1daeSBarry Smith 
415c6c1daeSBarry Smith   /* Values are of the form j * base */
425c6c1daeSBarry Smith   /* Find the starting value */
435c6c1daeSBarry Smith   if (x < low) x += base;
445c6c1daeSBarry Smith 
45e5ab1681SLisandro Dalcin   i = 0; eps = base/10;
46e5ab1681SLisandro Dalcin   while (i < maxtick && x <= high+eps) {
47e5ab1681SLisandro Dalcin     tickloc[i++] = x; x += base;
485c6c1daeSBarry Smith   }
495c6c1daeSBarry Smith   *ntick = i;
50e5ab1681SLisandro Dalcin   tickloc[i-1] = PetscMin(tickloc[i-1],high);
515c6c1daeSBarry Smith 
525c6c1daeSBarry Smith   if (i < 2 && num < 10) {
535c6c1daeSBarry Smith     ierr = PetscADefTicks(low,high,num+1,ntick,tickloc,maxtick);CHKERRQ(ierr);
545c6c1daeSBarry Smith   }
555c6c1daeSBarry Smith   PetscFunctionReturn(0);
565c6c1daeSBarry Smith }
575c6c1daeSBarry Smith 
585c6c1daeSBarry Smith #define EPS 1.e-6
595c6c1daeSBarry Smith 
605c6c1daeSBarry Smith PetscErrorCode PetscExp10(PetscReal d,PetscReal *result)
615c6c1daeSBarry Smith {
625c6c1daeSBarry Smith   PetscFunctionBegin;
6385ec1a3cSBarry Smith   *result = PetscPowReal((PetscReal)10.0,d);
645c6c1daeSBarry Smith   PetscFunctionReturn(0);
655c6c1daeSBarry Smith }
665c6c1daeSBarry Smith 
675c6c1daeSBarry Smith PetscErrorCode PetscMod(PetscReal x,PetscReal y,PetscReal *result)
685c6c1daeSBarry Smith {
695c6c1daeSBarry Smith   int i;
705c6c1daeSBarry Smith 
715c6c1daeSBarry Smith   PetscFunctionBegin;
725c6c1daeSBarry Smith   if (y == 1) {
735c6c1daeSBarry Smith     *result = 0.0;
745c6c1daeSBarry Smith     PetscFunctionReturn(0);
755c6c1daeSBarry Smith   }
765c6c1daeSBarry Smith   i = ((int)x) / ((int)y);
775c6c1daeSBarry Smith   x = x - i * y;
785c6c1daeSBarry Smith   while (x > y) x -= y;
795c6c1daeSBarry Smith   *result = x;
805c6c1daeSBarry Smith   PetscFunctionReturn(0);
815c6c1daeSBarry Smith }
825c6c1daeSBarry Smith 
835c6c1daeSBarry Smith PetscErrorCode PetscCopysign(PetscReal a,PetscReal b,PetscReal *result)
845c6c1daeSBarry Smith {
855c6c1daeSBarry Smith   PetscFunctionBegin;
865c6c1daeSBarry Smith   if (b >= 0) *result = a;
875c6c1daeSBarry Smith   else        *result = -a;
885c6c1daeSBarry Smith   PetscFunctionReturn(0);
895c6c1daeSBarry Smith }
905c6c1daeSBarry Smith 
915c6c1daeSBarry Smith /*
925c6c1daeSBarry Smith     Given a value "in" and a "base", return a nice value.
935c6c1daeSBarry Smith     based on "sign", extend up (+1) or down (-1)
945c6c1daeSBarry Smith  */
955c6c1daeSBarry Smith PetscErrorCode PetscAGetNice(PetscReal in,PetscReal base,int sign,PetscReal *result)
965c6c1daeSBarry Smith {
975c6c1daeSBarry Smith   PetscReal      etmp,s,s2,m;
985c6c1daeSBarry Smith   PetscErrorCode ierr;
995c6c1daeSBarry Smith 
1005c6c1daeSBarry Smith   PetscFunctionBegin;
1015c6c1daeSBarry Smith   ierr    = PetscCopysign (0.5,(double)sign,&s);CHKERRQ(ierr);
1025c6c1daeSBarry Smith   etmp    = in / base + 0.5 + s;
1035c6c1daeSBarry Smith   ierr    = PetscCopysign (0.5,etmp,&s);CHKERRQ(ierr);
1045c6c1daeSBarry Smith   ierr    = PetscCopysign (EPS * etmp,(double)sign,&s2);CHKERRQ(ierr);
1055c6c1daeSBarry Smith   etmp    = etmp - 0.5 + s - s2;
1065c6c1daeSBarry Smith   ierr    = PetscMod(etmp,1.0,&m);CHKERRQ(ierr);
1075c6c1daeSBarry Smith   etmp    = base * (etmp -  m);
1085c6c1daeSBarry Smith   *result = etmp;
1095c6c1daeSBarry Smith   PetscFunctionReturn(0);
1105c6c1daeSBarry Smith }
1115c6c1daeSBarry Smith 
1125c6c1daeSBarry Smith PetscErrorCode PetscAGetBase(PetscReal vmin,PetscReal vmax,int num,PetscReal *Base,int *power)
1135c6c1daeSBarry Smith {
1145c6c1daeSBarry Smith   PetscReal        base,ftemp,e10;
1155c6c1daeSBarry Smith   static PetscReal base_try[5] = {10.0,5.0,2.0,1.0,0.5};
1165c6c1daeSBarry Smith   PetscErrorCode   ierr;
1175c6c1daeSBarry Smith   int              i;
1185c6c1daeSBarry Smith 
1195c6c1daeSBarry Smith   PetscFunctionBegin;
1205c6c1daeSBarry Smith   /* labels of the form n * BASE */
1215c6c1daeSBarry Smith   /* get an approximate value for BASE */
1225c6c1daeSBarry Smith   base = (vmax - vmin) / (double)(num + 1);
1235c6c1daeSBarry Smith 
1245c6c1daeSBarry Smith   /* make it of form   m x 10^power,  m in [1.0, 10) */
1255c6c1daeSBarry Smith   if (base <= 0.0) {
1265c6c1daeSBarry Smith     base = PetscAbsReal(vmin);
1275c6c1daeSBarry Smith     if (base < 1.0) base = 1.0;
1285c6c1daeSBarry Smith   }
12977b4d14cSPeter Brune   ftemp = PetscLog10Real((1.0 + EPS) * base);
1305c6c1daeSBarry Smith   if (ftemp < 0.0) ftemp -= 1.0;
1315c6c1daeSBarry Smith   *power = (int)ftemp;
1325c6c1daeSBarry Smith   ierr   = PetscExp10((double)-*power,&e10);CHKERRQ(ierr);
1335c6c1daeSBarry Smith   base   = base * e10;
1345c6c1daeSBarry Smith   if (base < 1.0) base = 1.0;
1355c6c1daeSBarry Smith   /* now reduce it to one of 1, 2, or 5 */
1365c6c1daeSBarry Smith   for (i=1; i<5; i++) {
1375c6c1daeSBarry Smith     if (base >= base_try[i]) {
1385c6c1daeSBarry Smith       ierr = PetscExp10((double)*power,&e10);CHKERRQ(ierr);
1395c6c1daeSBarry Smith       base = base_try[i-1] * e10;
1405c6c1daeSBarry Smith       if (i == 1) *power = *power + 1;
1415c6c1daeSBarry Smith       break;
1425c6c1daeSBarry Smith     }
1435c6c1daeSBarry Smith   }
1445c6c1daeSBarry Smith   *Base = base;
1455c6c1daeSBarry Smith   PetscFunctionReturn(0);
1465c6c1daeSBarry Smith }
147