2 * @file OptionsInit.cpp
4 * @brief Options initialisation.
8 #include "OptionsDef.h"
9 #include "OptionsMgr.h"
10 #include "RegOptionsMgr.h"
11 #include "OptionsDiffOptions.h"
12 #include "OptionsDiffColors.h"
13 #include "OptionsFont.h"
14 #include "DiffWrapper.h" // CMP_CONTENT
16 #include "SourceControl.h"
18 #include "Environment.h"
19 #include "Constants.h"
21 // Functions to copy values set by installer from HKLM to HKCU.
22 static void CopyHKLMValues();
23 static bool OpenHKLM(HKEY *key, LPCTSTR relpath = NULL);
24 static bool OpenHKCU(HKEY *key, LPCTSTR relpath = NULL);
25 static bool IsFirstRun(HKEY key);
26 static void CopyFromLMtoCU(HKEY lmKey, HKEY cuKey, LPCTSTR valname);
27 static void ResetFirstRun(HKEY key);
33 * @brief Initialise options and set default values.
35 * @note Remember default values are what users see first time
36 * using WinMerge and many users never change them. So pick
37 * default values carefully!
39 void Init(COptionsMgr *pOptions)
41 // Copy some values from HKLM to HKCU
44 static_cast<CRegOptionsMgr *>(pOptions)->SetRegRootKey(_T("Thingamahoochie\\WinMerge\\"));
46 LANGID LangId = GetUserDefaultLangID();
47 if (PRIMARYLANGID(LangId) == LANG_JAPANESE)
49 // Default language to Japanese unless installer set it otherwise
50 pOptions->InitOption(OPT_SELECTED_LANGUAGE, 0x411);
54 // Default language to English unless installer set it otherwise
55 pOptions->InitOption(OPT_SELECTED_LANGUAGE, 0x409);
58 // Initialise options (name, default value)
59 pOptions->InitOption(OPT_SHOW_UNIQUE_LEFT, true);
60 pOptions->InitOption(OPT_SHOW_UNIQUE_MIDDLE, true);
61 pOptions->InitOption(OPT_SHOW_UNIQUE_RIGHT, true);
62 pOptions->InitOption(OPT_SHOW_DIFFERENT, true);
63 pOptions->InitOption(OPT_SHOW_IDENTICAL, true);
64 pOptions->InitOption(OPT_SHOW_BINARIES, true);
65 pOptions->InitOption(OPT_SHOW_SKIPPED, false);
67 pOptions->InitOption(OPT_SHOW_TOOLBAR, true);
68 pOptions->InitOption(OPT_SHOW_STATUSBAR, true);
69 pOptions->InitOption(OPT_SHOW_TABBAR, true);
70 pOptions->InitOption(OPT_TOOLBAR_SIZE, GetSystemMetrics(SM_CXSMICON) < 24 ? 0 : 1);
71 pOptions->InitOption(OPT_RESIZE_PANES, false);
73 pOptions->InitOption(OPT_SYNTAX_HIGHLIGHT, true);
74 pOptions->InitOption(OPT_WORDWRAP, false);
75 pOptions->InitOption(OPT_VIEW_LINENUMBERS, false);
76 pOptions->InitOption(OPT_VIEW_WHITESPACE, false);
77 pOptions->InitOption(OPT_CONNECT_MOVED_BLOCKS, 0);
78 pOptions->InitOption(OPT_SCROLL_TO_FIRST, false);
79 pOptions->InitOption(OPT_VERIFY_OPEN_PATHS, true);
80 pOptions->InitOption(OPT_AUTO_COMPLETE_SOURCE, (int)1);
81 pOptions->InitOption(OPT_VIEW_FILEMARGIN, false);
82 pOptions->InitOption(OPT_DIFF_CONTEXT, (int)-1);
83 pOptions->InitOption(OPT_SPLIT_HORIZONTALLY, false);
85 pOptions->InitOption(OPT_WORDDIFF_HIGHLIGHT, true);
86 pOptions->InitOption(OPT_BREAK_SEPARATORS, _T(".,:;?[](){}<>`'!\"#$%&^~\\|@+-*/"));
88 pOptions->InitOption(OPT_BACKUP_FOLDERCMP, false);
89 pOptions->InitOption(OPT_BACKUP_FILECMP, true);
90 pOptions->InitOption(OPT_BACKUP_LOCATION, (int)0);
91 pOptions->InitOption(OPT_BACKUP_GLOBALFOLDER, _T(""));
92 pOptions->InitOption(OPT_BACKUP_ADD_BAK, true);
93 pOptions->InitOption(OPT_BACKUP_ADD_TIME, false);
95 pOptions->InitOption(OPT_DIRVIEW_SORT_COLUMN, (int)-1);
96 pOptions->InitOption(OPT_DIRVIEW_SORT_COLUMN3, (int)-1);
97 pOptions->InitOption(OPT_DIRVIEW_SORT_ASCENDING, true);
98 pOptions->InitOption(OPT_SHOW_SELECT_FILES_AT_STARTUP, false);
99 pOptions->InitOption(OPT_DIRVIEW_EXPAND_SUBDIRS, false);
101 pOptions->InitOption(OPT_AUTOMATIC_RESCAN, false);
102 pOptions->InitOption(OPT_ALLOW_MIXED_EOL, false);
103 pOptions->InitOption(OPT_TAB_SIZE, (int)4);
104 pOptions->InitOption(OPT_TAB_TYPE, (int)0); // 0 means tabs inserted
106 pOptions->InitOption(OPT_EXT_EDITOR_CMD, paths_ConcatPath(env_GetWindowsDirectory(), _T("NOTEPAD.EXE")));
107 pOptions->InitOption(OPT_USE_RECYCLE_BIN, true);
108 pOptions->InitOption(OPT_SINGLE_INSTANCE, false);
109 pOptions->InitOption(OPT_MERGE_MODE, false);
110 // OPT_WORDDIFF_HIGHLIGHT is initialized above
111 pOptions->InitOption(OPT_BREAK_ON_WORDS, false);
112 pOptions->InitOption(OPT_BREAK_TYPE, 1);
114 pOptions->InitOption(OPT_CLOSE_WITH_ESC, true);
115 pOptions->InitOption(OPT_CLOSE_WITH_OK, false);
116 pOptions->InitOption(OPT_IGNORE_SMALL_FILETIME, false);
117 pOptions->InitOption(OPT_ASK_MULTIWINDOW_CLOSE, false);
118 pOptions->InitOption(OPT_PRESERVE_FILETIMES, false);
119 pOptions->InitOption(OPT_TREE_MODE, false);
121 pOptions->InitOption(OPT_CMP_METHOD, (int)CMP_CONTENT);
122 pOptions->InitOption(OPT_CMP_MOVED_BLOCKS, false);
123 pOptions->InitOption(OPT_CMP_MATCH_SIMILAR_LINES, false);
124 pOptions->InitOption(OPT_CMP_STOP_AFTER_FIRST, false);
125 pOptions->InitOption(OPT_CMP_QUICK_LIMIT, 4 * 1024 * 1024); // 4 Megs
126 pOptions->InitOption(OPT_CMP_WALK_UNIQUE_DIRS, false);
127 pOptions->InitOption(OPT_CMP_IGNORE_REPARSE_POINTS, false);
128 pOptions->InitOption(OPT_CMP_IGNORE_CODEPAGE, true);
129 pOptions->InitOption(OPT_CMP_INCLUDE_SUBDIRS, true);
131 pOptions->InitOption(OPT_CMP_BIN_FILEPATTERNS, _T("*.bin;*.frx"));
133 pOptions->InitOption(OPT_CMP_IMG_FILEPATTERNS, _T("*.bmp;*.cut;*.dds;*.exr;*.g3;*.gif;*.hdr;*.ico;*.iff;*.lbm;*.j2k;*.j2c;*.jng;*.jp2;*.jpg;*.jif;*.jpeg;*.jpe;*.jxr;*.wdp;*.hdp;*.koa;*.mng;*.pcd;*.pcx;*.pfm;*.pct;*.pict;*.pic;*.png;*.pbm;*.pgm;*.ppm;*.psd;*.ras;*.sgi;*.rgb;*.rgba;*.bw;*.tga;*.targa;*.tif;*.tiff;*.wap;*.wbmp;*.wbm;*.webp;*.xbm;*.xpm"));
134 pOptions->InitOption(OPT_CMP_IMG_SHOWDIFFERENCES, true);
135 pOptions->InitOption(OPT_CMP_IMG_OVERLAYMOVE, 0);
136 pOptions->InitOption(OPT_CMP_IMG_OVERLAYALPHA, 30);
137 pOptions->InitOption(OPT_CMP_IMG_ZOOM, 1000);
138 pOptions->InitOption(OPT_CMP_IMG_USEBACKCOLOR, true);
139 pOptions->InitOption(OPT_CMP_IMG_BACKCOLOR, 0xFFFFFF);
140 pOptions->InitOption(OPT_CMP_IMG_DIFFBLOCKSIZE, 8);
141 pOptions->InitOption(OPT_CMP_IMG_DIFFCOLORALPHA, 70);
142 pOptions->InitOption(OPT_CMP_IMG_THRESHOLD, 0);
144 pOptions->InitOption(OPT_PROJECTS_PATH, _T(""));
145 pOptions->InitOption(OPT_USE_SYSTEM_TEMP_PATH, true);
146 pOptions->InitOption(OPT_CUSTOM_TEMP_PATH, _T(""));
148 pOptions->InitOption(OPT_LINEFILTER_ENABLED, false);
149 pOptions->InitOption(OPT_FILEFILTER_CURRENT, _T("*.*"));
150 // CMainFrame initializes this when it is empty.
151 pOptions->InitOption(OPT_FILTER_USERPATH, paths_ConcatPath(env_GetMyDocuments(), DefaultRelativeFilterPath));
152 pOptions->InitOption(OPT_FILEFILTER_SHARED, false);
154 pOptions->InitOption(OPT_CP_DEFAULT_MODE, (int)0);
155 pOptions->InitOption(OPT_CP_DEFAULT_CUSTOM, (int)GetACP());
157 if (PRIMARYLANGID(LangId) == LANG_JAPANESE)
158 pOptions->InitOption(OPT_CP_DETECT, (int)(50932 << 16) | 3);
159 else if (LangId == MAKELANGID(LANG_CHINESE, SUBLANG_CHINESE_SIMPLIFIED))
160 pOptions->InitOption(OPT_CP_DETECT, (int)(50936 << 16) | 3);
161 else if (PRIMARYLANGID(LangId) == LANG_KOREAN)
162 pOptions->InitOption(OPT_CP_DETECT, (int)(50949 << 16) | 3);
163 else if (LangId == MAKELANGID(LANG_CHINESE, SUBLANG_CHINESE_TRADITIONAL))
164 pOptions->InitOption(OPT_CP_DETECT, (int)(50950 << 16) | 3);
166 pOptions->InitOption(OPT_CP_DETECT, (int)(50001 << 16) | 3);
168 pOptions->InitOption(OPT_VCS_SYSTEM, SourceControl::VCS_NONE);
169 pOptions->InitOption(OPT_VSS_PATH, _T(""));
170 pOptions->InitOption(OPT_VSS_DATABASE, _T(""));
171 pOptions->InitOption(OPT_VSS_PROJECT, _T(""));
172 pOptions->InitOption(OPT_VSS_USER, _T(""));
174 pOptions->InitOption(OPT_ARCHIVE_ENABLE, 1); // Enable by default
175 pOptions->InitOption(OPT_ARCHIVE_PROBETYPE, false);
177 pOptions->InitOption(OPT_PLUGINS_ENABLED, true);
178 pOptions->InitOption(OPT_PLUGINS_DISABLED_LIST, _T(""));
180 pOptions->InitOption(OPT_TABBAR_AUTO_MAXWIDTH, true);
182 pOptions->InitOption(OPT_MRU_MAX, 9);
184 pOptions->InitOption(OPT_CURRENT_VERSION_URL, CurrentVersionURL);
185 pOptions->InitOption(OPT_DOWNLOAD_URL, DownloadUrl);
187 Options::DiffOptions::SetDefaults(pOptions);
188 Options::DiffColors::SetDefaults(pOptions);
189 Options::Font::SetDefaults(pOptions);
195 * @brief Copy some HKLM values to HKCU.
196 * The installer sets HKLM values for "all users". This function copies
197 * few of those values for "user" values. E.g. enabling ShellExtension
198 * initially for user is done by this function.
200 static void CopyHKLMValues()
204 if (OpenHKLM(&LMKey))
206 if (OpenHKCU(&CUKey))
208 CopyFromLMtoCU(LMKey, CUKey, _T("ContextMenuEnabled"));
209 CopyFromLMtoCU(LMKey, CUKey, _T("Executable"));
214 if (OpenHKLM(&LMKey, _T("Locale")))
216 if (OpenHKCU(&CUKey, _T("Locale")))
218 CopyFromLMtoCU(LMKey, CUKey, _T("LanguageId"));
226 * @brief Open HKLM registry key.
227 * @param [out] key Pointer to open HKLM key.
228 * @param [in] relpath Relative registry path (to WinMerge reg path) to open, or NULL.
229 * @return true if opening succeeded.
231 static bool OpenHKLM(HKEY *key, LPCTSTR relpath)
233 TCHAR valuename[256];
235 wsprintf(valuename, _T("%s\\%s"), RegDir, relpath);
237 lstrcpy(valuename, RegDir);
238 LONG retval = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
239 valuename, 0, KEY_READ, key);
240 if (retval == ERROR_SUCCESS)
248 * @brief Open HKCU registry key.
249 * Opens the HKCU key for WinMerge. If the key does not exist, creates one.
250 * @param [out] key Pointer to open HKCU key.
251 * @param [in] relpath Relative registry path (to WinMerge reg path) to open, or NULL.
252 * @return true if opening succeeded.
254 static bool OpenHKCU(HKEY *key, LPCTSTR relpath)
256 TCHAR valuename[256];
258 wsprintf(valuename, _T("%s\\%s"), RegDir, relpath);
260 lstrcpy(valuename, RegDir);
261 LONG retval = RegOpenKeyEx(HKEY_CURRENT_USER,
262 valuename, 0, KEY_ALL_ACCESS, key);
263 if (retval == ERROR_SUCCESS)
267 else if (retval == ERROR_FILE_NOT_FOUND)
269 retval = RegCreateKeyEx(HKEY_CURRENT_USER,
270 valuename, 0, NULL, 0, KEY_ALL_ACCESS, NULL, key, NULL);
271 if (retval == ERROR_SUCCESS)
278 * @brief Copy value from HKLM to HKCU.
279 * @param [in] lmKey HKLM key from where to copy.
280 * @param [in] cuKey HKCU key to where to copy.
281 * @param [in] valname Name of the value to copy.
283 static void CopyFromLMtoCU(HKEY lmKey, HKEY cuKey, LPCTSTR valname)
286 LONG retval = RegQueryValueEx(cuKey, valname, 0, NULL, NULL, &len);
287 if (retval == ERROR_FILE_NOT_FOUND)
289 retval = RegQueryValueEx(lmKey, valname, 0, NULL, NULL, &len);
290 if (retval == ERROR_SUCCESS)
293 std::vector<BYTE> buf(len);
294 retval = RegQueryValueEx(lmKey, valname, 0, &type, &buf[0], &len);
295 if (retval == ERROR_SUCCESS)
297 retval = RegSetValueEx(cuKey, valname , 0, type,