QUICK CLUSTER: Use fixed format for cluster centers.
[pspp.git] / src / ui / gui / help-menu.c
blobe6dc4b5ebfed5a22f9511efcc19d9e978cacdd5f
1 /* PSPPIRE - a graphical user interface for PSPP.
2 Copyright (C) 2006, 2007, 2010, 2011, 2012, 2013, 2015, 2016,
3 2021 Free Software Foundation
5 This program is free software: you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation, either version 3 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>. */
19 #include <config.h>
21 #include <gtk/gtk.h>
23 #include "libpspp/cast.h"
24 #include "libpspp/copyleft.h"
25 #include "libpspp/message.h"
26 #include "libpspp/version.h"
27 #include "ui/gui/executor.h"
28 #include "ui/gui/help-menu.h"
29 #include "ui/gui/psppire-data-window.h"
31 #include "gl/configmake.h"
32 #include "gl/relocatable.h"
34 #include <gettext.h>
35 #define _(msgid) gettext (msgid)
36 #define N_(msgid) msgid
38 /* Try to open html documentation uri via the default
39 browser on the operating system */
40 #ifdef __APPLE__
41 #define HTMLOPENAPP "open"
42 #elif _WIN32
43 #define HTMLOPENAPP "wscript"
44 #else
45 #define HTMLOPENAPP "xdg-open"
46 #endif
48 static const gchar *artists[] = { "Bastián Díaz", "Hugo Alejandro", NULL};
50 /* Opening the htmluri in windows via cmd /start uri opens
51 the windows command shell for a moment. The alternative is
52 to start a script via wscript. This will not be visible*/
53 #ifdef _WIN32
54 static gboolean
55 open_windows_help (const gchar *helpuri, GError **err)
57 SHELLEXECUTEINFOA info;
58 memset (&info, 0, sizeof (info));
60 info.cbSize = sizeof (info);
61 info.fMask = SEE_MASK_FLAG_NO_UI;
62 info.lpVerb = "open";
63 info.lpFile = helpuri;
64 info.nShow = SW_SHOWNORMAL;
66 BOOL ret = ShellExecuteExA (&info);
68 if (ret)
69 return TRUE;
71 /* Contrary to what the microsoft documentation indicates, ShellExecuteExA does
72 not seem to setLastError. So we have to deal with errors ourselves here. */
73 const char *msg = 0;
74 switch (GPOINTER_TO_INT (info.hInstApp))
76 case SE_ERR_FNF:
77 msg = "File not found";
78 break;
79 case SE_ERR_PNF:
80 msg = "Path not found";
81 break;
82 case SE_ERR_ACCESSDENIED:
83 msg = "Access denied";
84 break;
85 case SE_ERR_OOM:
86 msg = "Out of memory";
87 break;
88 case SE_ERR_DLLNOTFOUND:
89 msg = "Dynamic-link library not found";
90 break;
91 case SE_ERR_SHARE:
92 msg = "Cannot share an open file";
93 break;
94 case SE_ERR_ASSOCINCOMPLETE:
95 msg = "File association information not complete";
96 break;
97 case SE_ERR_DDETIMEOUT:
98 msg = "DDE operation timed out";
99 break;
100 case SE_ERR_DDEFAIL:
101 msg = "DDE operation failed";
102 break;
103 case SE_ERR_DDEBUSY:
104 msg = "DDE operation is busy";
105 break;
106 case SE_ERR_NOASSOC:
107 msg = "File association not available";
108 break;
109 default:
110 msg = "Unknown error";
111 break;
114 *err = g_error_new_literal (g_quark_from_static_string ("pspp-help-error"),
116 msg);
118 return FALSE;
121 static gboolean
122 on_activate_link (GtkAboutDialog *label,
123 gchar *uri,
124 gpointer user_data)
126 return open_windows_help (uri, NULL);
128 #endif
130 static void
131 about_system_info (GtkMenuItem *mmm, GtkWindow *parent)
133 execute_const_syntax_string (psppire_default_data_window (), "SHOW SYSTEM.");
136 static void
137 about_new (GtkMenuItem *mmm, GtkWindow *parent)
139 GtkWidget *about = gtk_about_dialog_new ();
141 #ifdef _WIN32
142 /* The default handler for Windows doesn't appear to work. */
143 g_signal_connect (about, "activate-link", G_CALLBACK (on_activate_link), parent);
144 #endif
146 gtk_about_dialog_set_logo_icon_name (GTK_ABOUT_DIALOG (about), "org.gnu.pspp");
148 gtk_window_set_icon_name (GTK_WINDOW (about), "org.gnu.pspp");
150 gtk_about_dialog_set_website (GTK_ABOUT_DIALOG (about), PACKAGE_URL);
152 gtk_about_dialog_set_version (GTK_ABOUT_DIALOG (about),
153 announced_version);
155 gtk_about_dialog_set_authors (GTK_ABOUT_DIALOG (about),
156 (const gchar **) authors);
158 gtk_about_dialog_set_artists (GTK_ABOUT_DIALOG (about),
159 artists);
161 gtk_about_dialog_set_license (GTK_ABOUT_DIALOG (about),
162 copyleft);
164 gtk_about_dialog_set_comments (GTK_ABOUT_DIALOG (about),
165 _("A program for the analysis of sampled data"));
167 gtk_about_dialog_set_copyright (GTK_ABOUT_DIALOG (about),
168 "Free Software Foundation");
170 gtk_about_dialog_set_translator_credits
172 GTK_ABOUT_DIALOG (about),
173 /* TRANSLATORS: Do not translate this string. Instead, put the names of the people
174 who have helped in the translation. */
175 _("translator-credits")
178 gtk_window_set_transient_for (GTK_WINDOW (about), parent);
180 gtk_window_set_modal (GTK_WINDOW (about), TRUE);
182 gtk_dialog_run (GTK_DIALOG (about));
184 gtk_widget_hide (about);
189 /* Open the manual at PAGE with the following priorities
190 First: local yelp help system
191 Second: browser with local html doc dir in path pspp.html/<helppage>.html
192 Third: browers with Internet html help at gnu.org */
193 void
194 online_help (const char *page)
196 GError *htmlerr = NULL;
197 gchar *htmluri = NULL;
199 char *htmlfilename;
200 if (page == NULL)
201 htmlfilename = xstrdup ("index.html");
202 else
204 gchar **tokens = NULL;
205 const int maxtokens = 5;
206 int idx ;
207 /* The page will be translated to the htmlfilename
208 page htmlfilename
209 GRAPH#SCATTERPLOT SCATTERPLOT.html
210 QUICK-CLUSTER QUICK-CLUSTER.html
211 which is valid for the multiple page html doc*/
212 tokens = g_strsplit (page, "#", maxtokens);
213 for (idx = 0; idx < maxtokens && tokens[idx]; idx++)
215 htmlfilename = xasprintf ("%s.html", tokens[idx-1]);
216 g_strfreev (tokens);
218 /* Hint: pspp.html is a directory...*/
219 char *htmldir = relocate_clone (DOCDIR "/pspp.html");
220 char *htmlfullname = xasprintf ("%s/%s", htmldir, htmlfilename);
221 if (g_file_test (htmldir, G_FILE_TEST_IS_DIR))
223 GError *urierr = NULL;
224 htmluri = g_filename_to_uri (htmlfullname,NULL, &urierr);
225 if (!htmluri)
227 msg (ME, _("Help path conversion error: %s"), urierr->message);
228 htmluri = htmlfullname;
230 g_clear_error (&urierr);
232 else
233 htmluri = g_strdup_printf (PACKAGE_URL "manual/html_node/%s",
234 htmlfilename);
235 free (htmlfullname);
236 free (htmldir);
237 free (htmlfilename);
239 #ifdef _WIN32
240 bool ok = open_windows_help (htmluri, &htmlerr);
241 #else
242 gchar *htmlargv[3] = {CONST_CAST (char *, HTMLOPENAPP), htmluri, 0};
243 bool ok = g_spawn_async (NULL, htmlargv,
244 NULL, G_SPAWN_SEARCH_PATH,
245 NULL, NULL, NULL, &htmlerr);
246 #endif
247 if (!ok)
249 msg (ME, _("Cannot open via html: %s "
250 "with uri: %s "
251 "The PSSP manual is also available at %s"),
252 htmlerr->message,
253 htmluri,
254 PACKAGE_URL "documentation.html");
257 g_free (htmluri);
258 g_clear_error (&htmlerr);
261 static void
262 reference_manual (GtkMenuItem *menu, gpointer data)
264 online_help (NULL);
267 GtkWidget *
268 create_help_menu (GtkWindow *toplevel)
270 GtkWidget *menuitem = gtk_menu_item_new_with_mnemonic (_("_Help"));
271 GtkWidget *menu = gtk_menu_new ();
273 GtkWidget *help_about = gtk_menu_item_new_with_mnemonic (_("_About"));
274 GtkWidget *help_system_info = gtk_menu_item_new_with_mnemonic (_("_System Information"));
275 GtkWidget *help_ref = gtk_menu_item_new_with_mnemonic (_("_Reference Manual"));
277 GtkAccelGroup *accel_group = gtk_accel_group_new ();
279 gtk_window_add_accel_group (toplevel, accel_group);
281 gtk_widget_add_accelerator (help_ref,
282 "activate", accel_group,
283 GDK_KEY_F1, 0,
284 GTK_ACCEL_VISIBLE);
286 gtk_menu_attach (GTK_MENU (menu), help_ref, 0, 1, 0, 1);
287 gtk_menu_attach (GTK_MENU (menu), help_system_info, 0, 1, 1, 2);
288 gtk_menu_attach (GTK_MENU (menu), help_about, 0, 1, 2, 3);
290 g_signal_connect (help_about, "activate", G_CALLBACK (about_new), toplevel);
291 g_signal_connect (help_system_info, "activate", G_CALLBACK (about_system_info), toplevel);
292 g_signal_connect (help_ref, "activate", G_CALLBACK (reference_manual), NULL);
294 g_object_set (menuitem, "submenu", menu, NULL);
296 gtk_widget_show_all (menuitem);
298 g_object_unref (accel_group);
300 return menuitem;