Merge commit 'origin/stable'
[pspp-builds.git] / src / ui / gui / psppire.c
index aefb26e70ab524c8f10be4f93e71151efef78558..3555463f88e4d353c84bcd366b223955d4217604 100644 (file)
@@ -1,11 +1,9 @@
-/* 
-   PSPPIRE --- A Graphical User Interface for PSPP
-   Copyright (C) 2004, 2005, 2006  Free Software Foundation
-   Written by John Darrington
+/* 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/>. */
 
-/*
- * Initial main.c file generated by Glade. Edit as required.
- * Glade will not overwrite this file.
- */
+#include <config.h>
 
+#include <libpspp/i18n.h>
 #include <assert.h>
+#include <libintl.h>
+#include <gsl/gsl_errno.h>
+
+#include <xalloc.h>
+#include <argp.h>
+#include <ui/command-line.h>
+#include "relocatable.h"
+
+#include "psppire-data-window.h"
+#include "psppire.h"
+#include "widgets.h"
+
+#include <libpspp/getl.h>
+#include <unistd.h>
+#include <data/casereader.h>
+#include <data/datasheet.h>
+#include <data/file-handle-def.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 <libpspp/copyleft.h>
-#include <getopt.h>
-#include <gtk/gtk.h>
+#include <output/output.h>
+#include <output/journal.h>
+#include <language/syntax-string-source.h>
+
 #include <gtk/gtk.h>
-#include <glade/glade.h>
-#include "menu-actions.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 "psppire-case-array.h"
+#include "executor.h"
 #include "message-dialog.h"
+#include <ui/syntax-gen.h>
+
+#include "psppire-window-register.h"
+#include "psppire-output-window.h"
 
-GladeXML *xml;
+#include <data/sys-file-reader.h>
+#include <data/por-file-reader.h>
 
+#include <ui/source-init-opts.h>
 
-PsppireDict *the_dictionary = 0;
-PsppireCaseArray *the_cases = 0;
+GtkRecentManager *the_recent_mgr = 0;
+PsppireDataStore *the_data_store = 0;
+PsppireVarStore *the_var_store = 0;
 
+static void create_icon_factory (void);
+
+struct source_stream *the_source_stream ;
+struct dataset * the_dataset = NULL;
+
+static GtkWidget *the_data_window;
+
+static void
+replace_casereader (struct casereader *s)
+{
+  psppire_data_store_set_reader (the_data_store, s);
+}
 
-PsppireDataStore *data_store = 0;
+#define _(msgid) gettext (msgid)
+#define N_(msgid) msgid
 
 
-static bool parse_command_line (int *argc, char ***argv);
 
 
-int 
-main(int argc, char *argv[]) 
+void
+initialize (struct command_line_processor *clp, int argc, char **argv)
 {
-  PsppireVarStore *var_store ;
-  GtkWidget *data_editor ;
-  GtkSheet *var_sheet ; 
-  GtkSheet *data_sheet ;
+  PsppireDict *dictionary = 0;
 
-  gtk_init(&argc, &argv);
 
-  if ( ! parse_command_line(&argc, &argv) ) 
-    return 0;
+  i18n_init ();
+
+  preregister_widgets ();
+
+  gsl_set_error_handler_off ();
+  fn_init ();
+  outp_init ();
+  settings_init (&viewer_width, &viewer_length);
+  fh_init ();
+  the_source_stream =
+    create_source_stream (
+                         fn_getenv_default ("STAT_INCLUDE_PATH", include_path)
+                         );
 
+  the_dataset = create_dataset ();
 
-  glade_init();
 
-  message_dialog_init();
+  message_dialog_init (the_source_stream);
 
-  the_dictionary = psppire_dict_new();
+  dictionary = psppire_dict_new_from_dict (dataset_dict (the_dataset));
+
+  bind_textdomain_codeset (PACKAGE, "UTF-8");
 
   /* Create the model for the var_sheet */
-  var_store = psppire_var_store_new(the_dictionary);
+  the_var_store = psppire_var_store_new (dictionary);
 
-  /* Create the model for the data sheet */
-  the_cases = psppire_case_array_new(100, 20);
+  the_data_store = psppire_data_store_new (dictionary);
+  replace_casereader (NULL);
 
-  data_store = psppire_data_store_new(the_dictionary, the_cases);
+  create_icon_factory ();
 
-  /* load the interface */
-  xml = glade_xml_new(PKGDATADIR "/psppire.glade", NULL, NULL);
+  {
+    const char *filename = output_file_name ();
 
-  if ( !xml ) return 1;
+    struct string config_string;
 
-  data_editor = get_widget_assert(xml, "data_editor");
-  gtk_window_set_icon_from_file(GTK_WINDOW(data_editor), 
-                               PKGDATADIR "/psppicon.png",0);
+    ds_init_empty (&config_string);
 
-  /* connect the signals in the interface */
-  glade_xml_signal_autoconnect(xml);
+    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);
 
-  var_sheet  = GTK_SHEET(get_widget_assert(xml, "variable_sheet"));
-  data_sheet = GTK_SHEET(get_widget_assert(xml, "data_sheet"));
+    outp_configure_driver_line (ds_ss (&config_string));
 
-  gtk_sheet_set_model(var_sheet, G_SHEET_MODEL(var_store));
-  
-  gtk_sheet_set_model(data_sheet, G_SHEET_MODEL(data_store));
+    unlink (filename);
 
+    ds_destroy (&config_string);
+  }
 
-  gtk_init_add(callbacks_on_init, 0);
+  journal_enable ();
+  textdomain (PACKAGE);
 
-  /* start the event loop */
-  gtk_main();
 
-  message_dialog_done();
+  the_recent_mgr = gtk_recent_manager_get_default ();
 
-  return 0;
+  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)
+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 ();
+  i18n_done ();
+}
+
 
-  int c;
+static void
+func (gpointer key, gpointer value, gpointer data)
+{
+  gboolean rv;
+  PsppireWindow *window = PSPPIRE_WINDOW (value);
 
-  for (;;)
+  g_signal_emit_by_name (window, "delete-event", 0, &rv);
+}
+
+void
+psppire_quit (void)
+{
+  PsppireWindowRegister *reg = psppire_window_register_new ();
+  psppire_window_register_foreach (reg, func, NULL);
+
+  gtk_main_quit ();
+}
+
+
+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 (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 )
+       {
+         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
        {
-       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;
+         g_warning ("Cannot create icon: %s", err->message);
+         g_clear_error (&err);
        }
     }
 
-  return true;
+  {
+    /* Create our own "pspp-stock-reset" item, using the
+       GTK_STOCK_REFRESH icon set */
+
+    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)
+                         );
+
+    gtk_icon_factory_add (factory, "pspp-stock-select",
+                         gtk_icon_factory_lookup_default (GTK_STOCK_INDEX)
+                         );
+  }
+
+  gtk_icon_factory_add_default (factory);
+}
+
+\f
+
+static error_t
+parse_non_options (int key, char *arg, struct argp_state *state)
+{
+  struct source_stream *ss = state->input;
+
+  if ( NULL == ss )
+    return 0;
+
+  switch (key)
+    {
+    case ARGP_KEY_ARG:
+      {
+       gchar *filename = NULL;
+       gchar *utf8 = NULL;
+       const gchar *local_encoding = NULL;
+       gsize written = -1;
+       const gboolean local_is_utf8 = g_get_charset (&local_encoding);
+
+       /* There seems to be no Glib function to convert from local encoding
+          to filename encoding.  Therefore it has to be done in two steps:
+          the intermediate encoding is UTF8.
+
+          Either step could fail.  However, in many cases the file can still
+          be loaded even if the conversion fails. So in those cases, after showing
+          a warning, we simply copy the locally encoded filename to the destination
+          and hope for the best.
+       */
+
+       if ( local_is_utf8)
+         {
+           utf8 = xstrdup (arg);
+         }
+       else
+         {
+           GError *err = NULL;
+           utf8 = g_locale_to_utf8 (arg, -1, NULL, &written, &err);
+           if ( NULL == utf8)
+             {
+               g_warning ("Cannot convert filename from local encoding \"%s\" to UTF-8: %s",
+                          local_encoding,
+                          err->message);
+               g_clear_error (&err);
+             }
+         }
+
+       if ( NULL != utf8)
+         {
+           GError *err = NULL;
+           filename = g_filename_from_utf8 (utf8, written, NULL, NULL, &err);
+           if ( NULL == filename)
+             {
+               g_warning ("Cannot convert filename from UTF8 to filename encoding: %s",
+                          err->message);
+               g_clear_error (&err);
+             }
+         }
+
+       g_free (utf8);
+
+       if ( filename == NULL)
+         filename = xstrdup (arg);
+
+       psppire_window_load (PSPPIRE_WINDOW (the_data_window), filename);
+
+       g_free (filename);
+       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;
 }