summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGravatar Linus Torvalds <torvalds@linux-foundation.org>2011-09-07 13:51:35 -0700
committerGravatar Linus Torvalds <torvalds@linux-foundation.org>2011-09-07 13:51:35 -0700
commit96f5bea1ac89866aa58fe9fd39e625947bfa910a (patch)
tree662b8a1b9a49e2379719911cfa48cee26f53362d
parentd1ce43087834a881885371e0c7ece9fe55e46e64 (diff)
downloadsubsurface-96f5bea1ac89866aa58fe9fd39e625947bfa910a.tar.gz
Use a recursive (instead of iterative) minmax depth finder
This is a bit more natural, and makes it much easier to do scale independence. In particular, I want to make it possible to grow and shrink the graph, and this should make it particularly simple to react by giving more or fewer minmax points. Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r--profile.c70
1 files changed, 46 insertions, 24 deletions
diff --git a/profile.c b/profile.c
index 9c2f4ed12..7c72d0b8c 100644
--- a/profile.c
+++ b/profile.c
@@ -67,7 +67,7 @@ static void plot_text(cairo_t *cr, text_render_options_t *tro,
* We exit early if we hit "enough" of a depth reversal,
* which is roughly 10 feet.
*/
-static struct sample *next_minmax(struct dive *dive, struct sample *sample, struct sample *end, int minmax)
+static struct sample *next_minmax(struct sample *sample, struct sample *end, int minmax)
{
const int enough = 3000;
struct sample *result;
@@ -116,6 +116,50 @@ static struct sample *next_minmax(struct dive *dive, struct sample *sample, stru
/* Scale to 0,0 -> maxx,maxy */
#define SCALE(x,y) (x)*maxx/scalex,(y)*maxy/scaley
+void plot_text_samples(struct dive *dive, cairo_t *cr,
+ double maxx, double maxy,
+ double scalex, double scaley,
+ struct sample *a, struct sample *b)
+{
+ struct sample *max, *min;
+
+ if (b < a)
+ return;
+ if (b->time.seconds - a->time.seconds < 3*60)
+ return;
+
+ max = next_minmax(a, b, 1);
+ if (max) {
+ text_render_options_t tro = {1.0, 0.2, 0.2, CENTER};
+ int sec = max->time.seconds;
+ depth_t depth = max->depth;
+ const char *fmt;
+ double d;
+
+ min = next_minmax(max, b, 0);
+ plot_text_samples(dive, cr, maxx, maxy, scalex, scaley, a, max);
+ if (min) {
+ plot_text_samples(dive, cr, maxx, maxy, scalex, scaley, max, min);
+ plot_text_samples(dive, cr, maxx, maxy, scalex, scaley, min, b);
+ } else
+ plot_text_samples(dive, cr, maxx, maxy, scalex, scaley, max, b);
+
+ switch (output_units.length) {
+ case METERS:
+ d = depth.mm / 1000.0;
+ fmt = "%.1f";
+ break;
+ case FEET:
+ d = to_feet(depth);
+ fmt = "%.0f";
+ break;
+ }
+
+ plot_text(cr, &tro, SCALE(sec, depth.mm), fmt, d);
+ return;
+ }
+}
+
static void plot_depth_text(struct dive *dive, cairo_t *cr,
double maxx, double maxy)
{
@@ -141,29 +185,7 @@ static void plot_depth_text(struct dive *dive, cairo_t *cr,
sample = dive->sample;
end = dive->sample + dive->samples - 1;
- while ((sample = next_minmax(dive, sample, end, 1)) != NULL) {
- text_render_options_t tro = {1.0, 0.2, 0.2, CENTER};
- int sec = sample->time.seconds;
- depth_t depth = sample->depth;
- const char *fmt;
- double d;
-
- switch (output_units.length) {
- case METERS:
- d = depth.mm / 1000.0;
- fmt = "%.1f";
- break;
- case FEET:
- d = to_feet(depth);
- fmt = "%.0f";
- break;
- }
-
- plot_text(cr, &tro, SCALE(sec, depth.mm), fmt, d);
- sample = next_minmax(dive, sample, end, 0);
- if (!sample)
- break;
- }
+ plot_text_samples(dive, cr, maxx, maxy, scalex, scaley, sample, end);
}
static void plot_depth_profile(struct dive *dive, cairo_t *cr,