Fix some Org errors revealed by `make check-declare'.
[emacs.git] / nt / addpm.c
blob4fcebe2ca14cabc77af22f8faa01b733ac7bb17c
1 /* Add entries to the GNU Emacs Program Manager folder.
2 Copyright (C) 1995, 2001, 2002, 2003, 2004, 2005, 2006, 2007,
3 2008, 2009, 2010 Free Software Foundation, Inc.
5 This file is part of GNU Emacs.
7 GNU Emacs is free software: you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation, either version 3 of the License, or
10 (at your option) any later version.
12 GNU Emacs is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>. */
20 /****************************************************************************
22 * Program: addpm (adds emacs to the Windows program manager)
24 * Usage:
25 * argv[1] = install path for emacs
27 * argv[2] used to be an optional argument for setting the icon.
28 * But now Emacs has a professional looking icon of its own.
29 * If users really want to change it, they can go into the settings of
30 * the shortcut that is created and do it there.
33 /* Use parts of shell API that were introduced by the merge of IE4
34 into the desktop shell. If Windows 95 or NT4 users do not have IE4
35 installed, then the DDE fallback for creating icons the Windows 3.1
36 progman way will be used instead, but that is prone to lockups
37 caused by other applications not servicing their message queues. */
38 #define _WIN32_IE 0x400
39 /* Request C Object macros for COM interfaces. */
40 #define COBJMACROS 1
42 #include <windows.h>
43 #include <shlobj.h>
44 #include <ddeml.h>
45 #include <stdlib.h>
46 #include <stdio.h>
47 #include <malloc.h>
49 HDDEDATA CALLBACK
50 DdeCallback (UINT uType, UINT uFmt, HCONV hconv,
51 HSZ hsz1, HSZ hsz2, HDDEDATA hdata,
52 DWORD dwData1, DWORD dwData2)
54 return ((HDDEDATA) NULL);
57 #define DdeCommand(str) \
58 DdeClientTransaction (str, strlen (str)+1, conversation, (HSZ)NULL, \
59 CF_TEXT, XTYP_EXECUTE, 30000, NULL)
61 #define REG_ROOT "SOFTWARE\\GNU\\Emacs"
62 #define REG_GTK "SOFTWARE\\GTK\\2.0"
63 #define REG_APP_PATH \
64 "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\App Paths\\emacs.exe"
65 #define REG_RUNEMACS_PATH \
66 "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\App Paths\\runemacs.exe"
68 static struct entry
70 char *name;
71 char *value;
73 env_vars[] =
75 {"emacs_dir", NULL},
76 {"EMACSLOADPATH", "%emacs_dir%/site-lisp;%emacs_dir%/../site-lisp;%emacs_dir%/lisp;%emacs_dir%/leim"},
77 {"SHELL", "%emacs_dir%/bin/cmdproxy.exe"},
78 {"EMACSDATA", "%emacs_dir%/etc"},
79 {"EMACSPATH", "%emacs_dir%/bin"},
80 /* We no longer set INFOPATH because Info-default-directory-list
81 is then ignored. */
82 /* {"INFOPATH", "%emacs_dir%/info"}, */
83 {"EMACSDOC", "%emacs_dir%/etc"},
84 {"TERM", "cmd"}
87 BOOL
88 add_registry (char *path)
90 HKEY hrootkey = NULL;
91 int i;
92 BOOL ok = TRUE;
93 DWORD size;
95 /* Record the location of Emacs to the App Paths key if we have
96 sufficient permissions to do so. This helps Windows find emacs quickly
97 if the user types emacs.exe in the "Run Program" dialog without having
98 emacs on their path. Some applications also use the same registry key
99 to discover the installation directory for programs they are looking for.
100 Multiple installations cannot be handled by this method, but it does not
101 affect the general operation of other installations of Emacs, and we
102 are blindly overwriting the Start Menu entries already.
104 if (RegCreateKeyEx (HKEY_LOCAL_MACHINE, REG_APP_PATH, 0, "",
105 REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL,
106 &hrootkey, NULL) == ERROR_SUCCESS)
108 int len;
109 char *emacs_path;
110 HKEY gtk_key = NULL;
112 len = strlen (path) + 15; /* \bin\emacs.exe + terminator. */
113 emacs_path = (char *) alloca (len);
114 sprintf (emacs_path, "%s\\bin\\emacs.exe", path);
116 RegSetValueEx (hrootkey, NULL, 0, REG_SZ, emacs_path, len);
118 /* Look for a GTK installation. If found, add it to the library search
119 path for Emacs so that the image libraries it provides are available
120 to Emacs regardless of whether it is in the path or not. */
121 if (RegOpenKeyEx (HKEY_LOCAL_MACHINE, REG_GTK, REG_OPTION_NON_VOLATILE,
122 KEY_READ, &gtk_key) == ERROR_SUCCESS)
124 if (RegQueryValueEx (gtk_key, "DllPath", NULL, NULL,
125 NULL, &size) == ERROR_SUCCESS)
127 char *gtk_path = (char *) alloca (size);
128 if (RegQueryValueEx (gtk_key, "DllPath", NULL, NULL,
129 gtk_path, &size) == ERROR_SUCCESS)
131 /* Make sure the emacs bin directory continues to be searched
132 first by including it as well. */
133 char *dll_paths;
134 HKEY runemacs_key = NULL;
135 len = strlen (path) + 5 + size;
136 dll_paths = (char *) alloca (size + strlen (path) + 1);
137 sprintf (dll_paths, "%s\\bin;%s", path, gtk_path);
138 RegSetValueEx (hrootkey, "Path", 0, REG_SZ, dll_paths, len);
140 /* Set the same path for runemacs.exe, as the Explorer shell
141 looks this up, so the above does not take effect when
142 emacs.exe is spawned from runemacs.exe. */
143 if (RegCreateKeyEx (HKEY_LOCAL_MACHINE, REG_RUNEMACS_PATH,
144 0, "", REG_OPTION_NON_VOLATILE,
145 KEY_WRITE, NULL, &runemacs_key, NULL)
146 == ERROR_SUCCESS)
148 RegSetValueEx (runemacs_key, "Path", 0, REG_SZ,
149 dll_paths, len);
151 RegCloseKey (runemacs_key);
155 RegCloseKey (gtk_key);
157 RegCloseKey (hrootkey);
160 /* Previous versions relied on registry settings, but we do not need
161 them any more. If registry settings are installed from a previous
162 version, replace them to ensure they are the current settings.
163 Otherwise, do nothing. */
165 /* Check both the current user and the local machine to see if we
166 have any resources. */
168 if (RegOpenKeyEx (HKEY_LOCAL_MACHINE, REG_ROOT,
169 REG_OPTION_NON_VOLATILE,
170 KEY_WRITE, &hrootkey) != ERROR_SUCCESS
171 && RegOpenKeyEx (HKEY_CURRENT_USER, REG_ROOT,
172 REG_OPTION_NON_VOLATILE,
173 KEY_WRITE, &hrootkey) != ERROR_SUCCESS)
175 return FALSE;
178 for (i = 0; i < (sizeof (env_vars) / sizeof (env_vars[0])); i++)
180 char * value = env_vars[i].value ? env_vars[i].value : path;
182 if (RegSetValueEx (hrootkey, env_vars[i].name,
183 0, REG_EXPAND_SZ,
184 value, lstrlen (value) + 1) != ERROR_SUCCESS)
185 ok = FALSE;
188 RegCloseKey (hrootkey);
190 return (ok);
194 main (int argc, char *argv[])
196 char start_folder[MAX_PATH + 1];
197 int shortcuts_created = 0;
198 int com_available = 1;
199 char modname[MAX_PATH];
200 char *prog_name;
201 char *emacs_path;
202 char *p;
203 int quiet = 0;
204 HRESULT result;
205 IShellLinkA *shortcut;
207 /* If no args specified, use our location to set emacs_path. */
209 if (argc > 1
210 && (argv[1][0] == '/' || argv[1][0] == '-')
211 && argv[1][1] == 'q')
213 quiet = 1;
214 --argc;
215 ++argv;
218 if (argc > 1)
219 emacs_path = argv[1];
220 else
222 if (!GetModuleFileName (NULL, modname, MAX_PATH) ||
223 (p = strrchr (modname, '\\')) == NULL)
225 fprintf (stderr, "fatal error");
226 exit (1);
228 *p = 0;
230 /* Set emacs_path to emacs_dir if we are in "%emacs_dir%\bin". */
231 if ((p = strrchr (modname, '\\')) && stricmp (p, "\\bin") == 0)
233 *p = 0;
234 emacs_path = modname;
236 else
238 fprintf (stderr, "usage: addpm emacs_path\n");
239 exit (1);
242 /* Tell user what we are going to do. */
243 if (!quiet)
245 int result;
247 char msg[ MAX_PATH ];
248 sprintf (msg, "Install Emacs at %s?\n", emacs_path);
249 result = MessageBox (NULL, msg, "Install Emacs",
250 MB_OKCANCEL | MB_ICONQUESTION);
251 if (result != IDOK)
253 fprintf (stderr, "Install cancelled\n");
254 exit (1);
259 add_registry (emacs_path);
260 prog_name = "runemacs.exe";
262 /* Try to install globally. */
264 if (!SUCCEEDED (CoInitialize (NULL))
265 || !SUCCEEDED (CoCreateInstance (&CLSID_ShellLink, NULL,
266 CLSCTX_INPROC_SERVER, &IID_IShellLinkA,
267 (void **) &shortcut)))
269 com_available = 0;
272 if (com_available
273 && SHGetSpecialFolderPath (NULL, start_folder, CSIDL_COMMON_PROGRAMS, 0))
275 if (strlen (start_folder) < (MAX_PATH - 20))
277 BOOL retval;
279 strcat (start_folder, "\\Gnu Emacs");
280 if (CreateDirectory (start_folder, NULL)
281 || GetLastError () == ERROR_ALREADY_EXISTS)
283 char full_emacs_path[MAX_PATH + 1];
284 IPersistFile *lnk;
285 strcat (start_folder, "\\Emacs.lnk");
286 sprintf (full_emacs_path, "%s\\bin\\%s", emacs_path, prog_name);
287 IShellLinkA_SetPath (shortcut, full_emacs_path);
288 IShellLinkA_SetDescription (shortcut, "GNU Emacs");
289 result = IShellLinkA_QueryInterface (shortcut, &IID_IPersistFile,
290 (void **) &lnk);
291 if (SUCCEEDED (result))
293 wchar_t unicode_path[MAX_PATH];
294 MultiByteToWideChar (CP_ACP, 0, start_folder, -1,
295 unicode_path, MAX_PATH);
296 if (SUCCEEDED (IPersistFile_Save (lnk, unicode_path, TRUE)))
297 shortcuts_created = 1;
298 IPersistFile_Release (lnk);
304 if (!shortcuts_created && com_available
305 && SHGetSpecialFolderPath (NULL, start_folder, CSIDL_PROGRAMS, 0))
307 /* Ensure there is enough room for "...\GNU Emacs\Emacs.lnk". */
308 if (strlen (start_folder) < (MAX_PATH - 20))
310 BOOL retval;
312 strcat (start_folder, "\\Gnu Emacs");
313 if (CreateDirectory (start_folder, NULL)
314 || GetLastError () == ERROR_ALREADY_EXISTS)
316 char full_emacs_path[MAX_PATH + 1];
317 IPersistFile *lnk;
318 strcat (start_folder, "\\Emacs.lnk");
319 sprintf (full_emacs_path, "%s\\bin\\%s", emacs_path, prog_name);
320 IShellLinkA_SetPath (shortcut, full_emacs_path);
321 IShellLinkA_SetDescription (shortcut, "GNU Emacs");
322 result = IShellLinkA_QueryInterface (shortcut, &IID_IPersistFile,
323 (void **) &lnk);
324 if (SUCCEEDED (result))
326 wchar_t unicode_path[MAX_PATH];
327 MultiByteToWideChar (CP_ACP, 0, start_folder, -1,
328 unicode_path, MAX_PATH);
329 if (SUCCEEDED (IPersistFile_Save (lnk, unicode_path, TRUE)))
330 shortcuts_created = 1;
331 IPersistFile_Release (lnk);
338 if (com_available)
339 IShellLinkA_Release (shortcut);
341 /* Need to call uninitialize, even if ComInitialize failed. */
342 CoUninitialize ();
344 /* Fallback on old DDE method if the above failed. */
345 if (!shortcuts_created)
347 DWORD dde = 0;
348 HCONV conversation;
349 HSZ progman;
350 char add_item[MAX_PATH*2 + 100];
352 DdeInitialize (&dde, (PFNCALLBACK) DdeCallback, APPCMD_CLIENTONLY, 0);
353 progman = DdeCreateStringHandle (dde, "PROGMAN", CP_WINANSI);
354 conversation = DdeConnect (dde, progman, progman, NULL);
355 if (conversation)
357 DdeCommand ("[CreateGroup (\"Gnu Emacs\")]");
358 DdeCommand ("[ReplaceItem (Emacs)]");
359 sprintf (add_item, "[AddItem (\"%s\\bin\\%s\", Emacs)]",
360 emacs_path, prog_name);
361 DdeCommand (add_item);
363 DdeDisconnect (conversation);
366 DdeFreeStringHandle (dde, progman);
367 DdeUninitialize (dde);
370 return 0;
373 /* arch-tag: f923609d-b781-4ef4-abce-ca0da29cbbf0
374 (do not change this comment) */