| /* vi:set ts=8 sts=4 sw=4 noet: |
| * |
| * VIM - Vi IMproved by Bram Moolenaar |
| * |
| * Do ":help uganda" in Vim to read copying and usage conditions. |
| * Do ":help credits" in Vim to see a list of people who contributed. |
| * See README.txt for an overview of the Vim source code. |
| */ |
| |
| /* |
| * dosinst.c: Install program for Vim on MS-DOS and MS-Windows |
| * |
| * Compile with Make_mvc.mak, Make_cyg.mak or Make_ming.mak. |
| */ |
| |
| /* |
| * Include common code for dosinst.c and uninstall.c. |
| */ |
| #define DOSINST |
| #include "dosinst.h" |
| #include <io.h> |
| |
| #define GVIMEXT64_PATH "GvimExt64\\gvimext.dll" |
| #define GVIMEXT32_PATH "GvimExt32\\gvimext.dll" |
| |
| // Macro to do an error check I was typing over and over |
| #define CHECK_REG_ERROR(code) \ |
| do { \ |
| if (code != ERROR_SUCCESS) \ |
| { \ |
| printf("%ld error number: %ld\n", (long)__LINE__, (long)code); \ |
| return 1; \ |
| } \ |
| } while (0) |
| |
| int has_vim = 0; // installable vim.exe exists |
| int has_gvim = 0; // installable gvim.exe exists |
| |
| char oldvimrc[BUFSIZE]; // name of existing vimrc file |
| char vimrc[BUFSIZE]; // name of vimrc file to create |
| |
| char *default_bat_dir = NULL; // when not NULL, use this as the default |
| // directory to write .bat files in |
| char *default_vim_dir = NULL; // when not NULL, use this as the default |
| // install dir for NSIS |
| |
| /* |
| * Structure used for each choice the user can make. |
| */ |
| struct choice |
| { |
| int active; // non-zero when choice is active |
| char *text; // text displayed for this choice |
| void (*changefunc)(int idx); // function to change this choice |
| int arg; // argument for function |
| void (*installfunc)(int idx); // function to install this choice |
| }; |
| |
| struct choice choices[30]; // choices the user can make |
| int choice_count = 0; // number of choices available |
| |
| #define TABLE_SIZE(s) (int)ARRAYSIZE(s) |
| |
| enum |
| { |
| compat_vi = 1, |
| compat_vim, |
| compat_some_enhancements, |
| compat_all_enhancements |
| }; |
| char *(compat_choices[]) = |
| { |
| "\nChoose the default way to run Vim:", |
| "Vi compatible", |
| "Vim default", |
| "with some Vim enhancements", |
| "with syntax highlighting and other features switched on", |
| }; |
| int compat_choice = (int)compat_all_enhancements; |
| char *compat_text = "- run Vim %s"; |
| |
| enum |
| { |
| remap_no = 1, |
| remap_win |
| }; |
| char *(remap_choices[]) = |
| { |
| "\nChoose:", |
| "Do not remap keys for Windows behavior", |
| "Remap a few keys for Windows behavior (CTRL-V, CTRL-C, CTRL-F, etc)", |
| }; |
| int remap_choice = (int)remap_no; |
| char *remap_text = "- %s"; |
| |
| enum |
| { |
| mouse_xterm = 1, |
| mouse_mswin, |
| mouse_default |
| }; |
| char *(mouse_choices[]) = |
| { |
| "\nChoose the way how Vim uses the mouse:", |
| "right button extends selection (the Unix way)", |
| "right button has a popup menu, left button starts select mode (the Windows way)", |
| "right button has a popup menu, left button starts visual mode", |
| }; |
| int mouse_choice = (int)mouse_default; |
| char *mouse_text = "- The mouse %s"; |
| |
| enum |
| { |
| vimfiles_dir_none = 1, |
| vimfiles_dir_vim, |
| vimfiles_dir_home |
| }; |
| static char *(vimfiles_dir_choices[]) = |
| { |
| "\nCreate plugin directories:", |
| "No", |
| "In the VIM directory", |
| "In your HOME directory", |
| }; |
| |
| // non-zero when selected to install the popup menu entry. |
| static int install_popup = 0; |
| |
| // non-zero when selected to install the "Open with" entry. |
| static int install_openwith = 0; |
| |
| // non-zero when need to add an uninstall entry in the registry |
| static int need_uninstall_entry = 0; |
| |
| /* |
| * Definitions of the directory name (under $VIM) of the vimfiles directory |
| * and its subdirectories: |
| */ |
| static char *(vimfiles_subdirs[]) = |
| { |
| "colors", |
| "compiler", |
| "doc", |
| "ftdetect", |
| "ftplugin", |
| "indent", |
| "keymap", |
| "plugin", |
| "syntax", |
| }; |
| |
| /* |
| * Obtain a choice from a table. |
| * First entry is a question, others are choices. |
| */ |
| static int |
| get_choice(char **table, int entries) |
| { |
| int answer; |
| int idx; |
| char dummy[100]; |
| |
| do |
| { |
| for (idx = 0; idx < entries; ++idx) |
| { |
| if (idx) |
| printf("%2d ", idx); |
| puts(table[idx]); |
| } |
| printf("Choice: "); |
| if (scanf("%d", &answer) != 1) |
| { |
| scanf("%99s", dummy); |
| answer = 0; |
| } |
| } |
| while (answer < 1 || answer >= entries); |
| |
| return answer; |
| } |
| |
| /* |
| * Check if the user unpacked the archives properly. |
| * Sets "runtimeidx". |
| */ |
| static void |
| check_unpack(void) |
| { |
| char buf[BUFSIZE]; |
| FILE *fd; |
| struct stat st; |
| |
| // check for presence of the correct version number in installdir[] |
| runtimeidx = strlen(installdir) - strlen(VIM_VERSION_NODOT); |
| if (runtimeidx <= 0 |
| || stricmp(installdir + runtimeidx, VIM_VERSION_NODOT) != 0 |
| || (installdir[runtimeidx - 1] != '/' |
| && installdir[runtimeidx - 1] != '\\')) |
| { |
| printf("ERROR: Install program not in directory \"%s\"\n", |
| VIM_VERSION_NODOT); |
| printf("This program can only work when it is located in its original directory\n"); |
| myexit(1); |
| } |
| |
| // check if filetype.vim is present, which means the runtime archive has |
| // been unpacked |
| sprintf(buf, "%s\\filetype.vim", installdir); |
| if (stat(buf, &st) < 0) |
| { |
| printf("ERROR: Cannot find filetype.vim in \"%s\"\n", installdir); |
| printf("It looks like you did not unpack the runtime archive.\n"); |
| printf("You must unpack the runtime archive \"vim%srt.zip\" before installing.\n", |
| VIM_VERSION_NODOT + 3); |
| myexit(1); |
| } |
| |
| // Check if vim.exe or gvim.exe is in the current directory. |
| if ((fd = fopen("gvim.exe", "r")) != NULL) |
| { |
| fclose(fd); |
| has_gvim = 1; |
| } |
| if ((fd = fopen("vim.exe", "r")) != NULL) |
| { |
| fclose(fd); |
| has_vim = 1; |
| } |
| if (!has_gvim && !has_vim) |
| { |
| printf("ERROR: Cannot find any Vim executables in \"%s\"\n\n", |
| installdir); |
| myexit(1); |
| } |
| } |
| |
| /* |
| * Compare paths "p[plen]" to "q[qlen]". Return 0 if they match. |
| * Ignores case and differences between '/' and '\'. |
| * "plen" and "qlen" can be negative, strlen() is used then. |
| */ |
| static int |
| pathcmp(char *p, int plen, char *q, int qlen) |
| { |
| int i; |
| |
| if (plen < 0) |
| plen = strlen(p); |
| if (qlen < 0) |
| qlen = strlen(q); |
| for (i = 0; ; ++i) |
| { |
| // End of "p": check if "q" also ends or just has a slash. |
| if (i == plen) |
| { |
| if (i == qlen) // match |
| return 0; |
| if (i == qlen - 1 && (q[i] == '\\' || q[i] == '/')) |
| return 0; // match with trailing slash |
| return 1; // no match |
| } |
| |
| // End of "q": check if "p" also ends or just has a slash. |
| if (i == qlen) |
| { |
| if (i == plen) // match |
| return 0; |
| if (i == plen - 1 && (p[i] == '\\' || p[i] == '/')) |
| return 0; // match with trailing slash |
| return 1; // no match |
| } |
| |
| if (!(mytoupper(p[i]) == mytoupper(q[i]) |
| || ((p[i] == '/' || p[i] == '\\') |
| && (q[i] == '/' || q[i] == '\\')))) |
| return 1; // no match |
| } |
| //NOTREACHED |
| } |
| |
| /* |
| * If the executable "**destination" is in the install directory, find another |
| * one in $PATH. |
| * On input "**destination" is the path of an executable in allocated memory |
| * (or NULL). |
| * "*destination" is set to NULL or the location of the file. |
| */ |
| static void |
| findoldfile(char **destination) |
| { |
| char *bp = *destination; |
| size_t indir_l = strlen(installdir); |
| char *cp; |
| char *tmpname; |
| char *farname; |
| |
| /* |
| * No action needed if exe not found or not in this directory. |
| */ |
| if (bp == NULL || strnicmp(bp, installdir, indir_l) != 0) |
| return; |
| cp = bp + indir_l; |
| if (strchr("/\\", *cp++) == NULL |
| || strchr(cp, '\\') != NULL |
| || strchr(cp, '/') != NULL) |
| return; |
| |
| tmpname = alloc(strlen(cp) + 1); |
| strcpy(tmpname, cp); |
| tmpname[strlen(tmpname) - 1] = 'x'; // .exe -> .exx |
| |
| if (access(tmpname, 0) == 0) |
| { |
| printf("\nERROR: %s and %s clash. Remove or rename %s.\n", |
| tmpname, cp, tmpname); |
| myexit(1); |
| } |
| |
| if (rename(cp, tmpname) != 0) |
| { |
| printf("\nERROR: failed to rename %s to %s: %s\n", |
| cp, tmpname, strerror(0)); |
| myexit(1); |
| } |
| |
| farname = searchpath_save(cp); |
| |
| if (rename(tmpname, cp) != 0) |
| { |
| printf("\nERROR: failed to rename %s back to %s: %s\n", |
| tmpname, cp, strerror(0)); |
| myexit(1); |
| } |
| |
| free(*destination); |
| free(tmpname); |
| *destination = farname; |
| } |
| |
| /* |
| * Check if there is a vim.[exe|bat|, gvim.[exe|bat|, etc. in the path. |
| * When "check_bat_only" is TRUE, only find "default_bat_dir". |
| */ |
| static void |
| find_bat_exe(int check_bat_only) |
| { |
| int i; |
| |
| // avoid looking in the "installdir" by chdir to system root |
| mch_chdir(sysdrive); |
| mch_chdir("\\"); |
| |
| for (i = 1; i < TARGET_COUNT; ++i) |
| { |
| targets[i].oldbat = searchpath_save(targets[i].batname); |
| if (!check_bat_only) |
| targets[i].oldexe = searchpath_save(targets[i].exename); |
| |
| if (default_bat_dir == NULL && targets[i].oldbat != NULL) |
| { |
| default_bat_dir = alloc(strlen(targets[i].oldbat) + 1); |
| strcpy(default_bat_dir, targets[i].oldbat); |
| remove_tail(default_bat_dir); |
| } |
| if (check_bat_only && targets[i].oldbat != NULL) |
| { |
| free(targets[i].oldbat); |
| targets[i].oldbat = NULL; |
| } |
| } |
| |
| mch_chdir(installdir); |
| } |
| |
| /* |
| * Get the value of $VIMRUNTIME or $VIM and write it in $TEMP/vimini.ini, so |
| * that NSIS can read it. |
| * When not set, use the directory of a previously installed Vim. |
| */ |
| static void |
| get_vim_env(void) |
| { |
| char *vim; |
| char buf[BUFSIZE]; |
| FILE *fd; |
| char fname[BUFSIZE]; |
| |
| // First get $VIMRUNTIME. If it's set, remove the tail. |
| vim = getenv("VIMRUNTIME"); |
| if (vim != NULL && *vim != 0 && strlen(vim) < sizeof(buf)) |
| { |
| strcpy(buf, vim); |
| remove_tail(buf); |
| vim = buf; |
| } |
| else |
| { |
| vim = getenv("VIM"); |
| if (vim == NULL || *vim == 0) |
| { |
| // Use the directory from an old uninstall entry. |
| if (default_vim_dir != NULL) |
| vim = default_vim_dir; |
| else |
| // Let NSIS know there is no default, it should use |
| // $PROGRAMFILES. |
| vim = ""; |
| } |
| } |
| |
| // NSIS also uses GetTempPath(), thus we should get the same directory |
| // name as where NSIS will look for vimini.ini. |
| GetTempPath(sizeof(fname) - 12, fname); |
| add_pathsep(fname); |
| strcat(fname, "vimini.ini"); |
| |
| fd = fopen(fname, "w"); |
| if (fd != NULL) |
| { |
| // Make it look like an .ini file, so that NSIS can read it with a |
| // ReadINIStr command. |
| fprintf(fd, "[vimini]\n"); |
| fprintf(fd, "dir=\"%s\"\n", vim); |
| fclose(fd); |
| } |
| else |
| { |
| printf("Failed to open %s\n", fname); |
| sleep(2); |
| } |
| } |
| |
| static int num_windows; |
| |
| /* |
| * Callback used for EnumWindows(): |
| * Count the window if the title looks like it is for the uninstaller. |
| */ |
| //ARGSUSED |
| static BOOL CALLBACK |
| window_cb(HWND hwnd, LPARAM lparam) |
| { |
| char title[256]; |
| |
| title[0] = 0; |
| GetWindowText(hwnd, title, 256); |
| if (strstr(title, "Vim ") != NULL && strstr(title, " Uninstall") != NULL) |
| ++num_windows; |
| return TRUE; |
| } |
| |
| /* |
| * Run the uninstaller silently. |
| */ |
| static int |
| run_silent_uninstall(char *uninst_exe) |
| { |
| char vimrt_dir[BUFSIZE]; |
| char temp_uninst[BUFSIZE]; |
| char temp_dir[MAX_PATH]; |
| char buf[BUFSIZE * 2 + 10]; |
| int i; |
| DWORD tick; |
| |
| strcpy(vimrt_dir, uninst_exe); |
| remove_tail(vimrt_dir); |
| |
| if (!GetTempPath(sizeof(temp_dir), temp_dir)) |
| return FAIL; |
| |
| // Copy the uninstaller to a temporary exe. |
| tick = GetTickCount(); |
| for (i = 0; ; i++) |
| { |
| sprintf(temp_uninst, "%s\\vimun%04X.exe", temp_dir, |
| (unsigned int)((i + tick) & 0xFFFF)); |
| if (CopyFile(uninst_exe, temp_uninst, TRUE)) |
| break; |
| if (GetLastError() != ERROR_FILE_EXISTS) |
| return FAIL; |
| if (i == 65535) |
| return FAIL; |
| } |
| |
| // Run the copied uninstaller silently. |
| if (strchr(temp_uninst, ' ') != NULL) |
| sprintf(buf, "\"%s\" /S _?=%s", temp_uninst, vimrt_dir); |
| else |
| sprintf(buf, "%s /S _?=%s", temp_uninst, vimrt_dir); |
| run_command(buf); |
| |
| DeleteFile(temp_uninst); |
| return OK; |
| } |
| |
| /* |
| * Check for already installed Vims. |
| * Return non-zero when found one. |
| */ |
| static int |
| uninstall_check(int skip_question) |
| { |
| HKEY key_handle; |
| HKEY uninstall_key_handle; |
| char *uninstall_key = "software\\Microsoft\\Windows\\CurrentVersion\\Uninstall"; |
| char subkey_name_buff[BUFSIZE]; |
| char temp_string_buffer[BUFSIZE-2]; |
| DWORD local_bufsize; |
| FILETIME temp_pfiletime; |
| DWORD key_index; |
| char input; |
| long code; |
| DWORD value_type; |
| DWORD orig_num_keys; |
| DWORD new_num_keys; |
| DWORD allow_silent; |
| int foundone = 0; |
| |
| code = RegOpenKeyEx(HKEY_LOCAL_MACHINE, uninstall_key, 0, |
| KEY_WOW64_64KEY | KEY_READ, &key_handle); |
| CHECK_REG_ERROR(code); |
| |
| key_index = 0; |
| while (TRUE) |
| { |
| local_bufsize = sizeof(subkey_name_buff); |
| if (RegEnumKeyEx(key_handle, key_index, subkey_name_buff, &local_bufsize, |
| NULL, NULL, NULL, &temp_pfiletime) == ERROR_NO_MORE_ITEMS) |
| break; |
| |
| if (strncmp("Vim", subkey_name_buff, 3) == 0) |
| { |
| // Open the key named Vim* |
| code = RegOpenKeyEx(key_handle, subkey_name_buff, 0, |
| KEY_WOW64_64KEY | KEY_READ, &uninstall_key_handle); |
| CHECK_REG_ERROR(code); |
| |
| // get the DisplayName out of it to show the user |
| local_bufsize = sizeof(temp_string_buffer); |
| code = RegQueryValueEx(uninstall_key_handle, "displayname", 0, |
| &value_type, (LPBYTE)temp_string_buffer, |
| &local_bufsize); |
| CHECK_REG_ERROR(code); |
| |
| allow_silent = 0; |
| if (skip_question) |
| { |
| DWORD varsize = sizeof(DWORD); |
| |
| RegQueryValueEx(uninstall_key_handle, "AllowSilent", 0, |
| &value_type, (LPBYTE)&allow_silent, |
| &varsize); |
| } |
| |
| foundone = 1; |
| printf("\n*********************************************************\n"); |
| printf("Vim Install found what looks like an existing Vim version.\n"); |
| printf("The name of the entry is:\n"); |
| printf("\n \"%s\"\n\n", temp_string_buffer); |
| |
| printf("Installing the new version will disable part of the existing version.\n"); |
| printf("(The batch files used in a console and the \"Edit with Vim\" entry in\n"); |
| printf("the popup menu will use the new version)\n"); |
| |
| if (skip_question) |
| printf("\nRunning uninstall program for \"%s\"\n", temp_string_buffer); |
| else |
| printf("\nDo you want to uninstall \"%s\" now?\n(y)es/(n)o) ", temp_string_buffer); |
| fflush(stdout); |
| |
| // get the UninstallString |
| local_bufsize = sizeof(temp_string_buffer); |
| code = RegQueryValueEx(uninstall_key_handle, "uninstallstring", 0, |
| &value_type, (LPBYTE)temp_string_buffer, &local_bufsize); |
| CHECK_REG_ERROR(code); |
| |
| // Remember the directory, it is used as the default for NSIS. |
| default_vim_dir = alloc(strlen(temp_string_buffer) + 1); |
| strcpy(default_vim_dir, temp_string_buffer); |
| remove_tail(default_vim_dir); |
| remove_tail(default_vim_dir); |
| |
| input = 'n'; |
| do |
| { |
| if (input != 'n') |
| printf("%c is an invalid reply. Please enter either 'y' or 'n'\n", input); |
| |
| if (skip_question) |
| input = 'y'; |
| else |
| { |
| rewind(stdin); |
| scanf("%c", &input); |
| } |
| switch (input) |
| { |
| case 'y': |
| case 'Y': |
| // save the number of uninstall keys so we can know if |
| // it changed |
| RegQueryInfoKey(key_handle, NULL, NULL, NULL, |
| &orig_num_keys, NULL, NULL, NULL, |
| NULL, NULL, NULL, NULL); |
| |
| // Find existing .bat files before deleting them. |
| find_bat_exe(TRUE); |
| |
| if (allow_silent) |
| { |
| if (run_silent_uninstall(temp_string_buffer) |
| == FAIL) |
| allow_silent = 0; // Retry with non silent. |
| } |
| if (!allow_silent) |
| { |
| // Execute the uninstall program. Put it in double |
| // quotes if there is an embedded space. |
| { |
| char buf[BUFSIZE]; |
| |
| if (strchr(temp_string_buffer, ' ') != NULL) |
| sprintf(buf, "\"%s\"", temp_string_buffer); |
| else |
| strcpy(buf, temp_string_buffer); |
| run_command(buf); |
| } |
| |
| // Count the number of windows with a title that |
| // match the installer, so that we can check when |
| // it's done. The uninstaller copies itself, |
| // executes the copy and exits, thus we can't wait |
| // for the process to finish. |
| sleep(1); // wait for uninstaller to start up |
| num_windows = 0; |
| EnumWindows(window_cb, 0); |
| if (num_windows == 0) |
| { |
| // Did not find the uninstaller, ask user to |
| // press Enter when done. Just in case. |
| printf("Press Enter when the uninstaller is finished\n"); |
| rewind(stdin); |
| (void)getchar(); |
| } |
| else |
| { |
| printf("Waiting for the uninstaller to finish (press CTRL-C to abort)."); |
| do |
| { |
| printf("."); |
| fflush(stdout); |
| sleep(1); // wait for the uninstaller to |
| // finish |
| num_windows = 0; |
| EnumWindows(window_cb, 0); |
| } while (num_windows > 0); |
| } |
| } |
| printf("\nDone!\n"); |
| |
| // Check if an uninstall reg key was deleted. |
| // if it was, we want to decrement key_index. |
| // if we don't do this, we will skip the key |
| // immediately after any key that we delete. |
| RegQueryInfoKey(key_handle, NULL, NULL, NULL, |
| &new_num_keys, NULL, NULL, NULL, |
| NULL, NULL, NULL, NULL); |
| if (new_num_keys < orig_num_keys) |
| key_index--; |
| |
| input = 'y'; |
| break; |
| |
| case 'n': |
| case 'N': |
| // Do not uninstall |
| input = 'n'; |
| break; |
| |
| default: // just drop through and redo the loop |
| break; |
| } |
| |
| } while (input != 'n' && input != 'y'); |
| |
| RegCloseKey(uninstall_key_handle); |
| } |
| |
| key_index++; |
| } |
| RegCloseKey(key_handle); |
| |
| return foundone; |
| } |
| |
| /* |
| * Find out information about the system. |
| */ |
| static void |
| inspect_system(void) |
| { |
| char *p; |
| char buf[BUFSIZE]; |
| FILE *fd; |
| int i; |
| int foundone; |
| |
| // This may take a little while, let the user know what we're doing. |
| printf("Inspecting system...\n"); |
| |
| /* |
| * If $VIM is set, check that it's pointing to our directory. |
| */ |
| p = getenv("VIM"); |
| if (p != NULL && pathcmp(p, -1, installdir, runtimeidx - 1) != 0) |
| { |
| printf("------------------------------------------------------\n"); |
| printf("$VIM is set to \"%s\".\n", p); |
| printf("This is different from where this version of Vim is:\n"); |
| strcpy(buf, installdir); |
| *(buf + runtimeidx - 1) = NUL; |
| printf("\"%s\"\n", buf); |
| printf("You must adjust or remove the setting of $VIM,\n"); |
| if (interactive) |
| { |
| printf("to be able to use this install program.\n"); |
| myexit(1); |
| } |
| printf("otherwise Vim WILL NOT WORK properly!\n"); |
| printf("------------------------------------------------------\n"); |
| } |
| |
| /* |
| * If $VIMRUNTIME is set, check that it's pointing to our runtime directory. |
| */ |
| p = getenv("VIMRUNTIME"); |
| if (p != NULL && pathcmp(p, -1, installdir, -1) != 0) |
| { |
| printf("------------------------------------------------------\n"); |
| printf("$VIMRUNTIME is set to \"%s\".\n", p); |
| printf("This is different from where this version of Vim is:\n"); |
| printf("\"%s\"\n", installdir); |
| printf("You must adjust or remove the setting of $VIMRUNTIME,\n"); |
| if (interactive) |
| { |
| printf("to be able to use this install program.\n"); |
| myexit(1); |
| } |
| printf("otherwise Vim WILL NOT WORK properly!\n"); |
| printf("------------------------------------------------------\n"); |
| } |
| |
| /* |
| * Check if there is a vim.[exe|bat|, gvim.[exe|bat|, etc. in the path. |
| */ |
| find_bat_exe(FALSE); |
| |
| /* |
| * A .exe in the install directory may be found anyway on Windows 2000. |
| * Check for this situation and find another executable if necessary. |
| * w.briscoe@ponl.com 2001-01-20 |
| */ |
| foundone = 0; |
| for (i = 1; i < TARGET_COUNT; ++i) |
| { |
| findoldfile(&(targets[i].oldexe)); |
| if (targets[i].oldexe != NULL) |
| foundone = 1; |
| } |
| |
| if (foundone) |
| { |
| printf("Warning: Found Vim executable(s) in your $PATH:\n"); |
| for (i = 1; i < TARGET_COUNT; ++i) |
| if (targets[i].oldexe != NULL) |
| printf("%s\n", targets[i].oldexe); |
| printf("It will be used instead of the version you are installing.\n"); |
| printf("Please delete or rename it, or adjust your $PATH setting.\n"); |
| } |
| |
| /* |
| * Check if there is an existing ../_vimrc or ../.vimrc file. |
| */ |
| strcpy(oldvimrc, installdir); |
| strcpy(oldvimrc + runtimeidx, "_vimrc"); |
| if ((fd = fopen(oldvimrc, "r")) == NULL) |
| { |
| strcpy(oldvimrc + runtimeidx, "vimrc~1"); // short version of .vimrc |
| if ((fd = fopen(oldvimrc, "r")) == NULL) |
| { |
| strcpy(oldvimrc + runtimeidx, ".vimrc"); |
| fd = fopen(oldvimrc, "r"); |
| } |
| } |
| if (fd != NULL) |
| fclose(fd); |
| else |
| *oldvimrc = NUL; |
| } |
| |
| /* |
| * Add a dummy choice to avoid that the numbering changes depending on items |
| * in the environment. The user may type a number he remembered without |
| * looking. |
| */ |
| static void |
| add_dummy_choice(void) |
| { |
| choices[choice_count].installfunc = NULL; |
| choices[choice_count].active = 0; |
| choices[choice_count].changefunc = NULL; |
| choices[choice_count].text = NULL; |
| choices[choice_count].arg = 0; |
| ++choice_count; |
| } |
| |
| //////////////////////////////////////////////// |
| // stuff for creating the batch files. |
| |
| /* |
| * Install the vim.bat, gvim.bat, etc. files. |
| */ |
| static void |
| install_bat_choice(int idx) |
| { |
| char *batpath = targets[choices[idx].arg].batpath; |
| char *oldname = targets[choices[idx].arg].oldbat; |
| char *exename = targets[choices[idx].arg].exenamearg; |
| char *vimarg = targets[choices[idx].arg].exearg; |
| FILE *fd; |
| |
| if (*batpath != NUL) |
| { |
| fd = fopen(batpath, "w"); |
| if (fd == NULL) |
| printf("\nERROR: Cannot open \"%s\" for writing.\n", batpath); |
| else |
| { |
| need_uninstall_entry = 1; |
| |
| fprintf(fd, "@echo off\n"); |
| fprintf(fd, "rem -- Run Vim --\n"); |
| fprintf(fd, VIMBAT_UNINSTKEY "\n"); |
| fprintf(fd, "\n"); |
| fprintf(fd, "setlocal\n"); |
| |
| /* |
| * Don't use double quotes for the "set" argument, also when it |
| * contains a space. The quotes would be included in the value. |
| * The order of preference is: |
| * 1. $VIMRUNTIME/vim.exe (user preference) |
| * 2. $VIM/vim81/vim.exe (hard coded version) |
| * 3. installdir/vim.exe (hard coded install directory) |
| */ |
| fprintf(fd, "set VIM_EXE_DIR=%s\n", installdir); |
| fprintf(fd, "if exist \"%%VIM%%\\%s\\%s\" set VIM_EXE_DIR=%%VIM%%\\%s\n", |
| VIM_VERSION_NODOT, exename, VIM_VERSION_NODOT); |
| fprintf(fd, "if exist \"%%VIMRUNTIME%%\\%s\" set VIM_EXE_DIR=%%VIMRUNTIME%%\n", exename); |
| fprintf(fd, "\n"); |
| |
| // Give an error message when the executable could not be found. |
| fprintf(fd, "if not exist \"%%VIM_EXE_DIR%%\\%s\" (\n", exename); |
| fprintf(fd, " echo \"%%VIM_EXE_DIR%%\\%s\" not found\n", exename); |
| fprintf(fd, " goto :eof\n"); |
| fprintf(fd, ")\n"); |
| fprintf(fd, "\n"); |
| |
| if (*exename == 'g') |
| { |
| fprintf(fd, "rem check --nofork argument\n"); |
| fprintf(fd, "set VIMNOFORK=\n"); |
| fprintf(fd, ":loopstart\n"); |
| fprintf(fd, "if .%%1==. goto loopend\n"); |
| fprintf(fd, "if .%%1==.--nofork (\n"); |
| fprintf(fd, " set VIMNOFORK=1\n"); |
| fprintf(fd, ") else if .%%1==.-f (\n"); |
| fprintf(fd, " set VIMNOFORK=1\n"); |
| fprintf(fd, ")\n"); |
| fprintf(fd, "shift\n"); |
| fprintf(fd, "goto loopstart\n"); |
| fprintf(fd, ":loopend\n"); |
| fprintf(fd, "\n"); |
| } |
| |
| if (*exename == 'g') |
| { |
| // For gvim.exe use "start /b" to avoid that the console window |
| // stays open. |
| fprintf(fd, "if .%%VIMNOFORK%%==.1 (\n"); |
| fprintf(fd, " start \"dummy\" /b /wait "); |
| // Always use quotes, $VIM or $VIMRUNTIME might have a space. |
| fprintf(fd, "\"%%VIM_EXE_DIR%%\\%s\" %s %%*\n", |
| exename, vimarg); |
| fprintf(fd, ") else (\n"); |
| fprintf(fd, " start \"dummy\" /b "); |
| // Always use quotes, $VIM or $VIMRUNTIME might have a space. |
| fprintf(fd, "\"%%VIM_EXE_DIR%%\\%s\" %s %%*\n", |
| exename, vimarg); |
| fprintf(fd, ")\n"); |
| } |
| else |
| { |
| // Always use quotes, $VIM or $VIMRUNTIME might have a space. |
| fprintf(fd, "\"%%VIM_EXE_DIR%%\\%s\" %s %%*\n", |
| exename, vimarg); |
| } |
| |
| fclose(fd); |
| printf("%s has been %s\n", batpath, |
| oldname == NULL ? "created" : "overwritten"); |
| } |
| } |
| } |
| |
| /* |
| * Make the text string for choice "idx". |
| * The format "fmt" is must have one %s item, which "arg" is used for. |
| */ |
| static void |
| alloc_text(int idx, char *fmt, char *arg) |
| { |
| if (choices[idx].text != NULL) |
| free(choices[idx].text); |
| |
| choices[idx].text = alloc(strlen(fmt) + strlen(arg) - 1); |
| sprintf(choices[idx].text, fmt, arg); |
| } |
| |
| /* |
| * Toggle the "Overwrite .../vim.bat" to "Don't overwrite". |
| */ |
| static void |
| toggle_bat_choice(int idx) |
| { |
| char *batname = targets[choices[idx].arg].batpath; |
| char *oldname = targets[choices[idx].arg].oldbat; |
| |
| if (*batname == NUL) |
| { |
| alloc_text(idx, " Overwrite %s", oldname); |
| strcpy(batname, oldname); |
| } |
| else |
| { |
| alloc_text(idx, " Do NOT overwrite %s", oldname); |
| *batname = NUL; |
| } |
| } |
| |
| /* |
| * Do some work for a batch file entry: Append the batch file name to the path |
| * and set the text for the choice. |
| */ |
| static void |
| set_bat_text(int idx, char *batpath, char *name) |
| { |
| strcat(batpath, name); |
| |
| alloc_text(idx, " Create %s", batpath); |
| } |
| |
| /* |
| * Select a directory to write the batch file line. |
| */ |
| static void |
| change_bat_choice(int idx) |
| { |
| char *path; |
| char *batpath; |
| char *name; |
| int n; |
| char *s; |
| char *p; |
| int count; |
| char **names = NULL; |
| int i; |
| int target = choices[idx].arg; |
| |
| name = targets[target].batname; |
| batpath = targets[target].batpath; |
| |
| path = getenv("PATH"); |
| if (path == NULL) |
| { |
| printf("\nERROR: The variable $PATH is not set\n"); |
| return; |
| } |
| |
| /* |
| * first round: count number of names in path; |
| * second round: save names to names[]. |
| */ |
| for (;;) |
| { |
| count = 1; |
| for (p = path; *p; ) |
| { |
| s = strchr(p, ';'); |
| if (s == NULL) |
| s = p + strlen(p); |
| if (names != NULL) |
| { |
| names[count] = alloc(s - p + 1); |
| strncpy(names[count], p, s - p); |
| names[count][s - p] = NUL; |
| } |
| ++count; |
| p = s; |
| if (*p != NUL) |
| ++p; |
| } |
| if (names != NULL) |
| break; |
| names = alloc((count + 1) * sizeof(char *)); |
| } |
| names[0] = alloc(50); |
| sprintf(names[0], "Select directory to create %s in:", name); |
| names[count] = alloc(50); |
| if (choices[idx].arg == 0) |
| sprintf(names[count], "Do not create any .bat file."); |
| else |
| sprintf(names[count], "Do not create a %s file.", name); |
| n = get_choice(names, count + 1); |
| |
| if (n == count) |
| { |
| // Selected last item, don't create bat file. |
| *batpath = NUL; |
| if (choices[idx].arg != 0) |
| alloc_text(idx, " Do NOT create %s", name); |
| } |
| else |
| { |
| // Selected one of the paths. For the first item only keep the path, |
| // for the others append the batch file name. |
| strcpy(batpath, names[n]); |
| add_pathsep(batpath); |
| if (choices[idx].arg != 0) |
| set_bat_text(idx, batpath, name); |
| } |
| |
| for (i = 0; i <= count; ++i) |
| free(names[i]); |
| free(names); |
| } |
| |
| char *bat_text_yes = "Install .bat files to use Vim at the command line:"; |
| char *bat_text_no = "do NOT install .bat files to use Vim at the command line"; |
| |
| static void |
| change_main_bat_choice(int idx) |
| { |
| int i; |
| |
| // let the user select a default directory or NONE |
| change_bat_choice(idx); |
| |
| if (targets[0].batpath[0] != NUL) |
| choices[idx].text = bat_text_yes; |
| else |
| choices[idx].text = bat_text_no; |
| |
| // update the individual batch file selections |
| for (i = 1; i < TARGET_COUNT; ++i) |
| { |
| // Only make it active when the first item has a path and the vim.exe |
| // or gvim.exe exists (there is a changefunc then). |
| if (targets[0].batpath[0] != NUL |
| && choices[idx + i].changefunc != NULL) |
| { |
| choices[idx + i].active = 1; |
| if (choices[idx + i].changefunc == change_bat_choice |
| && targets[i].batpath[0] != NUL) |
| { |
| strcpy(targets[i].batpath, targets[0].batpath); |
| set_bat_text(idx + i, targets[i].batpath, targets[i].batname); |
| } |
| } |
| else |
| choices[idx + i].active = 0; |
| } |
| } |
| |
| /* |
| * Initialize a choice for creating a batch file. |
| */ |
| static void |
| init_bat_choice(int target) |
| { |
| char *batpath = targets[target].batpath; |
| char *oldbat = targets[target].oldbat; |
| char *p; |
| int i; |
| |
| choices[choice_count].arg = target; |
| choices[choice_count].installfunc = install_bat_choice; |
| choices[choice_count].active = 1; |
| choices[choice_count].text = NULL; // will be set below |
| if (oldbat != NULL) |
| { |
| // A [g]vim.bat exists: Only choice is to overwrite it or not. |
| choices[choice_count].changefunc = toggle_bat_choice; |
| *batpath = NUL; |
| toggle_bat_choice(choice_count); |
| } |
| else |
| { |
| if (default_bat_dir != NULL) |
| // Prefer using the same path as an existing .bat file. |
| strcpy(batpath, default_bat_dir); |
| else |
| { |
| // No [g]vim.bat exists: Write it to a directory in $PATH. Use |
| // $WINDIR by default, if it's empty the first item in $PATH. |
| p = getenv("WINDIR"); |
| if (p != NULL && *p != NUL) |
| strcpy(batpath, p); |
| else |
| { |
| p = getenv("PATH"); |
| if (p == NULL || *p == NUL) // "cannot happen" |
| strcpy(batpath, "C:/Windows"); |
| else |
| { |
| i = 0; |
| while (*p != NUL && *p != ';') |
| batpath[i++] = *p++; |
| batpath[i] = NUL; |
| } |
| } |
| } |
| add_pathsep(batpath); |
| set_bat_text(choice_count, batpath, targets[target].batname); |
| |
| choices[choice_count].changefunc = change_bat_choice; |
| } |
| ++choice_count; |
| } |
| |
| /* |
| * Set up the choices for installing .bat files. |
| * For these items "arg" is the index in targets[]. |
| */ |
| static void |
| init_bat_choices(void) |
| { |
| int i; |
| |
| // The first item is used to switch installing batch files on/off and |
| // setting the default path. |
| choices[choice_count].text = bat_text_yes; |
| choices[choice_count].changefunc = change_main_bat_choice; |
| choices[choice_count].installfunc = NULL; |
| choices[choice_count].active = 1; |
| choices[choice_count].arg = 0; |
| ++choice_count; |
| |
| // Add items for each batch file target. Only used when not disabled by |
| // the first item. When a .exe exists, don't offer to create a .bat. |
| for (i = 1; i < TARGET_COUNT; ++i) |
| if (targets[i].oldexe == NULL |
| && (targets[i].exenamearg[0] == 'g' ? has_gvim : has_vim)) |
| init_bat_choice(i); |
| else |
| add_dummy_choice(); |
| } |
| |
| /* |
| * Install the vimrc file. |
| */ |
| static void |
| install_vimrc(int idx) |
| { |
| FILE *fd, *tfd; |
| char *fname; |
| |
| // If an old vimrc file exists, overwrite it. |
| // Otherwise create a new one. |
| if (*oldvimrc != NUL) |
| fname = oldvimrc; |
| else |
| fname = vimrc; |
| |
| fd = fopen(fname, "w"); |
| if (fd == NULL) |
| { |
| printf("\nERROR: Cannot open \"%s\" for writing.\n", fname); |
| return; |
| } |
| switch (compat_choice) |
| { |
| case compat_vi: |
| fprintf(fd, "\" Vi compatible\n"); |
| fprintf(fd, "set compatible\n"); |
| break; |
| case compat_vim: |
| fprintf(fd, "\" Vim's default behavior\n"); |
| fprintf(fd, "if &compatible\n"); |
| fprintf(fd, " set nocompatible\n"); |
| fprintf(fd, "endif\n"); |
| break; |
| case compat_some_enhancements: |
| fprintf(fd, "\" Vim with some enhancements\n"); |
| fprintf(fd, "source $VIMRUNTIME/defaults.vim\n"); |
| break; |
| case compat_all_enhancements: |
| fprintf(fd, "\" Vim with all enhancements\n"); |
| fprintf(fd, "source $VIMRUNTIME/vimrc_example.vim\n"); |
| break; |
| } |
| switch (remap_choice) |
| { |
| case remap_no: |
| break; |
| case remap_win: |
| fprintf(fd, "\n"); |
| fprintf(fd, "\" Remap a few keys for Windows behavior\n"); |
| fprintf(fd, "source $VIMRUNTIME/mswin.vim\n"); |
| break; |
| } |
| switch (mouse_choice) |
| { |
| case mouse_xterm: |
| fprintf(fd, "\n"); |
| fprintf(fd, "\" Mouse behavior (the Unix way)\n"); |
| fprintf(fd, "behave xterm\n"); |
| break; |
| case mouse_mswin: |
| fprintf(fd, "\n"); |
| fprintf(fd, "\" Mouse behavior (the Windows way)\n"); |
| fprintf(fd, "behave mswin\n"); |
| break; |
| case mouse_default: |
| break; |
| } |
| if ((tfd = fopen("diff.exe", "r")) != NULL) |
| { |
| // Use the diff.exe that comes with the self-extracting gvim.exe. |
| fclose(tfd); |
| fprintf(fd, "\n"); |
| fprintf(fd, "\" Use the internal diff if available.\n"); |
| fprintf(fd, "\" Otherwise use the special 'diffexpr' for Windows.\n"); |
| fprintf(fd, "if &diffopt !~# 'internal'\n"); |
| fprintf(fd, " set diffexpr=MyDiff()\n"); |
| fprintf(fd, "endif\n"); |
| fprintf(fd, "function MyDiff()\n"); |
| fprintf(fd, " let opt = '-a --binary '\n"); |
| fprintf(fd, " if &diffopt =~ 'icase' | let opt = opt . '-i ' | endif\n"); |
| fprintf(fd, " if &diffopt =~ 'iwhite' | let opt = opt . '-b ' | endif\n"); |
| // Use quotes only when needed, they may cause trouble. |
| // Always escape "!". |
| fprintf(fd, " let arg1 = v:fname_in\n"); |
| fprintf(fd, " if arg1 =~ ' ' | let arg1 = '\"' . arg1 . '\"' | endif\n"); |
| fprintf(fd, " let arg1 = substitute(arg1, '!', '\\!', 'g')\n"); |
| fprintf(fd, " let arg2 = v:fname_new\n"); |
| fprintf(fd, " if arg2 =~ ' ' | let arg2 = '\"' . arg2 . '\"' | endif\n"); |
| fprintf(fd, " let arg2 = substitute(arg2, '!', '\\!', 'g')\n"); |
| fprintf(fd, " let arg3 = v:fname_out\n"); |
| fprintf(fd, " if arg3 =~ ' ' | let arg3 = '\"' . arg3 . '\"' | endif\n"); |
| fprintf(fd, " let arg3 = substitute(arg3, '!', '\\!', 'g')\n"); |
| |
| // If the path has a space: When using cmd.exe (Win NT/2000/XP) put |
| // quotes around the diff command and rely on the default value of |
| // shellxquote to solve the quoting problem for the whole command. |
| // |
| // Otherwise put a double quote just before the space and at the |
| // end of the command. Putting quotes around the whole thing |
| // doesn't work on Win 95/98/ME. This is mostly guessed! |
| fprintf(fd, " if $VIMRUNTIME =~ ' '\n"); |
| fprintf(fd, " if &sh =~ '\\<cmd'\n"); |
| fprintf(fd, " if empty(&shellxquote)\n"); |
| fprintf(fd, " let l:shxq_sav = ''\n"); |
| fprintf(fd, " set shellxquote&\n"); |
| fprintf(fd, " endif\n"); |
| fprintf(fd, " let cmd = '\"' . $VIMRUNTIME . '\\diff\"'\n"); |
| fprintf(fd, " else\n"); |
| fprintf(fd, " let cmd = substitute($VIMRUNTIME, ' ', '\" ', '') . '\\diff\"'\n"); |
| fprintf(fd, " endif\n"); |
| fprintf(fd, " else\n"); |
| fprintf(fd, " let cmd = $VIMRUNTIME . '\\diff'\n"); |
| fprintf(fd, " endif\n"); |
| fprintf(fd, " let cmd = substitute(cmd, '!', '\\!', 'g')\n"); |
| fprintf(fd, " silent execute '!' . cmd . ' ' . opt . arg1 . ' ' . arg2 . ' > ' . arg3\n"); |
| fprintf(fd, " if exists('l:shxq_sav')\n"); |
| fprintf(fd, " let &shellxquote=l:shxq_sav\n"); |
| fprintf(fd, " endif\n"); |
| fprintf(fd, "endfunction\n"); |
| fprintf(fd, "\n"); |
| } |
| fclose(fd); |
| printf("%s has been written\n", fname); |
| } |
| |
| static void |
| change_vimrc_choice(int idx) |
| { |
| if (choices[idx].installfunc != NULL) |
| { |
| // Switch to NOT change or create a vimrc file. |
| if (*oldvimrc != NUL) |
| alloc_text(idx, "Do NOT change startup file %s", oldvimrc); |
| else |
| alloc_text(idx, "Do NOT create startup file %s", vimrc); |
| choices[idx].installfunc = NULL; |
| choices[idx + 1].active = 0; |
| choices[idx + 2].active = 0; |
| choices[idx + 3].active = 0; |
| } |
| else |
| { |
| // Switch to change or create a vimrc file. |
| if (*oldvimrc != NUL) |
| alloc_text(idx, "Overwrite startup file %s with:", oldvimrc); |
| else |
| alloc_text(idx, "Create startup file %s with:", vimrc); |
| choices[idx].installfunc = install_vimrc; |
| choices[idx + 1].active = 1; |
| choices[idx + 2].active = 1; |
| choices[idx + 3].active = 1; |
| } |
| } |
| |
| /* |
| * Change the choice how to run Vim. |
| */ |
| static void |
| change_run_choice(int idx) |
| { |
| compat_choice = get_choice(compat_choices, TABLE_SIZE(compat_choices)); |
| alloc_text(idx, compat_text, compat_choices[compat_choice]); |
| } |
| |
| /* |
| * Change the choice if keys are to be remapped. |
| */ |
| static void |
| change_remap_choice(int idx) |
| { |
| remap_choice = get_choice(remap_choices, TABLE_SIZE(remap_choices)); |
| alloc_text(idx, remap_text, remap_choices[remap_choice]); |
| } |
| |
| /* |
| * Change the choice how to select text. |
| */ |
| static void |
| change_mouse_choice(int idx) |
| { |
| mouse_choice = get_choice(mouse_choices, TABLE_SIZE(mouse_choices)); |
| alloc_text(idx, mouse_text, mouse_choices[mouse_choice]); |
| } |
| |
| static void |
| init_vimrc_choices(void) |
| { |
| // set path for a new _vimrc file (also when not used) |
| strcpy(vimrc, installdir); |
| strcpy(vimrc + runtimeidx, "_vimrc"); |
| |
| // Set opposite value and then toggle it by calling change_vimrc_choice() |
| if (*oldvimrc == NUL) |
| choices[choice_count].installfunc = NULL; |
| else |
| choices[choice_count].installfunc = install_vimrc; |
| choices[choice_count].text = NULL; |
| change_vimrc_choice(choice_count); |
| choices[choice_count].changefunc = change_vimrc_choice; |
| choices[choice_count].active = 1; |
| ++choice_count; |
| |
| // default way to run Vim |
| alloc_text(choice_count, compat_text, compat_choices[compat_choice]); |
| choices[choice_count].changefunc = change_run_choice; |
| choices[choice_count].installfunc = NULL; |
| choices[choice_count].active = (*oldvimrc == NUL); |
| ++choice_count; |
| |
| // Whether to remap keys |
| alloc_text(choice_count, remap_text , remap_choices[remap_choice]); |
| choices[choice_count].changefunc = change_remap_choice; |
| choices[choice_count].installfunc = NULL; |
| choices[choice_count].active = (*oldvimrc == NUL); |
| ++choice_count; |
| |
| // default way to use the mouse |
| alloc_text(choice_count, mouse_text, mouse_choices[mouse_choice]); |
| choices[choice_count].changefunc = change_mouse_choice; |
| choices[choice_count].installfunc = NULL; |
| choices[choice_count].active = (*oldvimrc == NUL); |
| ++choice_count; |
| } |
| |
| static LONG |
| reg_create_key( |
| HKEY root, |
| const char *subkey, |
| PHKEY phKey, |
| DWORD flag) |
| { |
| DWORD disp; |
| |
| *phKey = NULL; |
| return RegCreateKeyEx( |
| root, subkey, |
| 0, NULL, REG_OPTION_NON_VOLATILE, |
| flag | KEY_WRITE, |
| NULL, phKey, &disp); |
| } |
| |
| static LONG |
| reg_set_string_value( |
| HKEY hKey, |
| const char *value_name, |
| const char *data) |
| { |
| return RegSetValueEx(hKey, value_name, 0, REG_SZ, |
| (LPBYTE)data, (DWORD)(1 + strlen(data))); |
| } |
| |
| static LONG |
| reg_create_key_and_value( |
| HKEY hRootKey, |
| const char *subkey, |
| const char *value_name, |
| const char *data, |
| DWORD flag) |
| { |
| HKEY hKey; |
| LONG lRet = reg_create_key(hRootKey, subkey, &hKey, flag); |
| |
| if (ERROR_SUCCESS == lRet) |
| { |
| lRet = reg_set_string_value(hKey, value_name, data); |
| RegCloseKey(hKey); |
| } |
| return lRet; |
| } |
| |
| static LONG |
| register_inproc_server( |
| HKEY hRootKey, |
| const char *clsid, |
| const char *extname, |
| const char *module, |
| const char *threading_model, |
| DWORD flag) |
| { |
| CHAR subkey[BUFSIZE]; |
| LONG lRet; |
| |
| sprintf(subkey, "CLSID\\%s", clsid); |
| lRet = reg_create_key_and_value(hRootKey, subkey, NULL, extname, flag); |
| if (ERROR_SUCCESS == lRet) |
| { |
| sprintf(subkey, "CLSID\\%s\\InProcServer32", clsid); |
| lRet = reg_create_key_and_value(hRootKey, subkey, NULL, module, flag); |
| if (ERROR_SUCCESS == lRet) |
| { |
| lRet = reg_create_key_and_value(hRootKey, subkey, |
| "ThreadingModel", threading_model, flag); |
| } |
| } |
| return lRet; |
| } |
| |
| static LONG |
| register_shellex( |
| HKEY hRootKey, |
| const char *clsid, |
| const char *name, |
| const char *exe_path, |
| DWORD flag) |
| { |
| LONG lRet = reg_create_key_and_value( |
| hRootKey, |
| "*\\shellex\\ContextMenuHandlers\\gvim", |
| NULL, |
| clsid, |
| flag); |
| |
| if (ERROR_SUCCESS == lRet) |
| { |
| lRet = reg_create_key_and_value( |
| HKEY_LOCAL_MACHINE, |
| "Software\\Microsoft\\Windows\\CurrentVersion\\Shell Extensions\\Approved", |
| clsid, |
| name, |
| flag); |
| |
| if (ERROR_SUCCESS == lRet) |
| { |
| lRet = reg_create_key_and_value( |
| HKEY_LOCAL_MACHINE, |
| "Software\\Vim\\Gvim", |
| "path", |
| exe_path, |
| flag); |
| } |
| } |
| return lRet; |
| } |
| |
| static LONG |
| register_openwith( |
| HKEY hRootKey, |
| const char *exe_path, |
| DWORD flag) |
| { |
| char exe_cmd[BUFSIZE]; |
| LONG lRet; |
| |
| sprintf(exe_cmd, "\"%s\" \"%%1\"", exe_path); |
| lRet = reg_create_key_and_value( |
| hRootKey, |
| "Applications\\gvim.exe\\shell\\edit\\command", |
| NULL, |
| exe_cmd, |
| flag); |
| |
| if (ERROR_SUCCESS == lRet) |
| { |
| int i; |
| static const char *openwith[] = { |
| ".htm\\OpenWithList\\gvim.exe", |
| ".vim\\OpenWithList\\gvim.exe", |
| "*\\OpenWithList\\gvim.exe", |
| }; |
| |
| for (i = 0; ERROR_SUCCESS == lRet && i < ARRAYSIZE(openwith); i++) |
| lRet = reg_create_key_and_value(hRootKey, openwith[i], NULL, "", flag); |
| } |
| |
| return lRet; |
| } |
| |
| static LONG |
| register_uninstall( |
| HKEY hRootKey, |
| const char *appname, |
| const char *display_name, |
| const char *uninstall_string, |
| const char *display_icon, |
| const char *display_version, |
| const char *publisher) |
| { |
| LONG lRet = reg_create_key_and_value(hRootKey, appname, |
| "DisplayName", display_name, KEY_WOW64_64KEY); |
| |
| if (ERROR_SUCCESS == lRet) |
| lRet = reg_create_key_and_value(hRootKey, appname, |
| "UninstallString", uninstall_string, KEY_WOW64_64KEY); |
| if (ERROR_SUCCESS == lRet) |
| lRet = reg_create_key_and_value(hRootKey, appname, |
| "DisplayIcon", display_icon, KEY_WOW64_64KEY); |
| if (ERROR_SUCCESS == lRet) |
| lRet = reg_create_key_and_value(hRootKey, appname, |
| "DisplayVersion", display_version, KEY_WOW64_64KEY); |
| if (ERROR_SUCCESS == lRet) |
| lRet = reg_create_key_and_value(hRootKey, appname, |
| "Publisher", publisher, KEY_WOW64_64KEY); |
| return lRet; |
| } |
| |
| /* |
| * Add some entries to the registry: |
| * - to add "Edit with Vim" to the context * menu |
| * - to add Vim to the "Open with..." list |
| * - to uninstall Vim |
| */ |
| //ARGSUSED |
| static int |
| install_registry(void) |
| { |
| LONG lRet = ERROR_SUCCESS; |
| const char *vim_ext_ThreadingModel = "Apartment"; |
| const char *vim_ext_name = "Vim Shell Extension"; |
| const char *vim_ext_clsid = "{51EEE242-AD87-11d3-9C1E-0090278BBD99}"; |
| char vim_exe_path[MAX_PATH]; |
| char display_name[BUFSIZE]; |
| char uninstall_string[BUFSIZE]; |
| char icon_string[BUFSIZE]; |
| char version_string[BUFSIZE]; |
| int i; |
| int loop_count = is_64bit_os() ? 2 : 1; |
| DWORD flag; |
| |
| sprintf(vim_exe_path, "%s\\gvim.exe", installdir); |
| |
| if (install_popup) |
| { |
| char bufg[BUFSIZE]; |
| |
| printf("Creating \"Edit with Vim\" popup menu entry\n"); |
| |
| for (i = 0; i < loop_count; i++) |
| { |
| if (i == 0) |
| { |
| sprintf(bufg, "%s\\" GVIMEXT32_PATH, installdir); |
| flag = KEY_WOW64_32KEY; |
| } |
| else |
| { |
| sprintf(bufg, "%s\\" GVIMEXT64_PATH, installdir); |
| flag = KEY_WOW64_64KEY; |
| } |
| |
| lRet = register_inproc_server( |
| HKEY_CLASSES_ROOT, vim_ext_clsid, vim_ext_name, |
| bufg, vim_ext_ThreadingModel, flag); |
| if (ERROR_SUCCESS != lRet) |
| return FAIL; |
| lRet = register_shellex( |
| HKEY_CLASSES_ROOT, vim_ext_clsid, vim_ext_name, |
| vim_exe_path, flag); |
| if (ERROR_SUCCESS != lRet) |
| return FAIL; |
| } |
| } |
| |
| if (install_openwith) |
| { |
| printf("Creating \"Open with ...\" list entry\n"); |
| |
| for (i = 0; i < loop_count; i++) |
| { |
| if (i == 0) |
| flag = KEY_WOW64_32KEY; |
| else |
| flag = KEY_WOW64_64KEY; |
| |
| lRet = register_openwith(HKEY_CLASSES_ROOT, vim_exe_path, flag); |
| if (ERROR_SUCCESS != lRet) |
| return FAIL; |
| } |
| } |
| |
| printf("Creating an uninstall entry\n"); |
| sprintf(display_name, "Vim " VIM_VERSION_SHORT |
| #ifdef _M_ARM64 |
| " (arm64)" |
| #elif _M_X64 |
| " (x64)" |
| #endif |
| ); |
| |
| // For the NSIS installer use the generated uninstaller. |
| if (interactive) |
| sprintf(uninstall_string, "%s\\uninstall.exe", installdir); |
| else |
| sprintf(uninstall_string, "%s\\uninstall-gui.exe", installdir); |
| |
| sprintf(icon_string, "%s\\gvim.exe,0", installdir); |
| |
| sprintf(version_string, VIM_VERSION_SHORT "." VIM_VERSION_PATCHLEVEL_STR); |
| |
| lRet = register_uninstall( |
| HKEY_LOCAL_MACHINE, |
| "Software\\Microsoft\\Windows\\CurrentVersion\\Uninstall\\Vim " VIM_VERSION_SHORT, |
| display_name, |
| uninstall_string, |
| icon_string, |
| version_string, |
| "Bram Moolenaar et al."); |
| if (ERROR_SUCCESS != lRet) |
| return FAIL; |
| |
| return OK; |
| } |
| |
| static void |
| change_popup_choice(int idx) |
| { |
| if (install_popup == 0) |
| { |
| choices[idx].text = "Install an entry for Vim in the popup menu for the right\n mouse button so that you can edit any file with Vim"; |
| install_popup = 1; |
| } |
| else |
| { |
| choices[idx].text = "Do NOT install an entry for Vim in the popup menu for the\n right mouse button to edit any file with Vim"; |
| install_popup = 0; |
| } |
| } |
| |
| /* |
| * Only add the choice for the popup menu entry when gvim.exe was found and |
| * both gvimext.dll and regedit.exe exist. |
| */ |
| static void |
| init_popup_choice(void) |
| { |
| struct stat st; |
| |
| if (has_gvim |
| && (stat(GVIMEXT32_PATH, &st) >= 0 |
| || stat(GVIMEXT64_PATH, &st) >= 0)) |
| { |
| choices[choice_count].changefunc = change_popup_choice; |
| choices[choice_count].installfunc = NULL; |
| choices[choice_count].active = 1; |
| change_popup_choice(choice_count); // set the text |
| ++choice_count; |
| } |
| else |
| add_dummy_choice(); |
| } |
| |
| static void |
| change_openwith_choice(int idx) |
| { |
| if (install_openwith == 0) |
| { |
| choices[idx].text = "Add Vim to the \"Open With...\" list in the popup menu for the right\n mouse button so that you can edit any file with Vim"; |
| install_openwith = 1; |
| } |
| else |
| { |
| choices[idx].text = "Do NOT add Vim to the \"Open With...\" list in the popup menu for the\n right mouse button to edit any file with Vim"; |
| install_openwith = 0; |
| } |
| } |
| |
| /* |
| * Only add the choice for the open-with menu entry when gvim.exe was found |
| * and regedit.exe exist. |
| */ |
| static void |
| init_openwith_choice(void) |
| { |
| if (has_gvim) |
| { |
| choices[choice_count].changefunc = change_openwith_choice; |
| choices[choice_count].installfunc = NULL; |
| choices[choice_count].active = 1; |
| change_openwith_choice(choice_count); // set the text |
| ++choice_count; |
| } |
| else |
| add_dummy_choice(); |
| } |
| |
| /* |
| * Create a shell link. |
| * |
| * returns 0 on failure, non-zero on successful completion. |
| * |
| * NOTE: Currently untested with mingw. |
| */ |
| int |
| create_shortcut( |
| const char *shortcut_name, |
| const char *iconfile_path, |
| int iconindex, |
| const char *shortcut_target, |
| const char *shortcut_args, |
| const char *workingdir |
| ) |
| { |
| IShellLink *shelllink_ptr; |
| HRESULT hres; |
| IPersistFile *persistfile_ptr; |
| |
| // Initialize COM library |
| hres = CoInitialize(NULL); |
| if (!SUCCEEDED(hres)) |
| { |
| printf("Error: Could not open the COM library. Not creating shortcut.\n"); |
| return FAIL; |
| } |
| |
| // Instantiate a COM object for the ShellLink, store a pointer to it |
| // in shelllink_ptr. |
| hres = CoCreateInstance(&CLSID_ShellLink, |
| NULL, |
| CLSCTX_INPROC_SERVER, |
| &IID_IShellLink, |
| (void **) &shelllink_ptr); |
| |
| if (SUCCEEDED(hres)) // If the instantiation was successful... |
| { |
| // ...Then build a PersistFile interface for the ShellLink so we can |
| // save it as a file after we build it. |
| hres = shelllink_ptr->lpVtbl->QueryInterface(shelllink_ptr, |
| &IID_IPersistFile, (void **) &persistfile_ptr); |
| |
| if (SUCCEEDED(hres)) |
| { |
| wchar_t wsz[BUFSIZE]; |
| |
| // translate the (possibly) multibyte shortcut filename to windows |
| // Unicode so it can be used as a file name. |
| MultiByteToWideChar(CP_ACP, 0, shortcut_name, -1, wsz, sizeof(wsz)/sizeof(wsz[0])); |
| |
| // set the attributes |
| shelllink_ptr->lpVtbl->SetPath(shelllink_ptr, shortcut_target); |
| shelllink_ptr->lpVtbl->SetWorkingDirectory(shelllink_ptr, |
| workingdir); |
| shelllink_ptr->lpVtbl->SetIconLocation(shelllink_ptr, |
| iconfile_path, iconindex); |
| shelllink_ptr->lpVtbl->SetArguments(shelllink_ptr, shortcut_args); |
| |
| // save the shortcut to a file and return the PersistFile object |
| persistfile_ptr->lpVtbl->Save(persistfile_ptr, wsz, 1); |
| persistfile_ptr->lpVtbl->Release(persistfile_ptr); |
| } |
| else |
| { |
| printf("QueryInterface Error\n"); |
| return FAIL; |
| } |
| |
| // Return the ShellLink object |
| shelllink_ptr->lpVtbl->Release(shelllink_ptr); |
| } |
| else |
| { |
| printf("CoCreateInstance Error - hres = %08x\n", (int)hres); |
| return FAIL; |
| } |
| |
| return OK; |
| } |
| |
| /* |
| * Build a path to where we will put a specified link. |
| * |
| * Return 0 on error, non-zero on success |
| */ |
| int |
| build_link_name( |
| char *link_path, |
| const char *link_name, |
| const char *shell_folder_name) |
| { |
| char shell_folder_path[MAX_PATH]; |
| |
| if (get_shell_folder_path(shell_folder_path, shell_folder_name) == FAIL) |
| { |
| printf("An error occurred while attempting to find the path to %s.\n", |
| shell_folder_name); |
| return FAIL; |
| } |
| |
| // Make sure the directory exists (create Start Menu\Programs\Vim). |
| // Ignore errors if it already exists. |
| vim_mkdir(shell_folder_path, 0755); |
| |
| // build the path to the shortcut and the path to gvim.exe |
| sprintf(link_path, "%s\\%s.lnk", shell_folder_path, link_name); |
| |
| return OK; |
| } |
| |
| static int |
| build_shortcut( |
| const char *name, // Name of the shortcut |
| const char *exename, // Name of the executable (e.g., vim.exe) |
| const char *args, |
| const char *shell_folder, |
| const char *workingdir) |
| { |
| char executable_path[BUFSIZE]; |
| char link_name[BUFSIZE]; |
| |
| sprintf(executable_path, "%s\\%s", installdir, exename); |
| |
| if (build_link_name(link_name, name, shell_folder) == FAIL) |
| { |
| printf("An error has occurred. A shortcut to %s will not be created %s.\n", |
| name, |
| *shell_folder == 'd' ? "on the desktop" : "in the Start menu"); |
| return FAIL; |
| } |
| |
| // Create the shortcut: |
| return create_shortcut(link_name, executable_path, 0, |
| executable_path, args, workingdir); |
| } |
| |
| /* |
| * We used to use "homedir" as the working directory, but that is a bad choice |
| * on multi-user systems. However, not specifying a directory results in the |
| * current directory to be c:\Windows\system32 on Windows 7. Use environment |
| * variables instead. |
| */ |
| #define WORKDIR "%HOMEDRIVE%%HOMEPATH%" |
| |
| /* |
| * Create shortcut(s) in the Start Menu\Programs\Vim folder. |
| */ |
| static void |
| install_start_menu(int idx) |
| { |
| need_uninstall_entry = 1; |
| printf("Creating start menu\n"); |
| if (has_vim) |
| { |
| if (build_shortcut("Vim", "vim.exe", "", |
| VIM_STARTMENU, WORKDIR) == FAIL) |
| return; |
| if (build_shortcut("Vim Read-only", "vim.exe", "-R", |
| VIM_STARTMENU, WORKDIR) == FAIL) |
| return; |
| if (build_shortcut("Vim Diff", "vim.exe", "-d", |
| VIM_STARTMENU, WORKDIR) == FAIL) |
| return; |
| } |
| if (has_gvim) |
| { |
| if (build_shortcut("gVim", "gvim.exe", "", |
| VIM_STARTMENU, WORKDIR) == FAIL) |
| return; |
| if (build_shortcut("gVim Easy", "gvim.exe", "-y", |
| VIM_STARTMENU, WORKDIR) == FAIL) |
| return; |
| if (build_shortcut("gVim Read-only", "gvim.exe", "-R", |
| VIM_STARTMENU, WORKDIR) == FAIL) |
| return; |
| if (build_shortcut("gVim Diff", "gvim.exe", "-d", |
| VIM_STARTMENU, WORKDIR) == FAIL) |
| return; |
| } |
| if (build_shortcut("Uninstall", |
| interactive ? "uninstall.exe" : "uninstall-gui.exe", "", |
| VIM_STARTMENU, installdir) == FAIL) |
| return; |
| // For Windows NT the working dir of the vimtutor.bat must be right, |
| // otherwise gvim.exe won't be found and using gvimbat doesn't work. |
| if (build_shortcut("Vim tutor", "vimtutor.bat", "", |
| VIM_STARTMENU, installdir) == FAIL) |
| return; |
| if (build_shortcut("Help", has_gvim ? "gvim.exe" : "vim.exe", "-c h", |
| VIM_STARTMENU, WORKDIR) == FAIL) |
| return; |
| { |
| char shell_folder_path[BUFSIZE]; |
| |
| // Creating the URL shortcut works a bit differently... |
| if (get_shell_folder_path(shell_folder_path, VIM_STARTMENU) == FAIL) |
| { |
| printf("Finding the path of the Start menu failed\n"); |
| return ; |
| } |
| add_pathsep(shell_folder_path); |
| strcat(shell_folder_path, "Vim Online.url"); |
| if (!WritePrivateProfileString("InternetShortcut", "URL", |
| "https://www.vim.org/", shell_folder_path)) |
| { |
| printf("Creating the Vim online URL failed\n"); |
| return; |
| } |
| } |
| } |
| |
| static void |
| toggle_startmenu_choice(int idx) |
| { |
| if (choices[idx].installfunc == NULL) |
| { |
| choices[idx].installfunc = install_start_menu; |
| choices[idx].text = "Add Vim to the Start menu"; |
| } |
| else |
| { |
| choices[idx].installfunc = NULL; |
| choices[idx].text = "Do NOT add Vim to the Start menu"; |
| } |
| } |
| |
| /* |
| * Function to actually create the shortcuts |
| * |
| * Currently I am supplying no working directory to the shortcut. This |
| * means that the initial working dir will be: |
| * - the location of the shortcut if no file is supplied |
| * - the location of the file being edited if a file is supplied (ie via |
| * drag and drop onto the shortcut). |
| */ |
| void |
| install_shortcut_gvim(int idx) |
| { |
| // Create shortcut(s) on the desktop |
| if (choices[idx].arg) |
| { |
| (void)build_shortcut(icon_names[0], "gvim.exe", |
| "", "desktop", WORKDIR); |
| need_uninstall_entry = 1; |
| } |
| } |
| |
| void |
| install_shortcut_evim(int idx) |
| { |
| if (choices[idx].arg) |
| { |
| (void)build_shortcut(icon_names[1], "gvim.exe", |
| "-y", "desktop", WORKDIR); |
| need_uninstall_entry = 1; |
| } |
| } |
| |
| void |
| install_shortcut_gview(int idx) |
| { |
| if (choices[idx].arg) |
| { |
| (void)build_shortcut(icon_names[2], "gvim.exe", |
| "-R", "desktop", WORKDIR); |
| need_uninstall_entry = 1; |
| } |
| } |
| |
| void |
| toggle_shortcut_choice(int idx) |
| { |
| char *arg; |
| |
| if (choices[idx].installfunc == install_shortcut_gvim) |
| arg = "gVim"; |
| else if (choices[idx].installfunc == install_shortcut_evim) |
| arg = "gVim Easy"; |
| else |
| arg = "gVim Read-only"; |
| if (choices[idx].arg) |
| { |
| choices[idx].arg = 0; |
| alloc_text(idx, "Do NOT create a desktop icon for %s", arg); |
| } |
| else |
| { |
| choices[idx].arg = 1; |
| alloc_text(idx, "Create a desktop icon for %s", arg); |
| } |
| } |
| |
| static void |
| init_startmenu_choice(void) |
| { |
| // Start menu |
| choices[choice_count].changefunc = toggle_startmenu_choice; |
| choices[choice_count].installfunc = NULL; |
| choices[choice_count].active = 1; |
| toggle_startmenu_choice(choice_count); // set the text |
| ++choice_count; |
| } |
| |
| /* |
| * Add the choice for the desktop shortcuts. |
| */ |
| static void |
| init_shortcut_choices(void) |
| { |
| // Shortcut to gvim |
| choices[choice_count].text = NULL; |
| choices[choice_count].arg = 0; |
| choices[choice_count].active = has_gvim; |
| choices[choice_count].changefunc = toggle_shortcut_choice; |
| choices[choice_count].installfunc = install_shortcut_gvim; |
| toggle_shortcut_choice(choice_count); |
| ++choice_count; |
| |
| // Shortcut to evim |
| choices[choice_count].text = NULL; |
| choices[choice_count].arg = 0; |
| choices[choice_count].active = has_gvim; |
| choices[choice_count].changefunc = toggle_shortcut_choice; |
| choices[choice_count].installfunc = install_shortcut_evim; |
| toggle_shortcut_choice(choice_count); |
| ++choice_count; |
| |
| // Shortcut to gview |
| choices[choice_count].text = NULL; |
| choices[choice_count].arg = 0; |
| choices[choice_count].active = has_gvim; |
| choices[choice_count].changefunc = toggle_shortcut_choice; |
| choices[choice_count].installfunc = install_shortcut_gview; |
| toggle_shortcut_choice(choice_count); |
| ++choice_count; |
| } |
| |
| /* |
| * Attempt to register OLE for Vim. |
| */ |
| static void |
| install_OLE_register(void) |
| { |
| char register_command_string[BUFSIZE + 30]; |
| |
| printf("\n--- Attempting to register Vim with OLE ---\n"); |
| printf("(There is no message whether this works or not.)\n"); |
| |
| sprintf(register_command_string, "\"%s\\gvim.exe\" -silent -register", installdir); |
| system(register_command_string); |
| } |
| |
| /* |
| * Remove the last part of directory "path[]" to get its parent, and put the |
| * result in "to[]". |
| */ |
| static void |
| dir_remove_last(const char *path, char to[MAX_PATH]) |
| { |
| char c; |
| long last_char_to_copy; |
| long path_length = strlen(path); |
| |
| // skip the last character just in case it is a '\\' |
| last_char_to_copy = path_length - 2; |
| c = path[last_char_to_copy]; |
| |
| while (c != '\\') |
| { |
| last_char_to_copy--; |
| c = path[last_char_to_copy]; |
| } |
| |
| strncpy(to, path, (size_t)last_char_to_copy); |
| to[last_char_to_copy] = NUL; |
| } |
| |
| static void |
| set_directories_text(int idx) |
| { |
| int vimfiles_dir_choice = choices[idx].arg; |
| |
| if (vimfiles_dir_choice == (int)vimfiles_dir_none) |
| alloc_text(idx, "Do NOT create plugin directories%s", ""); |
| else |
| alloc_text(idx, "Create plugin directories: %s", |
| vimfiles_dir_choices[vimfiles_dir_choice]); |
| } |
| |
| /* |
| * To get the "real" home directory: |
| * - get value of $HOME |
| * - if not found, get value of $HOMEDRIVE$HOMEPATH |
| * - if not found, get value of $USERPROFILE |
| * |
| * This code is based on init_homedir() in misc1.c, keep in sync! |
| */ |
| static char *homedir = NULL; |
| |
| void |
| init_homedir(void) |
| { |
| char *var; |
| char buf[MAX_PATH]; |
| |
| if (homedir != NULL) |
| { |
| free(homedir); |
| homedir = NULL; |
| } |
| |
| var = getenv("HOME"); |
| |
| /* |
| * Typically, $HOME is not defined on Windows, unless the user has |
| * specifically defined it for Vim's sake. However, on Windows NT |
| * platforms, $HOMEDRIVE and $HOMEPATH are automatically defined for |
| * each user. Try constructing $HOME from these. |
| */ |
| if (var == NULL || *var == NUL) |
| { |
| char *homedrive, *homepath; |
| |
| homedrive = getenv("HOMEDRIVE"); |
| homepath = getenv("HOMEPATH"); |
| if (homepath == NULL || *homepath == NUL) |
| homepath = "\\"; |
| if (homedrive != NULL |
| && strlen(homedrive) + strlen(homepath) < sizeof(buf)) |
| { |
| sprintf(buf, "%s%s", homedrive, homepath); |
| if (buf[0] != NUL) |
| var = buf; |
| } |
| } |
| |
| if (var == NULL) |
| var = getenv("USERPROFILE"); |
| |
| /* |
| * Weird but true: $HOME may contain an indirect reference to another |
| * variable, esp. "%USERPROFILE%". Happens when $USERPROFILE isn't set |
| * when $HOME is being set. |
| */ |
| if (var != NULL && *var == '%') |
| { |
| char *p; |
| char *exp; |
| |
| p = strchr(var + 1, '%'); |
| if (p != NULL) |
| { |
| strncpy(buf, var + 1, p - (var + 1)); |
| buf[p - (var + 1)] = NUL; |
| exp = getenv(buf); |
| if (exp != NULL && *exp != NUL |
| && strlen(exp) + strlen(p) < sizeof(buf)) |
| { |
| sprintf(buf, "%s%s", exp, p + 1); |
| var = buf; |
| } |
| } |
| } |
| |
| if (var != NULL && *var == NUL) // empty is same as not set |
| var = NULL; |
| |
| if (var == NULL) |
| homedir = NULL; |
| else |
| homedir = _strdup(var); |
| } |
| |
| /* |
| * Change the directory that the vim plugin directories will be created in: |
| * $HOME, $VIM or nowhere. |
| */ |
| static void |
| change_directories_choice(int idx) |
| { |
| int choice_count = TABLE_SIZE(vimfiles_dir_choices); |
| |
| // Don't offer the $HOME choice if $HOME isn't set. |
| if (homedir == NULL) |
| --choice_count; |
| choices[idx].arg = get_choice(vimfiles_dir_choices, choice_count); |
| set_directories_text(idx); |
| } |
| |
| /* |
| * Create the plugin directories... |
| */ |
| //ARGSUSED |
| static void |
| install_vimfilesdir(int idx) |
| { |
| int i; |
| int vimfiles_dir_choice = choices[idx].arg; |
| char *p; |
| char vimdir_path[MAX_PATH]; |
| char vimfiles_path[MAX_PATH + 9]; |
| char tmp_dirname[BUFSIZE]; |
| |
| // switch on the location that the user wants the plugin directories |
| // built in |
| switch (vimfiles_dir_choice) |
| { |
| case vimfiles_dir_vim: |
| { |
| // Go to the %VIM% directory - check env first, then go one dir |
| // below installdir if there is no %VIM% environment variable. |
| // The accuracy of $VIM is checked in inspect_system(), so we |
| // can be sure it is ok to use here. |
| p = getenv("VIM"); |
| if (p == NULL) // No $VIM in path |
| dir_remove_last(installdir, vimdir_path); |
| else |
| strcpy(vimdir_path, p); |
| break; |
| } |
| case vimfiles_dir_home: |
| { |
| // Find the $HOME directory. Its existence was already checked. |
| p = homedir; |
| if (p == NULL) |
| { |
| printf("Internal error: $HOME is NULL\n"); |
| p = "c:\\"; |
| } |
| strcpy(vimdir_path, p); |
| break; |
| } |
| case vimfiles_dir_none: |
| { |
| // Do not create vim plugin directory. |
| return; |
| } |
| } |
| |
| // Now, just create the directory. If it already exists, it will fail |
| // silently. |
| sprintf(vimfiles_path, "%s\\vimfiles", vimdir_path); |
| vim_mkdir(vimfiles_path, 0755); |
| |
| printf("Creating the following directories in \"%s\":\n", vimfiles_path); |
| for (i = 0; i < TABLE_SIZE(vimfiles_subdirs); i++) |
| { |
| sprintf(tmp_dirname, "%s\\%s", vimfiles_path, vimfiles_subdirs[i]); |
| printf(" %s", vimfiles_subdirs[i]); |
| vim_mkdir(tmp_dirname, 0755); |
| } |
| printf("\n"); |
| } |
| |
| /* |
| * Add the creation of runtime files to the setup sequence. |
| */ |
| static void |
| init_directories_choice(void) |
| { |
| struct stat st; |
| char tmp_dirname[BUFSIZE]; |
| char *p; |
| int vimfiles_dir_choice; |
| |
| choices[choice_count].text = alloc(150); |
| choices[choice_count].changefunc = change_directories_choice; |
| choices[choice_count].installfunc = install_vimfilesdir; |
| choices[choice_count].active = 1; |
| |
| // Check if the "compiler" directory already exists. That's a good |
| // indication that the plugin directories were already created. |
| p = getenv("HOME"); |
| if (p != NULL) |
| { |
| vimfiles_dir_choice = (int)vimfiles_dir_home; |
| sprintf(tmp_dirname, "%s\\vimfiles\\compiler", p); |
| if (stat(tmp_dirname, &st) == 0) |
| vimfiles_dir_choice = (int)vimfiles_dir_none; |
| } |
| else |
| { |
| vimfiles_dir_choice = (int)vimfiles_dir_vim; |
| p = getenv("VIM"); |
| if (p == NULL) // No $VIM in path, use the install dir. |
| dir_remove_last(installdir, tmp_dirname); |
| else |
| strcpy(tmp_dirname, p); |
| strcat(tmp_dirname, "\\vimfiles\\compiler"); |
| if (stat(tmp_dirname, &st) == 0) |
| vimfiles_dir_choice = (int)vimfiles_dir_none; |
| } |
| |
| choices[choice_count].arg = vimfiles_dir_choice; |
| set_directories_text(choice_count); |
| ++choice_count; |
| } |
| |
| /* |
| * Setup the choices and the default values. |
| */ |
| static void |
| setup_choices(void) |
| { |
| // install the batch files |
| init_bat_choices(); |
| |
| // (over) write _vimrc file |
| init_vimrc_choices(); |
| |
| // Whether to add Vim to the popup menu |
| init_popup_choice(); |
| |
| // Whether to add Vim to the "Open With..." menu |
| init_openwith_choice(); |
| |
| // Whether to add Vim to the Start Menu. |
| init_startmenu_choice(); |
| |
| // Whether to add shortcuts to the Desktop. |
| init_shortcut_choices(); |
| |
| // Whether to create the runtime directories. |
| init_directories_choice(); |
| } |
| |
| static void |
| print_cmd_line_help(void) |
| { |
| printf("Vim installer non-interactive command line arguments:\n"); |
| printf("\n"); |
| printf("-create-batfiles [vim gvim evim view gview vimdiff gvimdiff]\n"); |
| printf(" Create .bat files for Vim variants in the Windows directory.\n"); |
| printf("-create-vimrc\n"); |
| printf(" Create a default _vimrc file if one does not already exist.\n"); |
| printf("-vimrc-remap [no|win]\n"); |
| printf(" Remap keys when creating a default _vimrc file.\n"); |
| printf("-vimrc-behave [unix|mswin|default]\n"); |
| printf(" Set mouse behavior when creating a default _vimrc file.\n"); |
| printf("-vimrc-compat [vi|vim|defaults|all]\n"); |
| printf(" Set Vi compatibility when creating a default _vimrc file.\n"); |
| printf("-install-popup\n"); |
| printf(" Install the Edit-with-Vim context menu entry\n"); |
| printf("-install-openwith\n"); |
| printf(" Add Vim to the \"Open With...\" context menu list\n"); |
| printf("-add-start-menu"); |
| printf(" Add Vim to the start menu\n"); |
| printf("-install-icons"); |
| printf(" Create icons for gVim executables on the desktop\n"); |
| printf("-create-directories [vim|home]\n"); |
| printf(" Create runtime directories to drop plugins into; in the $VIM\n"); |
| printf(" or $HOME directory\n"); |
| printf("-register-OLE"); |
| printf(" Ignored\n"); |
| printf("\n"); |
| } |
| |
| /* |
| * Setup installation choices based on command line switches |
| */ |
| static void |
| command_line_setup_choices(int argc, char **argv) |
| { |
| int i, j; |
| |
| for (i = 1; i < argc; i++) |
| { |
| if (strcmp(argv[i], "-create-batfiles") == 0) |
| { |
| if (i + 1 == argc) |
| continue; |
| while (argv[i + 1][0] != '-' && i < argc) |
| { |
| i++; |
| for (j = 1; j < TARGET_COUNT; ++j) |
| if ((targets[j].exenamearg[0] == 'g' ? has_gvim : has_vim) |
| && strcmp(argv[i], targets[j].name) == 0) |
| { |
| init_bat_choice(j); |
| break; |
| } |
| if (j == TARGET_COUNT) |
| printf("%s is not a valid choice for -create-batfiles\n", |
| argv[i]); |
| |
| if (i + 1 == argc) |
| break; |
| } |
| } |
| else if (strcmp(argv[i], "-create-vimrc") == 0) |
| { |
| // Setup default vimrc choices. If there is already a _vimrc file, |
| // it will NOT be overwritten. |
| init_vimrc_choices(); |
| } |
| else if (strcmp(argv[i], "-vimrc-remap") == 0) |
| { |
| if (i + 1 == argc) |
| break; |
| i++; |
| if (strcmp(argv[i], "no") == 0) |
| remap_choice = remap_no; |
| else if (strcmp(argv[i], "win") == 0) |
| remap_choice = remap_win; |
| } |
| else if (strcmp(argv[i], "-vimrc-behave") == 0) |
| { |
| if (i + 1 == argc) |
| break; |
| i++; |
| if (strcmp(argv[i], "unix") == 0) |
| mouse_choice = mouse_xterm; |
| else if (strcmp(argv[i], "mswin") == 0) |
| mouse_choice = mouse_mswin; |
| else if (strcmp(argv[i], "default") == 0) |
| mouse_choice = mouse_default; |
| } |
| else if (strcmp(argv[i], "-vimrc-compat") == 0) |
| { |
| if (i + 1 == argc) |
| break; |
| i++; |
| if (strcmp(argv[i], "vi") == 0) |
| compat_choice = compat_vi; |
| else if (strcmp(argv[i], "vim") == 0) |
| compat_choice = compat_vim; |
| else if (strcmp(argv[i], "defaults") == 0) |
| compat_choice = compat_some_enhancements; |
| else if (strcmp(argv[i], "all") == 0) |
| compat_choice = compat_all_enhancements; |
| } |
| else if (strcmp(argv[i], "-install-popup") == 0) |
| { |
| init_popup_choice(); |
| } |
| else if (strcmp(argv[i], "-install-openwith") == 0) |
| { |
| init_openwith_choice(); |
| } |
| else if (strcmp(argv[i], "-add-start-menu") == 0) |
| { |
| init_startmenu_choice(); |
| } |
| else if (strcmp(argv[i], "-install-icons") == 0) |
| { |
| init_shortcut_choices(); |
| } |
| else if (strcmp(argv[i], "-create-directories") == 0) |
| { |
| int vimfiles_dir_choice = (int)vimfiles_dir_none; |
| |
| init_directories_choice(); |
| if (argv[i + 1][0] != '-') |
| { |
| i++; |
| if (strcmp(argv[i], "vim") == 0) |
| vimfiles_dir_choice = (int)vimfiles_dir_vim; |
| else if (strcmp(argv[i], "home") == 0) |
| { |
| if (homedir == NULL) // No $HOME in environment |
| vimfiles_dir_choice = (int)vimfiles_dir_none; |
| else |
| vimfiles_dir_choice = (int)vimfiles_dir_home; |
| } |
| else |
| { |
| printf("Unknown argument for -create-directories: %s\n", |
| argv[i]); |
| print_cmd_line_help(); |
| } |
| } |
| else // No choice specified, default to vim directory |
| vimfiles_dir_choice = (int)vimfiles_dir_vim; |
| choices[choice_count - 1].arg = vimfiles_dir_choice; |
| } |
| else if (strcmp(argv[i], "-register-OLE") == 0) |
| { |
| // This is always done when gvim is found |
| } |
| else // Unknown switch |
| { |
| printf("Got unknown argument argv[%d] = %s\n", i, argv[i]); |
| print_cmd_line_help(); |
| } |
| } |
| } |
| |
| |
| /* |
| * Show a few screens full of helpful information. |
| */ |
| static void |
| show_help(void) |
| { |
| static char *(items[]) = |
| { |
| "Installing .bat files\n" |
| "---------------------\n" |
| "The vim.bat file is written in one of the directories in $PATH.\n" |
| "This makes it possible to start Vim from the command line.\n" |
| "If vim.exe can be found in $PATH, the choice for vim.bat will not be\n" |
| "present. It is assumed you will use the existing vim.exe.\n" |
| "If vim.bat can already be found in $PATH this is probably for an old\n" |
| "version of Vim (but this is not checked!). You can overwrite it.\n" |
| "If no vim.bat already exists, you can select one of the directories in\n" |
| "$PATH for creating the batch file, or disable creating a vim.bat file.\n" |
| "\n" |
| "If you choose not to create the vim.bat file, Vim can still be executed\n" |
| "in other ways, but not from the command line.\n" |
| "\n" |
| "The same applies to choices for gvim, evim, (g)view, and (g)vimdiff.\n" |
| "The first item can be used to change the path for all of them.\n" |
| , |
| "Creating a _vimrc file\n" |
| "----------------------\n" |
| "The _vimrc file is used to set options for how Vim behaves.\n" |
| "The install program can create a _vimrc file with a few basic choices.\n" |
| "You can edit this file later to tune your preferences.\n" |
| "If you already have a _vimrc or .vimrc file it can be overwritten.\n" |
| "Don't do that if you have made changes to it.\n" |
| , |
| "Vim features\n" |
| "------------\n" |
| "(this choice is only available when creating a _vimrc file)\n" |
| "1. Vim can run in Vi-compatible mode. Many nice Vim features are then\n" |
| " disabled. Only choose Vi-compatible if you really need full Vi\n" |
| " compatibility.\n" |
| "2. Vim runs in not-Vi-compatible mode. Vim is still mostly Vi compatible,\n" |
| " but adds nice features like multi-level undo.\n" |
| "3. Running Vim with some enhancements is useful when you want some of\n" |
| " the nice Vim features, but have a slow computer and want to keep it\n" |
| " really fast.\n" |
| "4. Syntax highlighting shows many files in color. Not only does this look\n" |
| " nice, it also makes it easier to spot errors and you can work faster.\n" |
| " The other features include editing compressed files.\n" |
| , |
| "Windows key mapping\n" |
| "-------------------\n" |
| "(this choice is only available when creating a _vimrc file)\n" |
| "Under MS-Windows the CTRL-C key copies text to the clipboard and CTRL-V\n" |
| "pastes text from the clipboard. There are a few more keys like these.\n" |
| "Unfortunately, in Vim these keys normally have another meaning.\n" |
| "1. Choose to have the keys like they normally are in Vim (useful if you\n" |
| " also use Vim on other systems).\n" |
| "2. Choose to have the keys work like they are used on MS-Windows (useful\n" |
| " if you mostly work on MS-Windows).\n" |
| , |
| "Mouse use\n" |
| "---------\n" |
| "(this choice is only available when creating a _vimrc file)\n" |
| "The right mouse button can be used in two ways:\n" |
| "1. The Unix way is to extend an existing selection. The popup menu is\n" |
| " not available.\n" |
| "2. The MS-Windows way is to show a popup menu, which allows you to\n" |
| " copy/paste text, undo/redo, etc. Extending the selection can still be\n" |
| " done by keeping SHIFT pressed while using the left mouse button\n" |
| , |
| "Edit-with-Vim context menu entry\n" |
| "--------------------------------\n" |
| "(this choice is only available when gvim.exe and gvimext.dll are present)\n" |
| "You can associate different file types with Vim, so that you can (double)\n" |
| "click on a file to edit it with Vim. This means you have to individually\n" |
| "select each file type.\n" |
| "An alternative is the option offered here: Install an \"Edit with Vim\"\n" |
| "entry in the popup menu for the right mouse button. This means you can\n" |
| "edit any file with Vim.\n" |
| , |
| "\"Open With...\" context menu entry\n" |
| "--------------------------------\n" |
| "(this choice is only available when gvim.exe is present)\n" |
| "This option adds Vim to the \"Open With...\" entry in the popup menu for\n" |
| "the right mouse button. This also makes it possible to edit HTML files\n" |
| "directly from Internet Explorer.\n" |
| , |
| "Add Vim to the Start menu\n" |
| "-------------------------\n" |
| "In Windows 95 and later, Vim can be added to the Start menu. This will\n" |
| "create a submenu with an entry for vim, gvim, evim, vimdiff, etc..\n" |
| , |
| "Icons on the desktop\n" |
| "--------------------\n" |
| "(these choices are only available when installing gvim)\n" |
| "In Windows 95 and later, shortcuts (icons) can be created on the Desktop.\n" |
| , |
| "Create plugin directories\n" |
| "-------------------------\n" |
| "Plugin directories allow extending Vim by dropping a file into a directory.\n" |
| "This choice allows creating them in $HOME (if you have a home directory) or\n" |
| "$VIM (used for everybody on the system).\n" |
| , |
| NULL |
| }; |
| int i; |
| int c; |
| |
| rewind(stdin); |
| printf("\n"); |
| for (i = 0; items[i] != NULL; ++i) |
| { |
| puts(items[i]); |
| printf("Hit Enter to continue, b (back) or q (quit help): "); |
| c = getchar(); |
| rewind(stdin); |
| if (c == 'b' || c == 'B') |
| { |
| if (i == 0) |
| --i; |
| else |
| i -= 2; |
| } |
| if (c == 'q' || c == 'Q') |
| break; |
| printf("\n"); |
| } |
| } |
| |
| /* |
| * Install the choices. |
| */ |
| static void |
| install(void) |
| { |
| int i; |
| |
| // Install the selected choices. |
| for (i = 0; i < choice_count; ++i) |
| if (choices[i].installfunc != NULL && choices[i].active) |
| (choices[i].installfunc)(i); |
| |
| // Add some entries to the registry, if needed. |
| if (install_popup |
| || install_openwith |
| || (need_uninstall_entry && interactive) |
| || !interactive) |
| install_registry(); |
| |
| // Register gvim with OLE. |
| if (has_gvim) |
| install_OLE_register(); |
| } |
| |
| /* |
| * request_choice |
| */ |
| static void |
| request_choice(void) |
| { |
| int i; |
| |
| printf("\n\nInstall will do for you:\n"); |
| for (i = 0; i < choice_count; ++i) |
| if (choices[i].active) |
| printf("%2d %s\n", i + 1, choices[i].text); |
| printf("To change an item, enter its number\n\n"); |
| printf("Enter item number, h (help), d (do it) or q (quit): "); |
| } |
| |
| int |
| main(int argc, char **argv) |
| { |
| int i; |
| char buf[BUFSIZE]; |
| |
| /* |
| * Run interactively if there are no command line arguments. |
| */ |
| if (argc > 1) |
| interactive = 0; |
| else |
| interactive = 1; |
| |
| // Initialize this program. |
| do_inits(argv); |
| init_homedir(); |
| |
| if (argc > 1 && strcmp(argv[1], "-uninstall-check") == 0) |
| { |
| // Only check for already installed Vims. Used by NSIS installer. |
| i = uninstall_check(1); |
| |
| // Find the value of $VIM, because NSIS isn't able to do this by |
| // itself. |
| get_vim_env(); |
| |
| // When nothing found exit quietly. If something found wait for |
| // a little while, so that the user can read the messages. |
| if (i && _isatty(1)) |
| sleep(3); |
| exit(0); |
| } |
| |
| printf("This program sets up the installation of Vim " |
| VIM_VERSION_MEDIUM "\n\n"); |
| |
| // Check if the user unpacked the archives properly. |
| check_unpack(); |
| |
| // Check for already installed Vims. |
| if (interactive) |
| uninstall_check(0); |
| |
| // Find out information about the system. |
| inspect_system(); |
| |
| if (interactive) |
| { |
| // Setup all the choices. |
| setup_choices(); |
| |
| // Let the user change choices and finally install (or quit). |
| for (;;) |
| { |
| request_choice(); |
| rewind(stdin); |
| if (scanf("%99s", buf) == 1) |
| { |
| if (isdigit(buf[0])) |
| { |
| // Change a choice. |
| i = atoi(buf); |
| if (i > 0 && i <= choice_count && choices[i - 1].active) |
| (choices[i - 1].changefunc)(i - 1); |
| else |
| printf("\nIllegal choice\n"); |
| } |
| else if (buf[0] == 'h' || buf[0] == 'H') |
| { |
| // Help |
| show_help(); |
| } |
| else if (buf[0] == 'd' || buf[0] == 'D') |
| { |
| // Install! |
| install(); |
| printf("\nThat finishes the installation. Happy Vimming!\n"); |
| break; |
| } |
| else if (buf[0] == 'q' || buf[0] == 'Q') |
| { |
| // Quit |
| printf("\nExiting without anything done\n"); |
| break; |
| } |
| else |
| printf("\nIllegal choice\n"); |
| } |
| } |
| printf("\n"); |
| myexit(0); |
| } |
| else |
| { |
| /* |
| * Run non-interactive - setup according to the command line switches |
| */ |
| command_line_setup_choices(argc, argv); |
| install(); |
| |
| // Avoid that the user has to hit Enter, just wait a little bit to |
| // allow reading the messages. |
| sleep(2); |
| } |
| |
| return 0; |
| } |