Merge commit 'HEAD'; commit 'savannah/master'
[pspp-builds.git] / src / ui / gui / psppire.c
index b0de6cfa1bdfddc5a5cfc6b7d0fb4d995f5fb80c..bdd5ad5332f4062d4b0bf6927a42ea2b57223d30 100644 (file)
@@ -1,10 +1,9 @@
-/*
-   PSPPIRE --- A Graphical User Interface for PSPP
-   Copyright (C) 2004, 2005, 2006  Free Software Foundation
+/* PSPPIRE - a graphical user interface for PSPP.
+   Copyright (C) 2004, 2005, 2006, 2009  Free Software Foundation
 
-   This program is free software; you can redistribute it and/or modify
+   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
-   the Free Software Foundation; either version 2 of the License, or
+   the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.
 
    This program is distributed in the hope that it will be useful,
    GNU General Public License for more details.
 
    You should have received a copy of the GNU General Public License
-   along with this program; if not, write to the Free Software
-   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
-   02110-1301, USA. */
+   along with this program.  If not, see <http://www.gnu.org/licenses/>. */
 
 #include <config.h>
 
+#include <locale.h>
 #include <assert.h>
 #include <libintl.h>
+#include <gsl/gsl_errno.h>
 
+#include <argp.h>
+#include <ui/command-line.h>
+#include "relocatable.h"
 
-#include "data-editor.h"
-#include <libpspp/version.h>
-#include <libpspp/copyleft.h>
+#include "psppire-data-window.h"
+#include "psppire.h"
+
+#include <libpspp/getl.h>
+#include <unistd.h>
+#include <data/casereader.h>
+#include <data/datasheet.h>
 #include <data/file-handle-def.h>
-#include <data/format.h>
 #include <data/settings.h>
 #include <data/file-name.h>
 #include <data/procedure.h>
 #include <libpspp/getl.h>
 #include <language/lexer/lexer.h>
+#include <libpspp/version.h>
+#include <output/output.h>
+#include <output/journal.h>
+#include <language/syntax-string-source.h>
 
-#include <getopt.h>
-#include <gtk/gtk.h>
 #include <gtk/gtk.h>
 #include <glade/glade.h>
 #include "psppire-dict.h"
 #include "psppire-var-store.h"
 #include "psppire-data-store.h"
 #include "helper.h"
-#include "data-sheet.h"
-#include "var-sheet.h"
 #include "message-dialog.h"
-#include "flexifile-factory.h"
+#include <ui/syntax-gen.h>
 
-PsppireDataStore *the_data_store = 0;
 
+#include "psppire-output-window.h"
 
-static bool parse_command_line (int *argc, char ***argv,
-                               gchar **filename, GError **err);
+#include <data/sys-file-reader.h>
+#include <data/por-file-reader.h>
 
+#include <ui/source-init-opts.h>
 
+PsppireDataStore *the_data_store = 0;
 PsppireVarStore *the_var_store = 0;
 
-void create_icon_factory (void);
+static void create_icon_factory (void);
 
 struct source_stream *the_source_stream ;
 struct dataset * the_dataset = NULL;
 
+static GtkWidget *the_data_window;
 
-int
-main (int argc, char *argv[])
+static void
+replace_casereader (struct casereader *s)
 {
-  struct casefile_factory *factory;
-  PsppireDict *dictionary = 0;
+  psppire_data_store_set_reader (the_data_store, s);
+}
 
+#define _(msgid) gettext (msgid)
+#define N_(msgid) msgid
 
-  gchar *filename=0;
-  GError *err = 0;
-  gchar *vers;
 
-  gtk_init (&argc, &argv);
-  if ( (vers = gtk_check_version (GTK_MAJOR_VERSION,
-                                GTK_MINOR_VERSION,
-                                GTK_MICRO_VERSION)) )
-    {
-      g_critical (vers);
-    }
+const char * output_file_name (void);
+
 
+void
+initialize (struct command_line_processor *clp, int argc, char **argv)
+{
+  PsppireDict *dictionary = 0;
 
   /* gtk_init messes with the locale.
      So unset the bits we want to control ourselves */
@@ -88,149 +94,230 @@ main (int argc, char *argv[])
 
   bindtextdomain (PACKAGE, locale_dir);
 
-  textdomain (PACKAGE);
-
-  if ( ! parse_command_line (&argc, &argv, &filename, &err) )
-    {
-      g_clear_error (&err);
-      return 0;
-    }
 
   glade_init ();
 
-  fmt_init ();
-  settings_init ();
+  gsl_set_error_handler_off ();
+  fn_init ();
+  outp_init ();
+  settings_init (&viewer_width, &viewer_length);
   fh_init ();
-  factory = flexifile_factory_create ();
-  the_source_stream = create_source_stream (
+  the_source_stream =
+    create_source_stream (
                          fn_getenv_default ("STAT_INCLUDE_PATH", include_path)
                          );
 
-  the_dataset = create_dataset (factory);
+  the_dataset = create_dataset ();
+
 
   message_dialog_init (the_source_stream);
 
-  dictionary =
-    psppire_dict_new_from_dict (
-                               dataset_dict (the_dataset)
-                               );
+  dictionary = psppire_dict_new_from_dict (dataset_dict (the_dataset));
 
   bind_textdomain_codeset (PACKAGE, "UTF-8");
 
   /* Create the model for the var_sheet */
   the_var_store = psppire_var_store_new (dictionary);
 
-
   the_data_store = psppire_data_store_new (dictionary);
+  replace_casereader (NULL);
 
   create_icon_factory ();
 
-  new_data_window (NULL, NULL);
+  {
+    const char *filename = output_file_name ();
 
-  /* start the event loop */
-  gtk_main ();
+    struct string config_string;
 
-  destroy_source_stream (the_source_stream);
-  message_dialog_done ();
+    ds_init_empty (&config_string);
 
-  settings_done ();
+    ds_put_format (&config_string,
+                  "gui:ascii:screen:squeeze=on headers=off top-margin=0 "
+                  "bottom-margin=0 paginate=off length=auto width=auto "
+                  "emphasis=none "
+                  "output-file=\"%s\" append=yes", filename);
 
-  return 0;
+    outp_configure_driver_line (ds_ss (&config_string));
+
+    unlink (filename);
+
+    ds_destroy (&config_string);
+  }
+
+  journal_enable ();
+  textdomain (PACKAGE);
+
+  the_data_window = psppire_data_window_new ();
+
+  command_line_processor_replace_aux (clp, &post_init_argp, the_source_stream);
+  command_line_processor_replace_aux (clp, &non_option_argp, the_source_stream);
+
+  command_line_processor_parse (clp, argc, argv);
+
+  execute_syntax (create_syntax_string_source (""));
+
+  gtk_widget_show (the_data_window);
 }
 
 
-/* Parses the command line specified by ARGC and ARGV as received by
-   main ().  Returns true if normal execution should proceed,
-   false if the command-line indicates that PSPP should exit. */
-static bool
-parse_command_line (int *argc, char ***argv, gchar **filename, GError **err)
+void
+de_initialize (void)
 {
-  static struct option long_options[] =
-    {
-      {"help", no_argument, NULL, 'h'},
-      {"version", no_argument, NULL, 'V'},
-      {0, 0, 0, 0},
-    };
+  destroy_source_stream (the_source_stream);
+  message_dialog_done ();
+  settings_done ();
+  outp_done ();
+}
+
+
 
-  int c;
+struct icon_info
+{
+  const char *file_name;
+  const gchar *id;
+};
+
+
+static const struct icon_info icons[] =
+  {
+    {PKGDATADIR "/value-labels.png",    "pspp-value-labels"},
+    {PKGDATADIR "/weight-cases.png",    "pspp-weight-cases"},
+    {PKGDATADIR "/goto-variable.png",   "pspp-goto-variable"},
+    {PKGDATADIR "/insert-variable.png", "pspp-insert-variable"},
+    {PKGDATADIR "/insert-case.png",     "pspp-insert-case"},
+    {PKGDATADIR "/split-file.png",      "pspp-split-file"},
+    {PKGDATADIR "/select-cases.png",    "pspp-select-cases"},
+    {PKGDATADIR "/recent-dialogs.png",  "pspp-recent-dialogs"},
+    {PKGDATADIR "/nominal.png",         "var-nominal"},
+    {PKGDATADIR "/ordinal.png",         "var-ordinal"},
+    {PKGDATADIR "/scale.png",           "var-scale"},
+    {PKGDATADIR "/string.png",          "var-string"},
+    {PKGDATADIR "/date-scale.png",      "var-date-scale"}
+  };
+
+static void
+create_icon_factory (void)
+{
+  gint i;
+  GtkIconFactory *factory = gtk_icon_factory_new ();
 
-  for (;;)
+  for (i = 0 ; i < sizeof (icons) / sizeof(icons[0]); ++i)
     {
-      c = getopt_long (*argc, *argv, "hV", long_options, NULL);
-      if (c == -1)
-       break;
+      GError *err = NULL;
+      GdkPixbuf *pixbuf =
+       gdk_pixbuf_new_from_file (relocate (icons[i].file_name), &err);
 
-      switch (c)
+      if ( pixbuf )
        {
-       case 'h':
-         g_print ("Usage: psppire {|--help|--version}\n");
-          return false;
-       case 'V':
-         g_print (version);
-         g_print ("\n");
-         g_print (legal);
-         return false;
-       default:
-         return false;
+         GtkIconSet *icon_set = gtk_icon_set_new_from_pixbuf (pixbuf);
+         g_object_unref (pixbuf);
+         gtk_icon_factory_add ( factory, icons[i].id, icon_set);
+       }
+      else
+       {
+         g_warning ("Cannot create icon: %s", err->message);
+         g_clear_error (&err);
        }
     }
 
-  if ( optind < *argc)
-    {
-      *filename = (*argv)[optind];
-    }
+  {
+    /* Create our own "pspp-stock-reset" item, using the
+       GTK_STOCK_REFRESH icon set */
 
-  return true;
-}
+    GtkStockItem items[] = {
+      {"pspp-stock-reset", N_("_Reset"), 0, 0, PACKAGE},
+      {"pspp-stock-select", N_("_Select"), 0, 0, PACKAGE}
+    };
 
 
+    gtk_stock_add (items, 2);
+    gtk_icon_factory_add (factory, "pspp-stock-reset",
+                         gtk_icon_factory_lookup_default (GTK_STOCK_REFRESH)
+                         );
 
-void
-create_icon_factory (void)
-{
-  GtkIconFactory *factory = gtk_icon_factory_new ();
+    gtk_icon_factory_add (factory, "pspp-stock-select",
+                         gtk_icon_factory_lookup_default (GTK_STOCK_INDEX)
+                         );
+  }
+
+  gtk_icon_factory_add_default (factory);
+}
 
-  GtkIconSet *icon_set;
+\f
 
-  GdkPixbuf *pixbuf;
+static error_t
+parse_non_options (int key, char *arg, struct argp_state *state)
+{
+  struct source_stream *ss = state->input;
 
-  pixbuf = gdk_pixbuf_new_from_file (PKGDATADIR "/value-labels.png", 0);
-  icon_set = gtk_icon_set_new_from_pixbuf (pixbuf);
-  g_object_unref (pixbuf);
-  gtk_icon_factory_add ( factory, "pspp-value-labels", icon_set);
+  if ( NULL == ss )
+    return 0;
 
-  pixbuf = gdk_pixbuf_new_from_file (PKGDATADIR "/weight-cases.png", 0);
-  icon_set = gtk_icon_set_new_from_pixbuf (pixbuf);
-  g_object_unref (pixbuf);
-  gtk_icon_factory_add ( factory, "pspp-weight-cases", icon_set);
+  switch (key)
+    {
+    case ARGP_KEY_ARG:
+      {
+       struct string syntax;
+       FILE *fp = fopen (arg, "r");
+       if (NULL == fp)
+         {
+           const int errnum = errno;
+           fprintf (state->err_stream, _("Cannot open %s: %s.\n"),
+                    arg, strerror (errnum));
+           return 0;
+         }
+       if ( sfm_detect (fp))
+         {
+           ds_init_cstr (&syntax, "GET FILE=");
+           goto close;
+         }
+       rewind (fp);
+       if (pfm_detect (fp))
+         {
+           ds_init_cstr (&syntax, "IMPORT FILE=");
+           goto close;
+         }
+
+       fclose (fp);
+       msg (ME, _("%s is neither a system nor portable file"), arg);
+       break;
 
-  pixbuf = gdk_pixbuf_new_from_file (PKGDATADIR "/goto-variable.png", 0);
-  icon_set = gtk_icon_set_new_from_pixbuf (pixbuf);
-  g_object_unref (pixbuf);
-  gtk_icon_factory_add ( factory, "pspp-goto-variable", icon_set);
+      close:
+       fclose (fp);
 
-  pixbuf = gdk_pixbuf_new_from_file (PKGDATADIR "/insert-variable.png", 0);
-  icon_set = gtk_icon_set_new_from_pixbuf (pixbuf);
-  g_object_unref (pixbuf);
-  gtk_icon_factory_add ( factory, "pspp-insert-variable", icon_set);
+       syntax_gen_string (&syntax, ss_cstr (arg));
+       ds_put_cstr (&syntax, ".");
 
-  pixbuf = gdk_pixbuf_new_from_file (PKGDATADIR "/insert-case.png", 0);
-  icon_set = gtk_icon_set_new_from_pixbuf (pixbuf);
-  g_object_unref (pixbuf);
-  gtk_icon_factory_add ( factory, "pspp-insert-case", icon_set);
+       getl_append_source (ss,
+                           create_syntax_string_source (ds_cstr (&syntax)),
+                           GETL_BATCH,
+                           ERRMODE_CONTINUE);
 
-  pixbuf = gdk_pixbuf_new_from_file (PKGDATADIR "/split-file.png", 0);
-  icon_set = gtk_icon_set_new_from_pixbuf (pixbuf);
-  g_object_unref (pixbuf);
-  gtk_icon_factory_add ( factory, "pspp-split-file", icon_set);
+       ds_destroy (&syntax);
 
-  pixbuf = gdk_pixbuf_new_from_file (PKGDATADIR "/select-cases.png", 0);
-  icon_set = gtk_icon_set_new_from_pixbuf (pixbuf);
-  g_object_unref (pixbuf);
-  gtk_icon_factory_add ( factory, "pspp-select-cases", icon_set);
+       psppire_window_set_filename (the_data_window, arg);
 
-  gtk_icon_factory_add_default (factory);
+       break;
+      }
+    default:
+      return ARGP_ERR_UNKNOWN;
+    }
+  return 0;
 }
 
 
+const struct argp non_option_argp = {NULL, parse_non_options, 0, 0, 0, 0, 0};
 
+
+const char *
+output_file_name (void)
+{
+  const char *dir = default_output_path ();
+  static char *filename = NULL;
+
+  if ( NULL == filename )
+    filename = xasprintf ("%s%s", dir, OUTPUT_FILE_NAME);
+
+
+  return filename;
+}