1999739cfSJacob Faibussowitsch #include <petsc/private/drawimpl.h> /*I "petscdraw.h" I*/
25c6c1daeSBarry Smith
35c6c1daeSBarry Smith /*
45c6c1daeSBarry Smith val is the label value. sep is the separation to the next (or previous)
55c6c1daeSBarry Smith label; this is useful in determining how many significant figures to
65c6c1daeSBarry Smith keep.
75c6c1daeSBarry Smith */
PetscADefLabel(PetscReal val,PetscReal sep,char ** p)8d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscADefLabel(PetscReal val, PetscReal sep, char **p)
9d71ae5a4SJacob Faibussowitsch {
105c6c1daeSBarry Smith static char buf[40];
115c6c1daeSBarry Smith
125c6c1daeSBarry Smith PetscFunctionBegin;
135c6c1daeSBarry Smith /* Find the string */
14ba129bcfSBarry Smith if (PetscAbsReal(val) / sep < 1.e-4) {
159371c9d4SSatish Balay buf[0] = '0';
169371c9d4SSatish Balay buf[1] = 0;
1736c9bc0dSBarry Smith } else {
18a364092eSJacob Faibussowitsch PetscCall(PetscSNPrintf(buf, PETSC_STATIC_ARRAY_LENGTH(buf), "%0.1e", (double)val));
199566063dSJacob Faibussowitsch PetscCall(PetscStripZerosPlus(buf));
209566063dSJacob Faibussowitsch PetscCall(PetscStripe0(buf));
219566063dSJacob Faibussowitsch PetscCall(PetscStripInitialZero(buf));
229566063dSJacob Faibussowitsch PetscCall(PetscStripAllZeros(buf));
239566063dSJacob Faibussowitsch PetscCall(PetscStripTrailingZeros(buf));
245c6c1daeSBarry Smith }
255c6c1daeSBarry Smith *p = buf;
263ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
275c6c1daeSBarry Smith }
285c6c1daeSBarry Smith
295c6c1daeSBarry Smith /* Finds "nice" locations for the ticks */
PetscADefTicks(PetscReal low,PetscReal high,int num,int * ntick,PetscReal * tickloc,int maxtick)30d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscADefTicks(PetscReal low, PetscReal high, int num, int *ntick, PetscReal *tickloc, int maxtick)
31d71ae5a4SJacob Faibussowitsch {
325c6c1daeSBarry Smith int i, power;
33e5ab1681SLisandro Dalcin PetscReal x = 0.0, base = 0.0, eps;
345c6c1daeSBarry Smith
355c6c1daeSBarry Smith PetscFunctionBegin;
369566063dSJacob Faibussowitsch PetscCall(PetscAGetBase(low, high, num, &base, &power));
379566063dSJacob 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
439371c9d4SSatish Balay i = 0;
449371c9d4SSatish Balay eps = base / 10;
45e5ab1681SLisandro Dalcin while (i < maxtick && x <= high + eps) {
469371c9d4SSatish Balay tickloc[i++] = x;
479371c9d4SSatish Balay x += base;
485c6c1daeSBarry Smith }
495c6c1daeSBarry Smith *ntick = i;
50e5ab1681SLisandro Dalcin tickloc[i - 1] = PetscMin(tickloc[i - 1], high);
515c6c1daeSBarry Smith
5248a46eb9SPierre Jolivet if (i < 2 && num < 10) PetscCall(PetscADefTicks(low, high, num + 1, ntick, tickloc, maxtick));
533ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
545c6c1daeSBarry Smith }
555c6c1daeSBarry Smith
565c6c1daeSBarry Smith #define EPS 1.e-6
575c6c1daeSBarry Smith
PetscExp10(PetscReal d,PetscReal * result)5866976f2fSJacob Faibussowitsch static PetscErrorCode PetscExp10(PetscReal d, PetscReal *result)
59d71ae5a4SJacob Faibussowitsch {
605c6c1daeSBarry Smith PetscFunctionBegin;
6185ec1a3cSBarry Smith *result = PetscPowReal((PetscReal)10.0, d);
623ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
635c6c1daeSBarry Smith }
645c6c1daeSBarry Smith
PetscMod(PetscReal x,PetscReal y,PetscReal * result)6566976f2fSJacob Faibussowitsch static PetscErrorCode PetscMod(PetscReal x, PetscReal y, PetscReal *result)
66d71ae5a4SJacob Faibussowitsch {
675c6c1daeSBarry Smith int i;
685c6c1daeSBarry Smith
695c6c1daeSBarry Smith PetscFunctionBegin;
705c6c1daeSBarry Smith if (y == 1) {
715c6c1daeSBarry Smith *result = 0.0;
723ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
735c6c1daeSBarry Smith }
745c6c1daeSBarry Smith i = ((int)x) / ((int)y);
75*6497c311SBarry Smith x = x - ((PetscReal)i) * y;
765c6c1daeSBarry Smith while (x > y) x -= y;
775c6c1daeSBarry Smith *result = x;
783ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
795c6c1daeSBarry Smith }
805c6c1daeSBarry Smith
PetscCopysign(PetscReal a,PetscReal b,PetscReal * result)8166976f2fSJacob Faibussowitsch static PetscErrorCode PetscCopysign(PetscReal a, PetscReal b, PetscReal *result)
82d71ae5a4SJacob Faibussowitsch {
835c6c1daeSBarry Smith PetscFunctionBegin;
845c6c1daeSBarry Smith if (b >= 0) *result = a;
855c6c1daeSBarry Smith else *result = -a;
863ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
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 */
PetscAGetNice(PetscReal in,PetscReal base,int sign,PetscReal * result)93d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscAGetNice(PetscReal in, PetscReal base, int sign, PetscReal *result)
94d71ae5a4SJacob Faibussowitsch {
955c6c1daeSBarry Smith PetscReal etmp, s, s2, m;
965c6c1daeSBarry Smith
975c6c1daeSBarry Smith PetscFunctionBegin;
989566063dSJacob Faibussowitsch PetscCall(PetscCopysign(0.5, (double)sign, &s));
995c6c1daeSBarry Smith etmp = in / base + 0.5 + s;
1009566063dSJacob Faibussowitsch PetscCall(PetscCopysign(0.5, etmp, &s));
1019566063dSJacob Faibussowitsch PetscCall(PetscCopysign(EPS * etmp, (double)sign, &s2));
1025c6c1daeSBarry Smith etmp = etmp - 0.5 + s - s2;
1039566063dSJacob Faibussowitsch PetscCall(PetscMod(etmp, 1.0, &m));
1045c6c1daeSBarry Smith etmp = base * (etmp - m);
1055c6c1daeSBarry Smith *result = etmp;
1063ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
1075c6c1daeSBarry Smith }
1085c6c1daeSBarry Smith
PetscAGetBase(PetscReal vmin,PetscReal vmax,int num,PetscReal * Base,int * power)109d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscAGetBase(PetscReal vmin, PetscReal vmax, int num, PetscReal *Base, int *power)
110d71ae5a4SJacob Faibussowitsch {
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;
1289566063dSJacob 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]) {
1349566063dSJacob 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;
1413ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
1425c6c1daeSBarry Smith }
143