/* PSPPIRE - a graphical user interface for PSPP. Copyright (C) 2015 Free Software Foundation 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 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 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, see . */ #include #include "psppire-dialog-action-descriptives.h" #include "psppire-checkbox-treeview.h" #include "psppire-var-view.h" #include "psppire-dict.h" #include "psppire-dialog.h" #include "builder-wrapper.h" #include "gettext.h" #define _(msgid) gettext (msgid) #define N_(msgid) msgid static void psppire_dialog_action_descriptives_class_init (PsppireDialogActionDescriptivesClass *class); G_DEFINE_TYPE (PsppireDialogActionDescriptives, psppire_dialog_action_descriptives, PSPPIRE_TYPE_DIALOG_ACTION); #define DESCRIPTIVE_STATS \ DS (MEAN, N_("Mean"), NULL) \ DS (STDDEV, N_("Standard deviation"), NULL) \ DS (MINIMUM, N_("Minimum"), N_("Minimum value")) \ DS (MAXIMUM, N_("Maximum"), N_("Maximum value")) \ DS (RANGE, N_("Range"), NULL) \ DS (SUM, N_("Sum"), NULL) \ DS (SEMEAN, N_("Standard error"), N_("Standard error of mean")) \ DS (VARIANCE, N_("Variance"), NULL) \ DS (KURTOSIS, N_("Kurtosis"), N_("Kurtosis and standard error of kurtosis")) \ DS (SKEWNESS, N_("Skewness"), N_("Skewness and standard error of skewness")) enum { #define DS(NAME, LABEL, TOOLTIP) DS_##NAME, DESCRIPTIVE_STATS #undef DS N_DESCRIPTIVE_STATS }; enum { #define DS(NAME, LABEL, TOOLTIP) B_DS_##NAME = 1u << DS_##NAME, DESCRIPTIVE_STATS #undef DS B_DS_ALL = (1u << N_DESCRIPTIVE_STATS) - 1, B_DS_DEFAULT = B_DS_MEAN | B_DS_STDDEV | B_DS_MINIMUM | B_DS_MAXIMUM }; static const struct checkbox_entry_item stats[] = { #define DS(NAME, LABEL, TOOLTIP) {#NAME, LABEL, TOOLTIP}, DESCRIPTIVE_STATS #undef DS }; static char * generate_syntax (const PsppireDialogAction *act) { PsppireDialogActionDescriptives *scd = PSPPIRE_DIALOG_ACTION_DESCRIPTIVES (act); gchar *text; GString *string; GtkTreeIter iter; unsigned int selected; size_t i; bool listwise, include; bool ok; string = g_string_new ("DESCRIPTIVES"); g_string_append (string, "\n /VARIABLES="); psppire_var_view_append_names (PSPPIRE_VAR_VIEW (scd->stat_vars), 0, string); listwise = gtk_toggle_button_get_active (scd->exclude_missing_listwise); include = gtk_toggle_button_get_active (scd->include_user_missing); if (listwise || include) { g_string_append (string, "\n /MISSING="); if (listwise) { g_string_append (string, "LISTWISE"); if (include) g_string_append (string, " "); } if (include) g_string_append (string, "INCLUDE"); } selected = 0; for (i = 0, ok = gtk_tree_model_get_iter_first (scd->stats, &iter); ok; i++, ok = gtk_tree_model_iter_next (scd->stats, &iter)) { gboolean toggled; gtk_tree_model_get (scd->stats, &iter, CHECKBOX_COLUMN_SELECTED, &toggled, -1); if (toggled) selected |= 1u << i; } if (selected != B_DS_DEFAULT) { g_string_append (string, "\n /STATISTICS="); if (selected == B_DS_ALL) g_string_append (string, "ALL"); else if (selected == 0) g_string_append (string, "NONE"); else { int n = 0; if ((selected & B_DS_DEFAULT) == B_DS_DEFAULT) { g_string_append (string, "DEFAULT"); selected &= ~B_DS_DEFAULT; n++; } for (i = 0; i < N_DESCRIPTIVE_STATS; i++) if (selected & (1u << i)) { if (n++) g_string_append (string, " "); g_string_append (string, stats[i].name); } } } if (gtk_toggle_button_get_active (scd->save_z_scores)) g_string_append (string, "\n /SAVE"); g_string_append (string, "."); if (gtk_toggle_button_get_active (scd->save_z_scores)) g_string_append (string, "\nEXECUTE."); text = string->str; g_string_free (string, FALSE); return text; } static gboolean dialog_state_valid (gpointer data) { PsppireDialogActionDescriptives *dd = data; GtkTreeModel *vars = gtk_tree_view_get_model (dd->stat_vars); GtkTreeIter notused; return gtk_tree_model_get_iter_first (vars, ¬used); } static void dialog_refresh (PsppireDialogAction *scd_) { PsppireDialogActionDescriptives *scd = PSPPIRE_DIALOG_ACTION_DESCRIPTIVES (scd_); GtkTreeModel *liststore; GtkTreeIter iter; size_t i; bool ok; liststore = gtk_tree_view_get_model (scd->stat_vars); gtk_list_store_clear (GTK_LIST_STORE (liststore)); for (i = 0, ok = gtk_tree_model_get_iter_first (scd->stats, &iter); ok; i++, ok = gtk_tree_model_iter_next (scd->stats, &iter)) gtk_list_store_set (GTK_LIST_STORE (scd->stats), &iter, CHECKBOX_COLUMN_SELECTED, (B_DS_DEFAULT & (1u << i)) ? true : false, -1); gtk_toggle_button_set_active (scd->exclude_missing_listwise, false); gtk_toggle_button_set_active (scd->include_user_missing, false); gtk_toggle_button_set_active (scd->save_z_scores, false); } static GtkBuilder * psppire_dialog_action_descriptives_activate (PsppireDialogAction *a, GVariant *param) { PsppireDialogAction *pda = PSPPIRE_DIALOG_ACTION (a); PsppireDialogActionDescriptives *act = PSPPIRE_DIALOG_ACTION_DESCRIPTIVES (a); GtkBuilder *xml = builder_new ("descriptives.ui"); GtkWidget *stats_treeview = get_widget_assert (xml, "statistics"); psppire_checkbox_treeview_populate (PSPPIRE_CHECKBOX_TREEVIEW (stats_treeview), B_DS_DEFAULT, N_DESCRIPTIVE_STATS, stats); act->stats = gtk_tree_view_get_model (GTK_TREE_VIEW (stats_treeview)); pda->dialog = get_widget_assert (xml, "descriptives-dialog"); pda->source = get_widget_assert (xml, "all-variables"); act->variables = get_widget_assert (xml, "stat-variables"); g_object_set (pda->source, "predicate", var_is_numeric, NULL); act->stat_vars = GTK_TREE_VIEW (act->variables); act->include_user_missing = GTK_TOGGLE_BUTTON (get_widget_assert (xml, "include_user_missing")); act->exclude_missing_listwise = GTK_TOGGLE_BUTTON (get_widget_assert (xml, "exclude_missing_listwise")); act->save_z_scores = GTK_TOGGLE_BUTTON (get_widget_assert (xml, "save_z_scores")); psppire_dialog_action_set_valid_predicate (pda, dialog_state_valid); psppire_dialog_action_set_refresh (pda, dialog_refresh); return xml; } static void psppire_dialog_action_descriptives_class_init (PsppireDialogActionDescriptivesClass *class) { PSPPIRE_DIALOG_ACTION_CLASS (class)->initial_activate = psppire_dialog_action_descriptives_activate; PSPPIRE_DIALOG_ACTION_CLASS (class)->generate_syntax = generate_syntax; } static void psppire_dialog_action_descriptives_init (PsppireDialogActionDescriptives *act) { }