Update all #include directives to the currently preferred style.
[pspp-builds.git] / src / output / charts / plot-hist.c
index ab3b988daaf8c9b9bb8e98eeaace831f0ac09e68..5abbbb6c8c4a32ac695bb4f50478af79b538b0a7 100644 (file)
@@ -1,5 +1,5 @@
 /* PSPP - a program for statistical analysis.
-   Copyright  (C) 2004, 2009 Free Software Foundation, Inc.
+   Copyright (C) 2004, 2009, 2011 Free Software Foundation, Inc.
 
    This program is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
 #include <config.h>
 
 #include <stdio.h>
-#include <plot.h>
 #include <math.h>
-#include <gsl/gsl_histogram.h>
 #include <gsl/gsl_randist.h>
 #include <assert.h>
 
-#include <output/charts/plot-hist.h>
-#include <output/charts/plot-chart.h>
-#include <output/chart-provider.h>
-
-#include <data/variable.h>
-#include <libpspp/hash.h>
-#include <output/chart.h>
-#include <math/histogram.h>
-#include <math/moments.h>
+#include "libpspp/cast.h"
+#include "math/histogram.h"
+#include "math/moments.h"
+#include "output/chart-item-provider.h"
+#include "output/charts/plot-hist.h"
 
 #include "gettext.h"
 #define _(msgid) gettext (msgid)
 
-static const struct chart_class histogram_chart_class;
-
-/* Write the legend of the chart */
-static void
-histogram_write_legend (plPlotter *lp, const struct chart_geometry *geom,
-                        double n, double mean, double stddev)
-{
-  double y = geom->data_bottom;
-  pl_savestate_r (lp);
-
-  if (n != SYSMIS)
-    {
-      char *buf = xasprintf ("N = %.2f", n);
-      pl_move_r (lp, geom->legend_left, y);
-      pl_alabel_r (lp, 0, 'b', buf);
-      y += geom->font_size * 1.5;
-      free (buf);
-    }
-
-  if (mean != SYSMIS)
-    {
-      char *buf = xasprintf ("Mean = %.1f", mean);
-      pl_fmove_r (lp,geom->legend_left, y);
-      pl_alabel_r (lp, 0, 'b', buf);
-      y += geom->font_size * 1.5;
-      free (buf);
-    }
-
-  if (stddev != SYSMIS)
-    {
-      char *buf = xasprintf ("Std. Dev = %.2f", stddev);
-      pl_fmove_r (lp, geom->legend_left, y);
-      pl_alabel_r (lp, 0, 'b', buf);
-      free (buf);
-    }
-
-  pl_restorestate_r (lp);
-}
-
-static void
-hist_draw_bar (plPlotter *lp, const struct chart_geometry *geom,
-               const gsl_histogram *h, int bar)
-{
-  double upper;
-  double lower;
-  double height;
-
-  const size_t bins = gsl_histogram_bins (h);
-  const double x_pos = (geom->data_right - geom->data_left) * bar / (double) bins ;
-  const double width = (geom->data_right - geom->data_left) / (double) bins ;
-
-  assert ( 0 == gsl_histogram_get_range (h, bar, &lower, &upper));
-
-  assert ( upper >= lower);
-
-  height = gsl_histogram_get (h, bar) *
-    (geom->data_top - geom->data_bottom) / gsl_histogram_max_val (h);
-
-  pl_savestate_r (lp);
-  pl_move_r (lp,geom->data_left, geom->data_bottom);
-  pl_fillcolorname_r (lp, geom->fill_colour);
-  pl_filltype_r (lp,1);
-
-
-  pl_fboxrel_r (lp,
-                x_pos, 0,
-                x_pos + width, height);
-
-  pl_restorestate_r (lp);
-
-  draw_tick (lp, geom, TICK_ABSCISSA,
-             x_pos + width / 2.0, "%g", (upper + lower) / 2.0);
-}
-
-struct histogram_chart
-  {
-    struct chart chart;
-    gsl_histogram *gsl_hist;
-    char *label;
-    double n;
-    double mean;
-    double stddev;
-    bool show_normal;
-  };
-
 /* Plots a histogram of the data in HIST with the given LABEL.
    Labels the histogram with each of N, MEAN, and STDDEV that is
    not SYSMIS.  If all three are not SYSMIS and SHOW_NORMAL is
    true, also draws a normal curve on the histogram. */
-struct chart *
-histogram_chart_create (const struct histogram *hist, const char *label,
+struct chart_item *
+histogram_chart_create (const gsl_histogram *hist, const char *label,
                         double n, double mean, double stddev,
                         bool show_normal)
 {
   struct histogram_chart *h;
 
   h = xmalloc (sizeof *h);
-  chart_init (&h->chart, &histogram_chart_class);
-  h->gsl_hist = hist->gsl_hist ? gsl_histogram_clone (hist->gsl_hist) : NULL;
-  h->label = xstrdup (label);
+  chart_item_init (&h->chart_item, &histogram_chart_class, label);
+  h->gsl_hist = hist != NULL ? gsl_histogram_clone (hist) : NULL;
   h->n = n;
   h->mean = mean;
   h->stddev = stddev;
   h->show_normal = show_normal;
-  return &h->chart;
+  return &h->chart_item;
 }
 
 static void
-histogram_chart_draw (const struct chart *chart, plPlotter *lp)
-{
-  struct histogram_chart *h = (struct histogram_chart *) chart;
-  struct chart_geometry geom;
-  int i;
-  int bins;
-
-  chart_geometry_init (lp, &geom);
-
-  chart_write_title (lp, &geom, _("HISTOGRAM"));
-
-  chart_write_ylabel (lp, &geom, _("Frequency"));
-  chart_write_xlabel (lp, &geom, h->label);
-
-  if (h->gsl_hist == NULL)
-    {
-      /* Probably all values are SYSMIS. */
-      return;
-    }
-
-  bins = gsl_histogram_bins (h->gsl_hist);
-
-  chart_write_yscale (lp, &geom, 0, gsl_histogram_max_val (h->gsl_hist), 5);
-
-  for (i = 0; i < bins; i++)
-    hist_draw_bar (lp, &geom, h->gsl_hist, i);
-
-  histogram_write_legend (lp, &geom, h->n, h->mean, h->stddev);
-
-  if (h->show_normal
-      && h->n != SYSMIS && h->mean != SYSMIS && h->stddev != SYSMIS)
-    {
-      /* Draw the normal curve */
-      double d;
-      double x_min, x_max, not_used;
-      double abscissa_scale;
-      double ordinate_scale;
-      double range;
-
-      gsl_histogram_get_range (h->gsl_hist, 0, &x_min, &not_used);
-      range = not_used - x_min;
-      gsl_histogram_get_range (h->gsl_hist, bins - 1, &not_used, &x_max);
-
-      abscissa_scale = (geom.data_right - geom.data_left) / (x_max - x_min);
-      ordinate_scale = (geom.data_top - geom.data_bottom) /
-       gsl_histogram_max_val (h->gsl_hist);
-
-      pl_move_r (lp, geom.data_left, geom.data_bottom);
-      for (d = geom.data_left;
-          d <= geom.data_right;
-          d += (geom.data_right - geom.data_left) / 100.0)
-       {
-         const double x = (d - geom.data_left) / abscissa_scale + x_min;
-         const double y = h->n * range *
-           gsl_ran_gaussian_pdf (x - h->mean, h->stddev);
-
-         pl_fcont_r (lp,  d,  geom.data_bottom  + y * ordinate_scale);
-
-       }
-      pl_endpath_r (lp);
-    }
-
-  chart_geometry_free (lp);
-}
-
-
-static void
-histogram_chart_destroy (struct chart *chart)
+histogram_chart_destroy (struct chart_item *chart_item)
 {
-  struct histogram_chart *h = (struct histogram_chart *) chart;
+  struct histogram_chart *h = UP_CAST (chart_item, struct histogram_chart,
+                                       chart_item);
   if (h->gsl_hist != NULL)
     gsl_histogram_free (h->gsl_hist);
-  free (h->label);
   free (h);
 }
 
-static const struct chart_class histogram_chart_class =
+const struct chart_item_class histogram_chart_class =
   {
-    histogram_chart_draw,
     histogram_chart_destroy
   };