1 /* PSPPIRE - a graphical user interface for PSPP.
2 Copyright (C) 2006, 2007, 2008 Free Software Foundation
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 3 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. If not, see <http://www.gnu.org/licenses/>. */
21 #include <glade/glade.h>
24 #include "window-manager.h"
26 #include "psppire-data-editor.h"
30 #include <data/procedure.h>
31 #include "psppire-dialog.h"
32 #include "psppire-selector.h"
33 #include "weight-cases-dialog.h"
34 #include "split-file-dialog.h"
35 #include "transpose-dialog.h"
36 #include "sort-cases-dialog.h"
37 #include "select-cases-dialog.h"
38 #include "compute-dialog.h"
39 #include "goto-case-dialog.h"
40 #include "find-dialog.h"
41 #include "rank-dialog.h"
42 #include "recode-dialog.h"
43 #include "comments-dialog.h"
44 #include "variable-info-dialog.h"
45 #include "descriptives-dialog.h"
46 #include "crosstabs-dialog.h"
47 #include "frequencies-dialog.h"
48 #include "examine-dialog.h"
49 #include "dict-display.h"
50 #include "regression-dialog.h"
51 #include "text-data-import-dialog.h"
53 #include "oneway-anova-dialog.h"
54 #include "t-test-independent-samples-dialog.h"
55 #include "t-test-one-sample.h"
56 #include "t-test-paired-samples.h"
58 #define _(msgid) gettext (msgid)
59 #define N_(msgid) msgid
61 #include "data-editor.h"
62 #include "syntax-editor.h"
63 #include <language/syntax-string-source.h>
64 #include <language/command.h>
65 #include <ui/syntax-gen.h>
66 #include "window-manager.h"
68 #include "psppire-data-store.h"
69 #include "psppire-var-store.h"
71 static void on_edit_copy (GtkMenuItem *, gpointer);
72 static void on_edit_cut (GtkMenuItem *, gpointer);
73 static void on_edit_paste (GtkAction *a, gpointer data);
76 static GtkWidget * create_data_sheet_variable_popup_menu (struct data_editor *);
77 static GtkWidget * create_data_sheet_cases_popup_menu (struct data_editor *);
79 static void register_data_editor_actions (struct data_editor *de);
80 static void on_insert_variable (GtkAction *, gpointer data);
81 static void insert_case (GtkAction *a, gpointer data);
83 static void toggle_value_labels (GtkToggleAction *a, gpointer data);
84 static void toggle_split_window (GtkToggleAction *ta, gpointer data);
87 /* Callback for when the dictionary changes properties*/
88 static void on_weight_change (GObject *, gint, gpointer);
89 static void on_filter_change (GObject *, gint, gpointer);
90 static void on_split_change (PsppireDict *, gpointer);
92 static void on_switch_sheet (GtkNotebook *notebook,
93 GtkNotebookPage *page,
97 static void status_bar_activate (GtkCheckMenuItem *, gpointer);
99 static void grid_lines_activate (GtkCheckMenuItem *, gpointer);
101 static void data_view_activate (GtkCheckMenuItem *, gpointer);
103 static void variable_view_activate (GtkCheckMenuItem *, gpointer );
105 static void fonts_activate (GtkMenuItem *, gpointer);
107 static void file_quit (GtkCheckMenuItem *, gpointer );
110 enable_delete_cases (GtkWidget *w, gint case_num, gpointer data)
112 struct data_editor *de = data;
114 gtk_action_set_visible (de->delete_cases, case_num != -1);
119 enable_delete_variables (GtkWidget *w, gint var, gpointer data)
121 struct data_editor *de = data;
123 gtk_action_set_visible (de->delete_variables, var != -1);
128 /* Run the EXECUTE command. */
130 execute (GtkMenuItem *mi, gpointer data)
132 struct getl_interface *sss = create_syntax_string_source ("EXECUTE.");
134 execute_syntax (sss);
138 transformation_change_callback (bool transformations_pending,
141 struct data_editor *de = data;
142 GtkWidget *menuitem =
143 get_widget_assert (de->xml, "transform_run-pending");
144 GtkWidget *status_label =
145 get_widget_assert (de->xml, "case-counter-area");
147 gtk_widget_set_sensitive (menuitem, transformations_pending);
150 if ( transformations_pending)
151 gtk_label_set_text (GTK_LABEL (status_label),
152 _("Transformations Pending"));
154 gtk_label_set_text (GTK_LABEL (status_label), "");
158 static void open_data_file (const gchar *, struct data_editor *);
161 /* Puts FILE_NAME into the recent list.
162 If it's already in the list, it moves it to the top
165 add_most_recent (const char *file_name)
167 #if RECENT_LISTS_AVAILABLE
169 GtkRecentManager *manager = gtk_recent_manager_get_default();
170 gchar *uri = g_filename_to_uri (file_name, NULL, NULL);
172 gtk_recent_manager_remove_item (manager, uri, NULL);
174 if ( ! gtk_recent_manager_add_item (manager, uri))
175 g_warning ("Could not add item %s to recent list\n",uri);
183 #if RECENT_LISTS_AVAILABLE
186 on_recent_data_select (GtkMenuShell *menushell, gpointer user_data)
189 struct data_editor *de = user_data;
192 gtk_recent_chooser_get_current_uri (GTK_RECENT_CHOOSER (menushell));
194 file = g_filename_from_uri (uri, NULL, NULL);
198 open_data_file (file, de);
204 on_recent_files_select (GtkMenuShell *menushell, gpointer user_data)
208 struct syntax_editor *se ;
211 gtk_recent_chooser_get_current_uri (GTK_RECENT_CHOOSER (menushell));
213 file = g_filename_from_uri (uri, NULL, NULL);
217 se = (struct syntax_editor *)
218 window_create (WINDOW_SYNTAX, file);
220 load_editor_from_file (se, file, NULL);
229 update_paste_menuitems (GtkWidget *w, gboolean x, gpointer data)
231 struct data_editor *de = data;
233 GtkWidget * edit_paste = get_widget_assert (de->xml, "edit_paste");
235 gtk_widget_set_sensitive (edit_paste, x);
239 update_cut_copy_menuitems (GtkWidget *w, gboolean x, gpointer data)
241 struct data_editor *de = data;
243 GtkWidget * edit_copy = get_widget_assert (de->xml, "edit_copy");
244 GtkWidget * edit_cut = get_widget_assert (de->xml, "edit_cut");
246 gtk_widget_set_sensitive (edit_copy, x);
247 gtk_widget_set_sensitive (edit_cut, x);
250 extern PsppireVarStore *the_var_store;
251 extern struct dataset *the_dataset;
252 extern PsppireDataStore *the_data_store ;
256 Create a new data editor.
259 new_data_editor (void)
261 struct data_editor *de ;
262 struct editor_window *e;
266 de = g_malloc0 (sizeof (*de));
268 e = (struct editor_window *) de;
270 de->xml = XML_NEW ("data-editor.glade");
273 vbox = get_widget_assert (de->xml, "vbox1");
275 de->data_editor = PSPPIRE_DATA_EDITOR (psppire_data_editor_new (the_var_store, the_data_store));
277 g_signal_connect (de->data_editor, "data-selection-changed",
278 G_CALLBACK (update_cut_copy_menuitems), de);
280 g_signal_connect (de->data_editor, "data-available-changed",
281 G_CALLBACK (update_paste_menuitems), de);
284 gtk_widget_show (GTK_WIDGET (de->data_editor));
286 gtk_container_add (GTK_CONTAINER (vbox), GTK_WIDGET (de->data_editor));
287 gtk_box_reorder_child (GTK_BOX (vbox) , GTK_WIDGET (de->data_editor), 2);
288 dataset_add_transform_change_callback (the_dataset,
289 transformation_change_callback,
294 g_assert(vs); /* Traps a possible bug in w32 build */
296 g_signal_connect (vs->dict, "weight-changed",
297 G_CALLBACK (on_weight_change),
300 g_signal_connect (vs->dict, "filter-changed",
301 G_CALLBACK (on_filter_change),
304 g_signal_connect (vs->dict, "split-changed",
305 G_CALLBACK (on_split_change),
308 connect_help (de->xml);
312 g_signal_connect (get_widget_assert (de->xml, "edit_copy"),
314 G_CALLBACK (on_edit_copy), de);
316 g_signal_connect (get_widget_assert (de->xml, "edit_cut"),
318 G_CALLBACK (on_edit_cut), de);
321 register_data_editor_actions (de);
323 de->toggle_value_labels =
324 gtk_toggle_action_new ("toggle-value-labels",
326 _("Show/hide value labels"),
327 "pspp-value-labels");
329 g_signal_connect (de->toggle_value_labels, "toggled",
330 G_CALLBACK (toggle_value_labels), de);
333 gtk_action_connect_proxy (GTK_ACTION (de->toggle_value_labels),
334 get_widget_assert (de->xml,
335 "togglebutton-value-labels"));
338 gtk_action_connect_proxy (GTK_ACTION (de->toggle_value_labels),
339 get_widget_assert (de->xml,
340 "view_value-labels"));
343 gtk_action_new ("clear-cases",
345 _("Delete the cases at the selected position(s)"),
348 g_signal_connect_swapped (de->delete_cases, "activate",
349 G_CALLBACK (psppire_data_editor_delete_cases),
352 gtk_action_connect_proxy (de->delete_cases,
353 get_widget_assert (de->xml, "edit_clear-cases"));
355 g_signal_connect (get_widget_assert (de->xml, "edit_paste"), "activate",
356 G_CALLBACK (on_edit_paste),
359 gtk_action_set_visible (de->delete_cases, FALSE);
361 de->delete_variables =
362 gtk_action_new ("clear-variables",
364 _("Delete the variables at the selected position(s)"),
365 "pspp-clear-variables");
367 g_signal_connect_swapped (de->delete_variables, "activate",
368 G_CALLBACK (psppire_data_editor_delete_variables),
371 gtk_action_connect_proxy (de->delete_variables,
372 get_widget_assert (de->xml, "edit_clear-variables")
375 gtk_action_set_visible (de->delete_variables, FALSE);
377 de->insert_variable =
378 gtk_action_new ("insert-variable",
379 _("Insert _Variable"),
380 _("Create a new variable at the current position"),
381 "pspp-insert-variable");
383 g_signal_connect (de->insert_variable, "activate",
384 G_CALLBACK (on_insert_variable), de->data_editor);
387 gtk_action_connect_proxy (de->insert_variable,
388 get_widget_assert (de->xml, "button-insert-variable")
391 gtk_action_connect_proxy (de->insert_variable,
392 get_widget_assert (de->xml, "edit_insert-variable")
397 gtk_action_new ("insert-case",
399 _("Create a new case at the current position"),
402 g_signal_connect (de->insert_case, "activate",
403 G_CALLBACK (insert_case), de);
406 gtk_action_connect_proxy (de->insert_case,
407 get_widget_assert (de->xml, "button-insert-case")
411 gtk_action_connect_proxy (de->insert_case,
412 get_widget_assert (de->xml, "edit_insert-case")
417 de->invoke_goto_dialog =
418 gtk_action_new ("goto-case-dialog",
420 _("Jump to a Case in the Data Sheet"),
424 gtk_action_connect_proxy (de->invoke_goto_dialog,
425 get_widget_assert (de->xml, "button-goto-case")
428 gtk_action_connect_proxy (de->invoke_goto_dialog,
429 get_widget_assert (de->xml, "edit_goto-case")
433 g_signal_connect (de->invoke_goto_dialog, "activate",
434 G_CALLBACK (goto_case_dialog), de);
437 de->invoke_weight_cases_dialog =
438 gtk_action_new ("weight-cases-dialog",
440 _("Weight cases by variable"),
441 "pspp-weight-cases");
443 g_signal_connect (de->invoke_weight_cases_dialog, "activate",
444 G_CALLBACK (weight_cases_dialog), de);
447 de->invoke_transpose_dialog =
448 gtk_action_new ("transpose-dialog",
450 _("Transpose the cases with the variables"),
454 g_signal_connect (de->invoke_transpose_dialog, "activate",
455 G_CALLBACK (transpose_dialog), de);
459 de->invoke_split_file_dialog =
460 gtk_action_new ("split-file-dialog",
462 _("Split the active file"),
465 g_signal_connect (de->invoke_split_file_dialog, "activate",
466 G_CALLBACK (split_file_dialog), de);
470 de->invoke_sort_cases_dialog =
471 gtk_action_new ("sort-cases-dialog",
473 _("Sort cases in the active file"),
476 g_signal_connect (de->invoke_sort_cases_dialog, "activate",
477 G_CALLBACK (sort_cases_dialog), de);
479 de->invoke_select_cases_dialog =
480 gtk_action_new ("select-cases-dialog",
482 _("Select cases from the active file"),
483 "pspp-select-cases");
485 g_signal_connect (de->invoke_select_cases_dialog, "activate",
486 G_CALLBACK (select_cases_dialog), de);
489 de->invoke_compute_dialog =
490 gtk_action_new ("compute-dialog",
492 _("Compute new values for a variable"),
495 g_signal_connect (de->invoke_compute_dialog, "activate",
496 G_CALLBACK (compute_dialog), de);
498 de->invoke_oneway_anova_dialog =
499 gtk_action_new ("oneway-anova",
501 _("Perform one way analysis of variance"),
504 g_signal_connect (de->invoke_oneway_anova_dialog, "activate",
505 G_CALLBACK (oneway_anova_dialog), de);
507 de->invoke_t_test_independent_samples_dialog =
508 gtk_action_new ("t-test-independent-samples",
509 _("_Independent Samples T Test"),
510 _("Calculate T Test for samples from independent groups"),
513 g_signal_connect (de->invoke_t_test_independent_samples_dialog, "activate",
514 G_CALLBACK (t_test_independent_samples_dialog), de);
517 de->invoke_t_test_paired_samples_dialog =
518 gtk_action_new ("t-test-paired-samples",
519 _("_Paired Samples T Test"),
520 _("Calculate T Test for paired samples"),
523 g_signal_connect (de->invoke_t_test_paired_samples_dialog, "activate",
524 G_CALLBACK (t_test_paired_samples_dialog), de);
527 de->invoke_t_test_one_sample_dialog =
528 gtk_action_new ("t-test-one-sample",
529 _("One _Sample T Test"),
530 _("Calculate T Test for sample from a single distribution"),
533 g_signal_connect (de->invoke_t_test_one_sample_dialog, "activate",
534 G_CALLBACK (t_test_one_sample_dialog), de);
537 de->invoke_comments_dialog =
538 gtk_action_new ("commments-dialog",
539 _("Data File _Comments"),
540 _("Commentary text for the data file"),
543 g_signal_connect (de->invoke_comments_dialog, "activate",
544 G_CALLBACK (comments_dialog), de);
546 de->invoke_find_dialog =
547 gtk_action_new ("find-dialog",
552 g_signal_connect (de->invoke_find_dialog, "activate",
553 G_CALLBACK (find_dialog), de);
556 de->invoke_rank_dialog =
557 gtk_action_new ("rank-dialog",
562 g_signal_connect (de->invoke_rank_dialog, "activate",
563 G_CALLBACK (rank_dialog), de);
566 de->invoke_recode_same_dialog =
567 gtk_action_new ("recode-same-dialog",
568 _("Recode into _Same Variables"),
569 _("Recode values into the same Variables"),
572 g_signal_connect (de->invoke_recode_same_dialog, "activate",
573 G_CALLBACK (recode_same_dialog), de);
576 de->invoke_recode_different_dialog =
577 gtk_action_new ("recode-different-dialog",
578 _("Recode into _Different Variables"),
579 _("Recode values into different Variables"),
580 "pspp-recode-different");
582 g_signal_connect (de->invoke_recode_different_dialog, "activate",
583 G_CALLBACK (recode_different_dialog), de);
586 de->invoke_variable_info_dialog =
587 gtk_action_new ("variable-info-dialog",
589 _("Jump to Variable"),
590 "pspp-goto-variable");
592 g_signal_connect (de->invoke_variable_info_dialog, "activate",
593 G_CALLBACK (variable_info_dialog), de);
595 de->invoke_descriptives_dialog =
596 gtk_action_new ("descriptives-dialog",
598 _("Calculate descriptive statistics (mean, variance, ...)"),
599 "pspp-descriptives");
601 g_signal_connect (de->invoke_descriptives_dialog, "activate",
602 G_CALLBACK (descriptives_dialog), de);
605 de->invoke_frequencies_dialog =
606 gtk_action_new ("frequencies-dialog",
608 _("Generate frequency statistics"),
611 g_signal_connect (de->invoke_frequencies_dialog, "activate",
612 G_CALLBACK (frequencies_dialog), de);
614 de->invoke_crosstabs_dialog =
615 gtk_action_new ("crosstabs-dialog",
617 _("Generate crosstabulations"),
620 g_signal_connect (de->invoke_crosstabs_dialog, "activate",
621 G_CALLBACK (crosstabs_dialog), de);
624 de->invoke_examine_dialog =
625 gtk_action_new ("examine-dialog",
627 _("Examine Data by Factors"),
630 g_signal_connect (de->invoke_examine_dialog, "activate",
631 G_CALLBACK (examine_dialog), de);
634 de->invoke_regression_dialog =
635 gtk_action_new ("regression-dialog",
636 _("Linear _Regression"),
637 _("Estimate parameters of the linear model"),
640 g_signal_connect (de->invoke_regression_dialog, "activate",
641 G_CALLBACK (regression_dialog), de);
643 e->window = GTK_WINDOW (get_widget_assert (de->xml, "data_editor"));
645 g_signal_connect_swapped (get_widget_assert (de->xml,"file_new_data"),
647 G_CALLBACK (gtk_action_activate),
648 de->action_data_new);
650 g_signal_connect_swapped (get_widget_assert (de->xml,"file_open_data"),
652 G_CALLBACK (gtk_action_activate),
653 de->action_data_open);
655 #if RECENT_LISTS_AVAILABLE
657 GtkRecentManager *rm = gtk_recent_manager_get_default ();
658 GtkWidget *recent_data = get_widget_assert (de->xml, "file_recent-data");
659 GtkWidget *recent_files = get_widget_assert (de->xml, "file_recent-files");
660 GtkWidget *recent_separator = get_widget_assert (de->xml, "file_separator1");
662 GtkWidget *menu = gtk_recent_chooser_menu_new_for_manager (rm);
664 GtkRecentFilter *filter = gtk_recent_filter_new ();
666 gtk_widget_show (recent_data);
667 gtk_widget_show (recent_files);
668 gtk_widget_show (recent_separator);
670 gtk_recent_filter_add_pattern (filter, "*.sav");
671 gtk_recent_filter_add_pattern (filter, "*.SAV");
673 gtk_recent_chooser_add_filter (GTK_RECENT_CHOOSER (menu), filter);
675 gtk_widget_set_sensitive (recent_data, TRUE);
676 g_signal_connect (menu, "selection-done",
677 G_CALLBACK (on_recent_data_select), de);
679 gtk_menu_item_set_submenu (GTK_MENU_ITEM (recent_data), menu);
682 filter = gtk_recent_filter_new ();
683 menu = gtk_recent_chooser_menu_new_for_manager (rm);
685 gtk_recent_filter_add_pattern (filter, "*.sps");
686 gtk_recent_filter_add_pattern (filter, "*.SPS");
688 gtk_recent_chooser_add_filter (GTK_RECENT_CHOOSER (menu), filter);
690 gtk_widget_set_sensitive (recent_files, TRUE);
691 g_signal_connect (menu, "selection-done",
692 G_CALLBACK (on_recent_files_select), de);
694 gtk_menu_item_set_submenu (GTK_MENU_ITEM (recent_files), menu);
698 g_signal_connect (get_widget_assert (de->xml,"file_new_syntax"),
700 G_CALLBACK (new_syntax_window),
703 g_signal_connect (get_widget_assert (de->xml,"file_open_syntax"),
705 G_CALLBACK (open_syntax_window),
708 g_signal_connect_swapped (get_widget_assert (de->xml,"file_import-text"),
710 G_CALLBACK (gtk_action_activate),
711 de->invoke_text_import_assistant);
713 g_signal_connect_swapped (get_widget_assert (de->xml,"file_save"),
715 G_CALLBACK (gtk_action_activate),
716 de->action_data_save);
718 g_signal_connect_swapped (get_widget_assert (de->xml,"file_save_as"),
720 G_CALLBACK (gtk_action_activate),
721 de->action_data_save_as);
723 gtk_action_connect_proxy (de->invoke_find_dialog,
724 get_widget_assert (de->xml, "edit_find")
727 gtk_action_connect_proxy (de->invoke_find_dialog,
728 get_widget_assert (de->xml, "button-find")
731 gtk_action_connect_proxy (de->invoke_rank_dialog,
732 get_widget_assert (de->xml, "transform_rank")
735 gtk_action_connect_proxy (de->invoke_recode_same_dialog,
736 get_widget_assert (de->xml,
737 "transform_recode-same")
740 gtk_action_connect_proxy (de->invoke_recode_different_dialog,
741 get_widget_assert (de->xml,
742 "transform_recode-different")
745 gtk_action_connect_proxy (de->invoke_weight_cases_dialog,
746 get_widget_assert (de->xml, "data_weight-cases")
749 gtk_action_connect_proxy (de->invoke_transpose_dialog,
750 get_widget_assert (de->xml, "data_transpose")
753 gtk_action_connect_proxy (de->invoke_split_file_dialog,
754 get_widget_assert (de->xml, "data_split-file")
757 gtk_action_connect_proxy (de->invoke_sort_cases_dialog,
758 get_widget_assert (de->xml, "data_sort-cases")
761 gtk_action_connect_proxy (de->invoke_select_cases_dialog,
762 get_widget_assert (de->xml, "data_select-cases")
765 gtk_action_connect_proxy (de->invoke_compute_dialog,
766 get_widget_assert (de->xml, "transform_compute")
769 gtk_action_connect_proxy (de->invoke_t_test_independent_samples_dialog,
770 get_widget_assert (de->xml,
775 gtk_action_connect_proxy (de->invoke_t_test_paired_samples_dialog,
776 get_widget_assert (de->xml,
781 gtk_action_connect_proxy (de->invoke_t_test_one_sample_dialog,
782 get_widget_assert (de->xml,
787 gtk_action_connect_proxy (de->invoke_oneway_anova_dialog,
788 get_widget_assert (de->xml,
793 gtk_action_connect_proxy (de->invoke_comments_dialog,
794 get_widget_assert (de->xml, "utilities_comments")
797 gtk_action_connect_proxy (de->invoke_variable_info_dialog,
798 get_widget_assert (de->xml, "utilities_variables")
801 gtk_action_connect_proxy (de->invoke_descriptives_dialog,
802 get_widget_assert (de->xml, "analyze_descriptives")
805 gtk_action_connect_proxy (de->invoke_crosstabs_dialog,
806 get_widget_assert (de->xml, "crosstabs")
809 gtk_action_connect_proxy (de->invoke_frequencies_dialog,
810 get_widget_assert (de->xml, "analyze_frequencies")
814 gtk_action_connect_proxy (de->invoke_examine_dialog,
815 get_widget_assert (de->xml, "analyze_explore")
818 gtk_action_connect_proxy (de->invoke_regression_dialog,
819 get_widget_assert (de->xml, "linear-regression")
822 g_signal_connect (get_widget_assert (de->xml,"help_about"),
824 G_CALLBACK (about_new),
828 g_signal_connect (get_widget_assert (de->xml,"help_reference"),
830 G_CALLBACK (reference_manual),
834 g_signal_connect (de->data_editor,
836 G_CALLBACK (enable_delete_cases),
840 g_signal_connect (de->data_editor,
841 "variables-selected",
842 G_CALLBACK (enable_delete_variables),
846 g_signal_connect (GTK_NOTEBOOK (de->data_editor),
848 G_CALLBACK (on_switch_sheet), de);
850 gtk_notebook_set_current_page (GTK_NOTEBOOK (de->data_editor), PSPPIRE_DATA_EDITOR_VARIABLE_VIEW);
851 gtk_notebook_set_current_page (GTK_NOTEBOOK (de->data_editor), PSPPIRE_DATA_EDITOR_DATA_VIEW);
853 g_signal_connect (get_widget_assert (de->xml, "view_statusbar"),
855 G_CALLBACK (status_bar_activate), de);
858 g_signal_connect (get_widget_assert (de->xml, "view_gridlines"),
860 G_CALLBACK (grid_lines_activate), de);
864 g_signal_connect (get_widget_assert (de->xml, "view_data"),
866 G_CALLBACK (data_view_activate), de);
868 g_signal_connect (get_widget_assert (de->xml, "view_variables"),
870 G_CALLBACK (variable_view_activate), de);
874 g_signal_connect (get_widget_assert (de->xml, "view_fonts"),
876 G_CALLBACK (fonts_activate), de);
881 gtk_action_connect_proxy (de->action_data_open,
882 get_widget_assert (de->xml, "button-open")
885 gtk_action_connect_proxy (de->action_data_save,
886 get_widget_assert (de->xml, "button-save")
889 gtk_action_connect_proxy (de->invoke_variable_info_dialog,
890 get_widget_assert (de->xml, "button-goto-variable")
893 gtk_action_connect_proxy (de->invoke_weight_cases_dialog,
894 get_widget_assert (de->xml, "button-weight-cases")
897 gtk_action_connect_proxy (de->invoke_split_file_dialog,
898 get_widget_assert (de->xml, "button-split-file")
901 gtk_action_connect_proxy (de->invoke_select_cases_dialog,
902 get_widget_assert (de->xml, "button-select-cases")
906 g_signal_connect (get_widget_assert (de->xml, "file_quit"),
908 G_CALLBACK (file_quit), de);
910 g_signal_connect (get_widget_assert (de->xml, "transform_run-pending"),
912 G_CALLBACK (execute), de);
915 g_signal_connect (get_widget_assert (de->xml, "windows_minimise_all"),
917 G_CALLBACK (minimise_all_windows), NULL);
919 de->toggle_split_window =
920 gtk_toggle_action_new ("toggle-split-window",
922 _("Split the window vertically and horizontally"),
923 "pspp-split-window");
925 g_signal_connect (de->toggle_split_window, "toggled",
926 G_CALLBACK (toggle_split_window),
929 gtk_action_connect_proxy (GTK_ACTION (de->toggle_split_window),
930 get_widget_assert (de->xml,
933 de->data_sheet_variable_popup_menu =
934 GTK_MENU (create_data_sheet_variable_popup_menu (de));
936 de->data_sheet_cases_popup_menu =
937 GTK_MENU (create_data_sheet_cases_popup_menu (de));
940 g_object_set (de->data_editor,
941 "column-menu", de->data_sheet_variable_popup_menu, NULL);
944 g_object_set (de->data_editor,
945 "row-menu", de->data_sheet_cases_popup_menu, NULL);
952 new_data_window (GtkMenuItem *menuitem, gpointer parent)
954 window_create (WINDOW_DATA, NULL);
957 /* Callback for when the datasheet/varsheet is selected */
959 on_switch_sheet (GtkNotebook *notebook,
960 GtkNotebookPage *page,
964 struct data_editor *de = user_data;
966 GtkWidget *view_data = get_widget_assert (de->xml, "view_data");
967 GtkWidget *view_variables = get_widget_assert (de->xml, "view_variables");
971 case PSPPIRE_DATA_EDITOR_VARIABLE_VIEW:
972 gtk_widget_hide (view_variables);
973 gtk_widget_show (view_data);
974 gtk_action_set_sensitive (de->insert_variable, TRUE);
975 gtk_action_set_sensitive (de->insert_case, FALSE);
976 gtk_action_set_sensitive (de->invoke_goto_dialog, FALSE);
978 case PSPPIRE_DATA_EDITOR_DATA_VIEW:
979 gtk_widget_show (view_variables);
980 gtk_widget_hide (view_data);
981 gtk_action_set_sensitive (de->invoke_goto_dialog, TRUE);
982 gtk_action_set_sensitive (de->insert_case, TRUE);
985 g_assert_not_reached ();
990 update_paste_menuitem (de, page_num);
996 status_bar_activate (GtkCheckMenuItem *menuitem, gpointer data)
998 struct data_editor *de = data;
999 GtkWidget *statusbar = get_widget_assert (de->xml, "status-bar");
1001 if ( gtk_check_menu_item_get_active (menuitem) )
1002 gtk_widget_show (statusbar);
1004 gtk_widget_hide (statusbar);
1009 grid_lines_activate (GtkCheckMenuItem *menuitem, gpointer data)
1011 struct data_editor *de = data;
1012 const gboolean grid_visible = gtk_check_menu_item_get_active (menuitem);
1014 psppire_data_editor_show_grid (de->data_editor, grid_visible);
1020 data_view_activate (GtkCheckMenuItem *menuitem, gpointer data)
1022 struct data_editor *de = data;
1024 gtk_notebook_set_current_page (GTK_NOTEBOOK (de->data_editor), PSPPIRE_DATA_EDITOR_DATA_VIEW);
1029 variable_view_activate (GtkCheckMenuItem *menuitem, gpointer data)
1031 struct data_editor *de = data;
1033 gtk_notebook_set_current_page (GTK_NOTEBOOK (de->data_editor), PSPPIRE_DATA_EDITOR_VARIABLE_VIEW);
1038 fonts_activate (GtkMenuItem *menuitem, gpointer data)
1040 struct data_editor *de = data;
1042 gtk_font_selection_dialog_new (_("Font Selection"));
1044 gtk_window_set_transient_for (GTK_WINDOW (dialog),
1045 GTK_WINDOW (get_widget_assert (de->xml,
1047 if ( GTK_RESPONSE_OK == gtk_dialog_run (GTK_DIALOG (dialog)) )
1049 const gchar *font = gtk_font_selection_dialog_get_font_name
1050 (GTK_FONT_SELECTION_DIALOG (dialog));
1052 PangoFontDescription* font_desc =
1053 pango_font_description_from_string (font);
1055 psppire_data_editor_set_font (de->data_editor, font_desc);
1058 gtk_widget_hide (dialog);
1063 /* Callback for the value labels action */
1065 toggle_value_labels (GtkToggleAction *ta, gpointer data)
1067 struct data_editor *de = data;
1069 g_object_set (de->data_editor, "value-labels", gtk_toggle_action_get_active (ta), NULL);
1075 toggle_split_window (GtkToggleAction *ta, gpointer data)
1077 struct data_editor *de = data;
1079 psppire_data_editor_split_window (de->data_editor,
1080 gtk_toggle_action_get_active (ta));
1087 file_quit (GtkCheckMenuItem *menuitem, gpointer data)
1089 /* FIXME: Need to be more intelligent here.
1090 Give the user the opportunity to save any unsaved data.
1092 g_object_unref (the_data_store);
1098 insert_case (GtkAction *action, gpointer data)
1100 struct data_editor *de = data;
1102 psppire_data_editor_insert_case (de->data_editor);
1106 on_insert_variable (GtkAction *action, gpointer data)
1108 PsppireDataEditor *de = PSPPIRE_DATA_EDITOR (data);
1109 psppire_data_editor_insert_variable (de);
1113 /* Callback for when the dictionary changes its split variables */
1115 on_split_change (PsppireDict *dict, gpointer data)
1117 struct data_editor *de = data;
1119 size_t n_split_vars = dict_get_split_cnt (dict->dict);
1121 GtkWidget *split_status_area =
1122 get_widget_assert (de->xml, "split-file-status-area");
1124 if ( n_split_vars == 0 )
1126 gtk_label_set_text (GTK_LABEL (split_status_area), _("No Split"));
1132 const struct variable *const * split_vars =
1133 dict_get_split_vars (dict->dict);
1135 text = g_string_new (_("Split by "));
1137 for (i = 0 ; i < n_split_vars - 1; ++i )
1139 g_string_append_printf (text, "%s, ", var_get_name (split_vars[i]));
1141 g_string_append (text, var_get_name (split_vars[i]));
1143 gtk_label_set_text (GTK_LABEL (split_status_area), text->str);
1145 g_string_free (text, TRUE);
1150 /* Callback for when the dictionary changes its filter variable */
1152 on_filter_change (GObject *o, gint filter_index, gpointer data)
1154 struct data_editor *de = data;
1155 GtkWidget *filter_status_area =
1156 get_widget_assert (de->xml, "filter-use-status-area");
1158 if ( filter_index == -1 )
1160 gtk_label_set_text (GTK_LABEL (filter_status_area), _("Filter off"));
1164 PsppireVarStore *vs = NULL;
1165 struct variable *var ;
1168 g_object_get (de->data_editor, "var-store", &vs, NULL);
1170 var = psppire_dict_get_variable (vs->dict, filter_index);
1172 text = g_strdup_printf (_("Filter by %s"), var_get_name (var));
1174 gtk_label_set_text (GTK_LABEL (filter_status_area), text);
1180 /* Callback for when the dictionary changes its weights */
1182 on_weight_change (GObject *o, gint weight_index, gpointer data)
1184 struct data_editor *de = data;
1185 GtkWidget *weight_status_area =
1186 get_widget_assert (de->xml, "weight-status-area");
1188 if ( weight_index == -1 )
1190 gtk_label_set_text (GTK_LABEL (weight_status_area), _("Weights off"));
1194 struct variable *var ;
1195 PsppireVarStore *vs = NULL;
1198 g_object_get (de->data_editor, "var-store", &vs, NULL);
1200 var = psppire_dict_get_variable (vs->dict, weight_index);
1202 text = g_strdup_printf (_("Weight by %s"), var_get_name (var));
1204 gtk_label_set_text (GTK_LABEL (weight_status_area), text);
1213 static void data_save_as_dialog (GtkAction *, struct data_editor *de);
1214 static void new_file (GtkAction *, struct editor_window *de);
1215 static void open_data_dialog (GtkAction *, struct data_editor *de);
1216 static void data_save (GtkAction *action, struct data_editor *e);
1219 /* Create the GtkActions and connect to their signals */
1221 register_data_editor_actions (struct data_editor *de)
1223 de->action_data_open =
1224 gtk_action_new ("data-open-dialog",
1226 _("Open a data file"),
1229 g_signal_connect (de->action_data_open, "activate",
1230 G_CALLBACK (open_data_dialog), de);
1233 de->action_data_save = gtk_action_new ("data-save",
1235 _("Save data to file"),
1238 g_signal_connect (de->action_data_save, "activate",
1239 G_CALLBACK (data_save), de);
1243 de->action_data_save_as = gtk_action_new ("data-save-as-dialog",
1245 _("Save data to file"),
1248 g_signal_connect (de->action_data_save_as, "activate",
1249 G_CALLBACK (data_save_as_dialog), de);
1251 de->action_data_new =
1252 gtk_action_new ("data-new",
1257 g_signal_connect (de->action_data_new, "activate",
1258 G_CALLBACK (new_file), de);
1260 de->invoke_text_import_assistant =
1261 gtk_action_new ("file_import-text",
1262 _("_Import Text Data"),
1263 _("Import text data file"),
1266 g_signal_connect (de->invoke_text_import_assistant, "activate",
1267 G_CALLBACK (text_data_import_assistant), de);
1270 /* Returns true if NAME has a suffix which might denote a PSPP file */
1272 name_has_suffix (const gchar *name)
1274 if ( g_str_has_suffix (name, ".sav"))
1276 if ( g_str_has_suffix (name, ".SAV"))
1278 if ( g_str_has_suffix (name, ".por"))
1280 if ( g_str_has_suffix (name, ".POR"))
1286 /* Append SUFFIX to the filename of DE */
1288 append_filename_suffix (struct data_editor *de, const gchar *suffix)
1290 if ( ! name_has_suffix (de->file_name))
1292 gchar *s = de->file_name;
1293 de->file_name = g_strconcat (de->file_name, suffix, NULL);
1298 /* Save DE to file */
1300 save_file (struct data_editor *de)
1302 struct getl_interface *sss;
1303 struct string file_name ;
1305 g_assert (de->file_name);
1307 ds_init_empty (&file_name);
1308 syntax_gen_string (&file_name, ss_cstr (de->file_name));
1310 if ( de->save_as_portable )
1312 append_filename_suffix (de, ".por");
1313 sss = create_syntax_string_source ("EXPORT OUTFILE=%s.",
1314 ds_cstr (&file_name));
1318 append_filename_suffix (de, ".sav");
1319 sss = create_syntax_string_source ("SAVE OUTFILE=%s.",
1320 ds_cstr (&file_name));
1323 ds_destroy (&file_name);
1325 execute_syntax (sss);
1329 /* Callback for data_save action.
1330 If there's an existing file name, then just save,
1331 otherwise prompt for a file name, then save */
1333 data_save (GtkAction *action, struct data_editor *de)
1338 data_save_as_dialog (action, de);
1342 /* Callback for data_save_as action. Prompt for a filename and save */
1344 data_save_as_dialog (GtkAction *action, struct data_editor *de)
1346 struct editor_window *e = (struct editor_window *) de;
1348 GtkWidget *button_sys;
1350 gtk_file_chooser_dialog_new (_("Save"),
1351 GTK_WINDOW (e->window),
1352 GTK_FILE_CHOOSER_ACTION_SAVE,
1353 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
1354 GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT,
1357 GtkFileFilter *filter = gtk_file_filter_new ();
1358 gtk_file_filter_set_name (filter, _("System Files (*.sav)"));
1359 gtk_file_filter_add_pattern (filter, "*.sav");
1360 gtk_file_filter_add_pattern (filter, "*.SAV");
1361 gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (dialog), filter);
1363 filter = gtk_file_filter_new ();
1364 gtk_file_filter_set_name (filter, _("Portable Files (*.por) "));
1365 gtk_file_filter_add_pattern (filter, "*.por");
1366 gtk_file_filter_add_pattern (filter, "*.POR");
1367 gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (dialog), filter);
1369 filter = gtk_file_filter_new ();
1370 gtk_file_filter_set_name (filter, _("All Files"));
1371 gtk_file_filter_add_pattern (filter, "*");
1372 gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (dialog), filter);
1375 GtkWidget *button_por;
1376 GtkWidget *vbox = gtk_vbox_new (TRUE, 5);
1378 gtk_radio_button_new_with_label (NULL, _("System File"));
1381 gtk_radio_button_new_with_label
1382 (gtk_radio_button_get_group (GTK_RADIO_BUTTON(button_sys)),
1383 _("Portable File"));
1385 gtk_box_pack_start_defaults (GTK_BOX (vbox), button_sys);
1386 gtk_box_pack_start_defaults (GTK_BOX (vbox), button_por);
1388 gtk_widget_show_all (vbox);
1390 gtk_file_chooser_set_extra_widget (GTK_FILE_CHOOSER(dialog), vbox);
1393 switch (gtk_dialog_run (GTK_DIALOG (dialog)))
1395 case GTK_RESPONSE_ACCEPT:
1397 g_free (de->file_name);
1400 gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog));
1402 de->save_as_portable =
1403 ! gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (button_sys));
1407 window_set_name_from_filename (e, de->file_name);
1414 gtk_widget_destroy (dialog);
1418 /* Callback for data_new action.
1419 Performs the NEW FILE command */
1421 new_file (GtkAction *action, struct editor_window *e)
1423 struct data_editor *de = (struct data_editor *) e;
1425 struct getl_interface *sss =
1426 create_syntax_string_source ("NEW FILE.");
1428 execute_syntax (sss);
1430 g_free (de->file_name);
1431 de->file_name = NULL;
1433 default_window_name (e);
1438 open_data_file (const gchar *file_name, struct data_editor *de)
1440 struct getl_interface *sss;
1441 struct string filename;
1443 ds_init_empty (&filename);
1444 syntax_gen_string (&filename, ss_cstr (file_name));
1446 sss = create_syntax_string_source ("GET FILE=%s.",
1447 ds_cstr (&filename));
1448 ds_destroy (&filename);
1450 if (execute_syntax (sss) )
1452 window_set_name_from_filename ((struct editor_window *) de, file_name);
1453 add_most_recent (file_name);
1459 /* Callback for the data_open action.
1460 Prompts for a filename and opens it */
1462 open_data_dialog (GtkAction *action, struct data_editor *de)
1464 struct editor_window *e = (struct editor_window *) de;
1467 gtk_file_chooser_dialog_new (_("Open"),
1468 GTK_WINDOW (e->window),
1469 GTK_FILE_CHOOSER_ACTION_OPEN,
1470 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
1471 GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT,
1474 GtkFileFilter *filter = gtk_file_filter_new ();
1475 gtk_file_filter_set_name (filter, _("System Files (*.sav)"));
1476 gtk_file_filter_add_pattern (filter, "*.sav");
1477 gtk_file_filter_add_pattern (filter, "*.SAV");
1478 gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (dialog), filter);
1480 filter = gtk_file_filter_new ();
1481 gtk_file_filter_set_name (filter, _("Portable Files (*.por) "));
1482 gtk_file_filter_add_pattern (filter, "*.por");
1483 gtk_file_filter_add_pattern (filter, "*.POR");
1484 gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (dialog), filter);
1486 filter = gtk_file_filter_new ();
1487 gtk_file_filter_set_name (filter, _("All Files"));
1488 gtk_file_filter_add_pattern (filter, "*");
1489 gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (dialog), filter);
1494 gchar *dir_name = g_path_get_dirname (de->file_name);
1495 gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog),
1500 switch (gtk_dialog_run (GTK_DIALOG (dialog)))
1502 case GTK_RESPONSE_ACCEPT:
1504 g_free (de->file_name);
1506 gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog));
1508 open_data_file (de->file_name, de);
1515 gtk_widget_destroy (dialog);
1520 create_data_sheet_variable_popup_menu (struct data_editor *de)
1522 GtkWidget *menu = gtk_menu_new ();
1524 GtkWidget *sort_ascending =
1525 gtk_menu_item_new_with_label (_("Sort Ascending"));
1527 GtkWidget *sort_descending =
1528 gtk_menu_item_new_with_label (_("Sort Descending"));
1530 GtkWidget *insert_variable =
1531 gtk_menu_item_new_with_label (_("Insert Variable"));
1533 GtkWidget *clear_variable =
1534 gtk_menu_item_new_with_label (_("Clear"));
1537 gtk_action_connect_proxy (de->delete_variables,
1541 gtk_menu_shell_append (GTK_MENU_SHELL (menu), insert_variable);
1544 gtk_menu_shell_append (GTK_MENU_SHELL (menu),
1545 gtk_separator_menu_item_new ());
1548 gtk_menu_shell_append (GTK_MENU_SHELL (menu), clear_variable);
1551 gtk_menu_shell_append (GTK_MENU_SHELL (menu),
1552 gtk_separator_menu_item_new ());
1555 gtk_menu_shell_append (GTK_MENU_SHELL (menu), sort_ascending);
1558 g_signal_connect_swapped (G_OBJECT (sort_ascending), "activate",
1559 G_CALLBACK (psppire_data_editor_sort_ascending),
1562 g_signal_connect_swapped (G_OBJECT (sort_descending), "activate",
1563 G_CALLBACK (psppire_data_editor_sort_descending),
1566 g_signal_connect_swapped (G_OBJECT (insert_variable), "activate",
1567 G_CALLBACK (gtk_action_activate),
1568 de->insert_variable);
1571 gtk_menu_shell_append (GTK_MENU_SHELL (menu), sort_descending);
1573 gtk_widget_show_all (menu);
1580 create_data_sheet_cases_popup_menu (struct data_editor *de)
1582 GtkWidget *menu = gtk_menu_new ();
1584 GtkWidget *insert_case =
1585 gtk_menu_item_new_with_label (_("Insert Case"));
1587 GtkWidget *delete_case =
1588 gtk_menu_item_new_with_label (_("Clear"));
1591 gtk_action_connect_proxy (de->delete_cases,
1595 gtk_menu_shell_append (GTK_MENU_SHELL (menu), insert_case);
1597 g_signal_connect_swapped (G_OBJECT (insert_case), "activate",
1598 G_CALLBACK (gtk_action_activate),
1602 gtk_menu_shell_append (GTK_MENU_SHELL (menu),
1603 gtk_separator_menu_item_new ());
1606 gtk_menu_shell_append (GTK_MENU_SHELL (menu), delete_case);
1609 gtk_widget_show_all (menu);
1618 on_edit_paste (GtkAction *a, gpointer data)
1620 struct data_editor *de = data;
1622 psppire_data_editor_clip_paste (de->data_editor);
1626 on_edit_copy (GtkMenuItem *m, gpointer data)
1628 struct data_editor *de = data;
1630 psppire_data_editor_clip_copy (de->data_editor);
1636 on_edit_cut (GtkMenuItem *m, gpointer data)
1638 struct data_editor *de = data;
1640 psppire_data_editor_clip_cut (de->data_editor);