OSDN Git Service

Rename m_XkeymacsData m_Data, LoadRegistryData LoadData, and
[xkeymacs/xkeymacs.git] / xkeymacs / profile.cpp
1 // Profile.cpp: implementation of the CProfile class\r
2 //\r
3 //////////////////////////////////////////////////////////////////////\r
4 \r
5 #include "stdafx.h"\r
6 #include "xkeymacs.h"\r
7 #include "Profile.h"\r
8 #include "MainFrm.h"\r
9 #include "DotXkeymacs.h"\r
10 #include <Imm.h>\r
11 #include <Shlwapi.h>\r
12 #include <TlHelp32.h>\r
13 \r
14 \r
15 #ifdef _DEBUG\r
16 #undef THIS_FILE\r
17 static char THIS_FILE[]=__FILE__;\r
18 #define new DEBUG_NEW\r
19 #endif\r
20 \r
21 struct CommandTypeName\r
22 {\r
23         int nCommandType;\r
24         LPCTSTR szCommandTypeName;\r
25 };\r
26 \r
27 static const CommandTypeName CommandTypes[] = {\r
28         {NONE,                                                  _T("")},\r
29         {SHIFT,                                                 _T("Shift+")},\r
30         {CONTROL,                                               _T("Ctrl+")},\r
31         {CONTROL+SHIFT,                                 _T("Ctrl+Shift+")},\r
32         {META,                                                  _T("Meta+")},\r
33         {META+SHIFT,                                    _T("Meta+Shift+")},\r
34         {CONTROL+META,                                  _T("Ctrl+Meta+")},\r
35         {CONTROL+META+SHIFT,                    _T("Ctrl+Meta+Shift+")},\r
36         {CONTROLX+NONE,                                 _T("Ctrl+X ")},\r
37         {CONTROLX+SHIFT,                                _T("Ctrl+X Shift+")},\r
38         {CONTROLX+CONTROL,                              _T("Ctrl+X Ctrl+")},\r
39         {CONTROLX+CONTROL+SHIFT,                _T("Ctrl+X Ctrl+Shift+")},\r
40         {CONTROLX+META,                                 _T("Ctrl+X Meta+")},\r
41         {CONTROLX+META+SHIFT,                   _T("Ctrl+X Meta+Shift+")},\r
42         {CONTROLX+CONTROL+META,                 _T("Ctrl+X Ctrl+Meta+")},\r
43         {CONTROLX+CONTROL+META+SHIFT,   _T("Ctrl+X Ctrl+Meta+Shift+")},\r
44 };\r
45 \r
46 static const KeyName KeyNames[] = {\r
47         {0,                             _T("")},\r
48         {VK_LBUTTON,    _T("Left mouse button")},\r
49         {VK_RBUTTON,    _T("Right mouse button")},\r
50         {VK_CANCEL,             _T("Break")},\r
51         {VK_MBUTTON,    _T("Middle mouse button")},\r
52         {0x05,                  _T("X1 mouse button")}, // VK_XBUTTON1\r
53         {0x06,                  _T("X2 mouse button")}, // VK_XBUTTON2\r
54         {0x07,                  _T("Undefined")},\r
55         {VK_BACK,               _T("Backspace")},\r
56         {VK_TAB,                _T("Tab")},\r
57         {0x0a,                  _T("Reserved")},\r
58         {0x0b,                  _T("Reserved")},\r
59         {VK_CLEAR,              _T("Clear")},\r
60         {VK_RETURN,             _T("Enter")},\r
61         {0x0e,                  _T("Undefined")},\r
62         {0x0f,                  _T("Undefined")},\r
63         {VK_SHIFT,              _T("Shift")},\r
64         {VK_CONTROL,    _T("Ctrl")},\r
65         {VK_MENU,               _T("Alt")},\r
66         {VK_PAUSE,              _T("Pause")},\r
67         {VK_CAPITAL,    _T("Caps Lock")},\r
68         {VK_KANA,               _T("Kana")},    //      {VK_HANGUEL,    "Hanguel"},     {VK_HANGUL,             "Hangul"},\r
69         {0x16,                  _T("Undefined")},\r
70         {VK_JUNJA,              _T("Junja")},\r
71         {VK_FINAL,              _T("Final")},\r
72         {VK_KANJI,              _T("Kanji")},   //      {VK_HANJA,              "Hanja"},\r
73         {0x1a,                  _T("Undefined")},\r
74         {VK_ESCAPE,             _T("Esc")},\r
75         {VK_CONVERT,    _T("Convert")},         // \95Ï\8a·\r
76         {VK_NONCONVERT, _T("Nonconvert")},      // \96³\95Ï\8a·\r
77         {VK_ACCEPT,             _T("Accept")},\r
78         {VK_MODECHANGE, _T("Mode change")},\r
79         {VK_SPACE,              _T("Space")},\r
80         {VK_PRIOR,              _T("Page Up")},\r
81         {VK_NEXT,               _T("Page Down")},\r
82         {VK_END,                _T("End")},\r
83         {VK_HOME,               _T("Home")},\r
84         {VK_LEFT,               _T("Left")},\r
85         {VK_UP,                 _T("Up")},\r
86         {VK_RIGHT,              _T("Right")},\r
87         {VK_DOWN,               _T("Down")},\r
88         {VK_SELECT,             _T("Select")},\r
89         {VK_PRINT,              _T("Print")},\r
90         {VK_EXECUTE,    _T("Execute")},\r
91         {VK_SNAPSHOT,   _T("Print Screen")},\r
92         {VK_INSERT,             _T("Ins")},\r
93         {VK_DELETE,             _T("Del")},\r
94         {VK_HELP,               _T("Help")},\r
95         {'0',                   _T("0")},\r
96         {'1',                   _T("1")},\r
97         {'2',                   _T("2")},\r
98         {'3',                   _T("3")},\r
99         {'4',                   _T("4")},\r
100         {'5',                   _T("5")},\r
101         {'6',                   _T("6")},\r
102         {'7',                   _T("7")},\r
103         {'8',                   _T("8")},\r
104         {'9',                   _T("9")},\r
105         {0x3a,                  _T("Undefined")},\r
106         {0x3b,                  _T("Undefined")},\r
107         {0x3c,                  _T("Undefined")},\r
108         {0x3d,                  _T("Undefined")},\r
109         {0x3e,                  _T("Undefined")},\r
110         {0x3f,                  _T("Undefined")},\r
111         {0x40,                  _T("Undefined")},\r
112         {'A',                   _T("A")},\r
113         {'B',                   _T("B")},\r
114         {'C',                   _T("C")},\r
115         {'D',                   _T("D")},\r
116         {'E',                   _T("E")},\r
117         {'F',                   _T("F")},\r
118         {'G',                   _T("G")},\r
119         {'H',                   _T("H")},\r
120         {'I',                   _T("I")},\r
121         {'J',                   _T("J")},\r
122         {'K',                   _T("K")},\r
123         {'L',                   _T("L")},\r
124         {'M',                   _T("M")},\r
125         {'N',                   _T("N")},\r
126         {'O',                   _T("O")},\r
127         {'P',                   _T("P")},\r
128         {'Q',                   _T("Q")},\r
129         {'R',                   _T("R")},\r
130         {'S',                   _T("S")},\r
131         {'T',                   _T("T")},\r
132         {'U',                   _T("U")},\r
133         {'V',                   _T("V")},\r
134         {'W',                   _T("W")},\r
135         {'X',                   _T("X")},\r
136         {'Y',                   _T("Y")},\r
137         {'Z',                   _T("Z")},\r
138         {VK_LWIN,               _T("Left Windows")},\r
139         {VK_RWIN,               _T("Right Windows")},\r
140         {VK_APPS,               _T("Application")},\r
141         {0x5e,                  _T("Reserved")},\r
142         {0x5f,                  _T("Sleep")},   // VK_SLEEP\r
143         {VK_NUMPAD0,    _T("Num 0")},\r
144         {VK_NUMPAD1,    _T("Num 1")},\r
145         {VK_NUMPAD2,    _T("Num 2")},\r
146         {VK_NUMPAD3,    _T("Num 3")},\r
147         {VK_NUMPAD4,    _T("Num 4")},\r
148         {VK_NUMPAD5,    _T("Num 5")},\r
149         {VK_NUMPAD6,    _T("Num 6")},\r
150         {VK_NUMPAD7,    _T("Num 7")},\r
151         {VK_NUMPAD8,    _T("Num 8")},\r
152         {VK_NUMPAD9,    _T("Num 9")},\r
153         {VK_MULTIPLY,   _T("Num *")},\r
154         {VK_ADD,                _T("Num +")},\r
155         {VK_SEPARATOR,  _T("Separator")},\r
156         {VK_SUBTRACT,   _T("Num -")},\r
157         {VK_DECIMAL,    _T("Num .")},\r
158         {VK_DIVIDE,             _T("Num /")},\r
159         {VK_F1,                 _T("F1")},\r
160         {VK_F2,                 _T("F2")},\r
161         {VK_F3,                 _T("F3")},\r
162         {VK_F4,                 _T("F4")},\r
163         {VK_F5,                 _T("F5")},\r
164         {VK_F6,                 _T("F6")},\r
165         {VK_F7,                 _T("F7")},\r
166         {VK_F8,                 _T("F8")},\r
167         {VK_F9,                 _T("F9")},\r
168         {VK_F10,                _T("F10")},\r
169         {VK_F11,                _T("F11")},\r
170         {VK_F12,                _T("F12")},\r
171         {VK_F13,                _T("F13")},\r
172         {VK_F14,                _T("F14")},\r
173         {VK_F15,                _T("F15")},\r
174         {VK_F16,                _T("F16")},\r
175         {VK_F17,                _T("F17")},\r
176         {VK_F18,                _T("F18")},\r
177         {VK_F19,                _T("F19")},\r
178         {VK_F20,                _T("F20")},\r
179         {VK_F21,                _T("F21")},\r
180         {VK_F22,                _T("F22")},\r
181         {VK_F23,                _T("F23")},\r
182         {VK_F24,                _T("F24")},\r
183         {0x88,                  _T("Unassigned")},\r
184         {0x89,                  _T("Unassigned")},\r
185         {0x8a,                  _T("Unassigned")},\r
186         {0x8b,                  _T("Unassigned")},\r
187         {0x8c,                  _T("Unassigned")},\r
188         {0x8d,                  _T("Unassigned")},\r
189         {0x8e,                  _T("Unassigned")},\r
190         {0x8f,                  _T("Unassigned")},\r
191         {VK_NUMLOCK,    _T("Num Lock")},\r
192         {VK_SCROLL,             _T("Scroll Lock")},\r
193         {0x92,                  _T("OEM specific")},\r
194         {0x93,                  _T("OEM specific")},\r
195         {0x94,                  _T("OEM specific")},\r
196         {0x95,                  _T("OEM specific")},\r
197         {0x96,                  _T("OEM specific")},\r
198         {0x97,                  _T("Unassigned")},\r
199         {0x98,                  _T("Unassigned")},\r
200         {0x99,                  _T("Unassigned")},\r
201         {0x9a,                  _T("Unassigned")},\r
202         {0x9b,                  _T("Unassigned")},\r
203         {0x9c,                  _T("Unassigned")},\r
204         {0x9d,                  _T("Unassigned")},\r
205         {0x9e,                  _T("Unassigned")},\r
206         {0x9f,                  _T("Unassigned")},\r
207         {VK_LSHIFT,             _T("Left Shift")},\r
208         {VK_RSHIFT,             _T("Right Shift")},\r
209         {VK_LCONTROL,   _T("Left Ctrl")},\r
210         {VK_RCONTROL,   _T("Right Ctrl")},\r
211         {VK_LMENU,              _T("Left Alt")},\r
212         {VK_RMENU,              _T("Right Alt")},\r
213         {0xa6,                  _T("Browser Back")},            // VK_BROWSER_BACK\r
214         {0xa7,                  _T("Browser Forward")},         // VK_BROWSER_FORWARD\r
215         {0xa8,                  _T("Browser Refresh")},         // VK_BROWSER_REFRESH\r
216         {0xa9,                  _T("Browser Stop")},            // VK_BROWSER_STOP\r
217         {0xaa,                  _T("Browser Search")},          // VK_BROWSER_SEARCH\r
218         {0xab,                  _T("Browser Favorites")},       // VK_BROWSER_FAVORITES\r
219         {0xac,                  _T("Browser Start")},           // VK_BROWSER_HOME\r
220         {0xad,                  _T("Volume Mute")},                     // VK_VOLUME_MUTE\r
221         {0xae,                  _T("Volume Down")},                     // VK_VOLUME_DOWN\r
222         {0xaf,                  _T("Volume Up")},                       // VK_VOLUME_UP\r
223         {0xb0,                  _T("Next Track")},                      // VK_MEDIA_NEXT_TRACK\r
224         {0xb1,                  _T("Previous Track")},          // VK_MEDIA_PREV_TRACK\r
225         {0xb2,                  _T("Stop Media")},                      // VK_MEDIA_STOP\r
226         {0xb3,                  _T("Play/Pause Media")},        // VK_MEDIA_PLAY_PAUSE\r
227         {0xb4,                  _T("Start Mail")},                      // VK_LAUNCH_MAIL\r
228         {0xb5,                  _T("Select Media")},            // VK_LAUNCH_MEDIA_SELECT\r
229         {0xb6,                  _T("Start Application 1")},     // VK_LAUNCH_APP1\r
230         {0xb7,                  _T("Start Application 2")},     // VK_LAUNCH_APP2\r
231         {0xb8,                  _T("Reserved")},\r
232         {0xb9,                  _T("Reserved")},\r
233         {0xba,                  _T(";")},                                       // VK_OEM_1\r
234 //      {0xba,                  _T(":")},                                       // VK_OEM_1             // for Japanese keyboard\r
235         {0xbb,                  _T("+")},                                       // VK_OEM_PLUS\r
236 //      {0xbb,                  _T(";")},                                       // VK_OEM_PLUS  // for Japanese keyboard\r
237         {0xbc,                  _T(",")},                                       // VK_OEM_COMMA\r
238         {0xbd,                  _T("-")},                                       // VK_OEM_MINUS\r
239         {0xbe,                  _T(".")},                                       // VK_OEM_PERIOD\r
240         {0xbf,                  _T("/")},                                       // VK_OEM_2\r
241         {0xc0,                  _T("`")},                                       // VK_OEM_3\r
242 //      {0xc0,                  _T("@")},                                       // VK_OEM_3             // for Japanese keyboard\r
243         {0xc1,                  _T("Reserved")},\r
244         {0xc2,                  _T("Reserved")},\r
245         {0xc3,                  _T("Reserved")},\r
246         {0xc4,                  _T("Reserved")},\r
247         {0xc5,                  _T("Reserved")},\r
248         {0xc6,                  _T("Reserved")},\r
249         {0xc7,                  _T("Reserved")},\r
250         {0xc8,                  _T("Reserved")},\r
251         {0xc9,                  _T("Reserved")},\r
252         {0xca,                  _T("Reserved")},\r
253         {0xcb,                  _T("Reserved")},\r
254         {0xcc,                  _T("Reserved")},\r
255         {0xcd,                  _T("Reserved")},\r
256         {0xce,                  _T("Reserved")},\r
257         {0xcf,                  _T("Reserved")},\r
258         {0xd0,                  _T("Reserved")},\r
259         {0xd1,                  _T("Reserved")},\r
260         {0xd2,                  _T("Reserved")},\r
261         {0xd3,                  _T("Reserved")},\r
262         {0xd4,                  _T("Reserved")},\r
263         {0xd5,                  _T("Reserved")},\r
264         {0xd6,                  _T("Reserved")},\r
265         {0xd7,                  _T("Reserved")},\r
266         {0xd8,                  _T("Unassigned")},\r
267         {0xd9,                  _T("Unassigned")},\r
268         {0xda,                  _T("Unassigned")},\r
269         {0xdb,                  _T("[")},                                       // VK_OEM_4\r
270         {0xdc,                  _T("Backslash")},                       // VK_OEM_5\r
271         {0xdd,                  _T("]")},                                       // VK_OEM_6\r
272         {0xde,                  _T("'")},                                       // VK_OEM_7\r
273         {0xdf,                  _T("OEM specific")},            // VK_OEM_8\r
274         {0xe0,                  _T("Reserved")},\r
275         {0xe1,                  _T("OEM specific")},\r
276         {0xe2,                  _T("Backslash for 106 keyboard")},      // VK_OEM_102\r
277         {0xe3,                  _T("OEM specific")},\r
278         {0xe4,                  _T("OEM specific")},\r
279         {0xe5,                  _T("Process")},                         // VK_PROCESSKEY\r
280         {0xe6,                  _T("OEM specific")},\r
281         {0xe7,                  _T("Packet")},  // VK_PACKET\r
282         {0xe8,                  _T("Unassigned")},\r
283         {0xe9,                  _T("OEM specific")},\r
284         {0xea,                  _T("OEM specific")},\r
285         {0xeb,                  _T("OEM specific")},\r
286         {0xec,                  _T("OEM specific")},\r
287         {0xed,                  _T("OEM specific")},\r
288         {0xee,                  _T("OEM specific")},\r
289         {0xef,                  _T("OEM specific")},\r
290         {0xf0,                  _T("Eisu")},                                    // \89p\90\94\r
291         {0xf1,                  _T("OEM specific")},\r
292         {0xf2,                  _T("Hiragana")},                                // \82Ð\82ç\82ª\82È\r
293         {0xf3,                  _T("Hankaku/Zenkaku 0xf3")},    // "\94¼\8ap/\91S\8ap"\r
294         {0xf4,                  _T("Hankaku/Zenkaku 0xf4")},    // "\94¼\8ap/\91S\8ap"\r
295         {0xf5,                  _T("OEM specific")},\r
296         {VK_ATTN,               _T("Attn")},\r
297         {VK_CRSEL,              _T("CrSel")},\r
298         {VK_EXSEL,              _T("ExSel")},\r
299         {VK_EREOF,              _T("Erace EOF")},\r
300         {VK_PLAY,               _T("Play")},\r
301         {VK_ZOOM,               _T("Zoom")},\r
302         {VK_NONAME,             _T("Noname")},\r
303         {VK_PA1,                _T("PA1")},\r
304         {VK_OEM_CLEAR,  _T("OEM Clear")},\r
305         {0xff,                  _T("Fn")},\r
306 };\r
307 \r
308 CData CProfile::m_Data[MAX_APP];\r
309 TASK_LIST CProfile::m_TaskList[MAX_TASKS];\r
310 DWORD CProfile::m_dwTasks;\r
311 \r
312 enum { INITIAL_SIZE     = 51200 };\r
313 enum { EXTEND_SIZE      = 25600 };\r
314 \r
315 void CProfile::Item2AppName(CString *const sz)\r
316 {\r
317         if (IsTheString(*sz, IDS_DEFAULT_TITLE)) {\r
318                 sz->LoadString(IDS_DEFAULT);\r
319         }\r
320 \r
321         if (IsTheString(*sz, IDS_DIALOG_TITLE)) {\r
322                 sz->LoadString(IDS_DIALOG);\r
323         }\r
324 \r
325         int nStart, nEnd, nCount;\r
326 \r
327         nStart  = sz->ReverseFind(_T('(')) + 1;\r
328         nEnd    = sz->Find(_T(')'), nStart) - 1;\r
329         nCount  = (nEnd + 1) - nStart;\r
330         *sz             = sz->Mid(nStart, nCount);\r
331 }\r
332 \r
333 int CProfile::IsNotSameString(CComboBox *const pApplication, const CString szListItem)\r
334 {\r
335         CString szItem, szList;\r
336         szList = szListItem;\r
337         Item2AppName(&szList);\r
338 \r
339         for (int i = 0; i < pApplication->GetCount(); ++i) {\r
340                 pApplication->GetLBText(i, szItem);\r
341                 Item2AppName(&szItem);\r
342                 if (!_tcsicmp(szItem, szList)) {\r
343                         return 0;\r
344                 }\r
345         }\r
346 \r
347         return 1;\r
348 }\r
349 \r
350 int CProfile::CountSeparator(const CString szMainString, const CString szSeparator)\r
351 {\r
352         int index       = 0;\r
353         int counter     = 0;\r
354 \r
355         while ((index = szMainString.Find(szSeparator, index)) != -1) {\r
356                 ++index;\r
357                 ++counter;\r
358         }\r
359 \r
360         return counter;\r
361 }\r
362 \r
363 void CProfile::GetNthString(CString *const szAppName, const CString szWindowName, const CString szSeparator, int n)\r
364 {\r
365         int index = -1;\r
366 \r
367         while (--n) {\r
368                 index = szWindowName.Find(szSeparator, index + 1);\r
369         }\r
370 \r
371         int nStart;\r
372         if (index != -1) {\r
373                 nStart = index + szSeparator.GetLength();\r
374         } else {\r
375                 nStart = 0;\r
376         }\r
377 \r
378         int nEnd = szWindowName.Find(szSeparator, nStart);\r
379         if (nEnd == -1) {\r
380                 nEnd = szWindowName.GetLength();\r
381         }\r
382 \r
383         *szAppName = szWindowName.Mid(nStart, nEnd - nStart);\r
384 }\r
385 \r
386 void CProfile::GetAppName(CString *const szAppName, LPCTSTR pWindowName)\r
387 {\r
388         CString szWindowName(pWindowName);\r
389         CString szSeparator(MAKEINTRESOURCE(IDS_SEPARATE_WINDOWTITLE));\r
390         int nWord = CountSeparator(szWindowName, szSeparator) + 1;\r
391 \r
392         while (nWord) {\r
393                 GetNthString(szAppName, szWindowName, szSeparator, nWord);\r
394                 if (szAppName->GetAt(0) == _T('[')\r
395                  || szAppName->Find(_T('.'), 0) != -1           // for Microsoft Project\r
396                  || szAppName->Find(_T(']'), 0) != -1) {        // for the file name like [foo - bar]\r
397                         --nWord;\r
398                 } else {\r
399                         return;\r
400                 }\r
401         }\r
402 \r
403         *szAppName = szWindowName;\r
404 }\r
405 \r
406 BOOL CALLBACK CProfile::EnumWindowsProc(const HWND hWnd, const LPARAM lParam)\r
407 {\r
408         CComboBox               *pApplication   = (CComboBox*)lParam;\r
409         PTASK_LIST              pTask                   = CProfile::m_TaskList;\r
410         \r
411         TCHAR szWindowName[WINDOW_TEXT_LENGTH];\r
412         TCHAR szClassName[CLASS_NAME_LENGTH];\r
413         WINDOWPLACEMENT wpl;\r
414         \r
415         wpl.length = sizeof(WINDOWPLACEMENT);\r
416         ::GetWindowText(hWnd, szWindowName, sizeof(szWindowName));\r
417         GetClassName(hWnd, szClassName, sizeof(szClassName));\r
418 \r
419         CString szAppName;\r
420         // Get Process Name\r
421         DWORD dwProcessId = 0;\r
422         GetWindowThreadProcessId(hWnd, &dwProcessId);\r
423         for (DWORD i = 0; i < CProfile::m_dwTasks; ++i) {\r
424                 if (pTask[i].dwProcessId == dwProcessId) {\r
425 \r
426                         // Get Application Name\r
427                         if (szWindowName[0] == '\0') {\r
428                                 continue;\r
429                         }\r
430                         if (!_tcsnicmp(pTask[i].ProcessName, CString(MAKEINTRESOURCE(IDS_B2)), sizeof(pTask[i].ProcessName))) {\r
431                                 szAppName.LoadString(IDS_BECKY);\r
432                         } else if (!_tcsnicmp(pTask[i].ProcessName, CString(MAKEINTRESOURCE(IDS_EXPLORER)), sizeof(pTask[i].ProcessName))) {\r
433                                 szAppName.LoadString(IDS_PROGRAM_MANAGER);\r
434                         } else if (!_tcsnicmp(pTask[i].ProcessName, CString(MAKEINTRESOURCE(IDS_MSIMN)), sizeof(pTask[i].ProcessName))) {\r
435                                 szAppName.LoadString(IDS_OUTLOOK_EXPRESS);\r
436                         } else if (!_tcsnicmp(pTask[i].ProcessName, CString(MAKEINTRESOURCE(IDS_PROJECT)), sizeof(pTask[i].ProcessName))\r
437                                         || !_tcsnicmp(pTask[i].ProcessName, CString(MAKEINTRESOURCE(IDS_EXCEL)), sizeof(pTask[i].ProcessName))\r
438                                         || !_tcsnicmp(pTask[i].ProcessName, _T("psp.exe"), sizeof(pTask[i].ProcessName))) {\r
439                                 GetNthString(&szAppName, szWindowName, CString(MAKEINTRESOURCE(IDS_SEPARATE_WINDOWTITLE)), 1);\r
440                         } else if (!_tcsnicmp(pTask[i].ProcessName, _T("sakura.exe"), sizeof(pTask[i].ProcessName))) {\r
441                                 GetNthString(&szAppName, szWindowName, CString(MAKEINTRESOURCE(IDS_SEPARATE_WINDOWTITLE)), 2);  // '.' is included, so...\r
442                         } else if (!_tcsnicmp(pTask[i].ProcessName, CString(MAKEINTRESOURCE(IDS_MSDN)), sizeof(pTask[i].ProcessName))) {\r
443                                 szAppName = szWindowName;\r
444                         } else if (!_tcsnicmp(pTask[i].ProcessName, _T("devenv.exe"), sizeof(pTask[i].ProcessName))) {\r
445                                 szAppName.Format(_T("Microsoft Visual Studio .NET"));\r
446                         } else if (!_tcsnicmp(pTask[i].ProcessName, _T("vb6.exe"), sizeof(pTask[i].ProcessName))) {\r
447                                 szAppName.Format(_T("Microsoft Visual Basic"));\r
448                         } else if (!_tcsnicmp(pTask[i].ProcessName, _T("ssexp.exe"), sizeof(pTask[i].ProcessName))) {\r
449                                 szAppName.LoadString(IDS_VISUAL_SOURCESAFE_EXPLORER);\r
450                         } else if (!_tcsnicmp(pTask[i].ProcessName, _T("sh.exe"), sizeof(pTask[i].ProcessName))) {\r
451                                 szAppName.Format(_T("MKS Korn Shell"));\r
452                         } else if (!_tcsnicmp(pTask[i].ProcessName, _T("csh.exe"), sizeof(pTask[i].ProcessName))) {\r
453                                 szAppName.Format(_T("C Shell"));\r
454                         } else if (!_tcsnicmp(pTask[i].ProcessName, _T("vim.exe"), sizeof(pTask[i].ProcessName))) {\r
455                                 szAppName.Format(_T("VIM"));\r
456                         } else {\r
457                                 CUtils::SetCorrectApplicationName(pTask[i].ProcessName, szWindowName);\r
458                                 GetAppName(&szAppName, szWindowName);\r
459                         }\r
460                         break;\r
461                 }\r
462         }\r
463         \r
464         \r
465         if ((IsWindowVisible(hWnd))                                                                     // Is visible?\r
466          && (GetWindow(hWnd, GW_OWNER) == NULL)                                         // Is top level window?\r
467          && (lstrlen(szWindowName) > 0)                                                         // Have caption?\r
468          && (pApplication->FindString(-1, szClassName) == CB_ERR)) {// Is not same string?\r
469                 CString szListItem;\r
470                 szListItem.Format(IDS_APPLICATION_LIST_ITEM, szAppName, pTask[i].ProcessName);\r
471                 if (IsNotSameString(pApplication, szListItem)) {\r
472                         pApplication->AddString(szListItem);\r
473                 }\r
474         }\r
475         return TRUE;\r
476 }\r
477 \r
478 //////////////////////////////////////////////////////////////////////\r
479 // Construction/Destruction\r
480 //////////////////////////////////////////////////////////////////////\r
481 \r
482 CProfile::CProfile()\r
483 {\r
484 \r
485 }\r
486 \r
487 CProfile::~CProfile()\r
488 {\r
489 \r
490 }\r
491 \r
492 // This method initializes data in the registry, or retrieves and validates registry data.\r
493 // bSaveAndValidate specifies a flag that indicates \r
494 // whether registry data is being initialized (FALSE) or data is being retrieved (TRUE). \r
495 void CProfile::UpdateRegistryData(const BOOL bSaveAndValidate)\r
496 {\r
497         CString szEntry;\r
498         CString szApplicationName;\r
499         CString szApplicationTitle;\r
500         CString szWindowText;\r
501         CString szWindowTextType;\r
502 \r
503         BOOL bUseDialogSetting = FALSE;\r
504 \r
505         for (int nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
506                 // application name\r
507                 CString szSection(MAKEINTRESOURCE(IDS_REG_SECTION_APPLICATION));\r
508                 szEntry.Format(IDS_REG_ENTRY_APPLICATION, nApplicationID);\r
509                 if (bSaveAndValidate) { // retrieve\r
510                         m_Data[nApplicationID].ClearAll();\r
511                         szApplicationName = AfxGetApp()->GetProfileString(szSection, szEntry);\r
512                         if (szApplicationName.IsEmpty()) {\r
513                                 if (nApplicationID) {\r
514                                         if (!bUseDialogSetting) {\r
515                                                 szApplicationName.LoadString(IDS_DIALOG);\r
516                                                 bUseDialogSetting = TRUE;\r
517                                         } else {\r
518                                                 continue;\r
519                                         }\r
520                                 } else {\r
521                                         szApplicationName.LoadString(IDS_DEFAULT);\r
522                                 }\r
523                         } else {\r
524                                 if (szApplicationName == CString(MAKEINTRESOURCE(IDS_DIALOG))) {\r
525                                         bUseDialogSetting = TRUE;\r
526                                 }\r
527                         }\r
528                         m_Data[nApplicationID].SetApplicationName(szApplicationName);\r
529                 } else {                                // initialize\r
530                         szApplicationName = m_Data[nApplicationID].GetApplicationName();\r
531                         if (szApplicationName.IsEmpty()) {\r
532                                 continue;\r
533                         }\r
534                         AfxGetApp()->WriteProfileString(szSection, szEntry, szApplicationName);\r
535                 }\r
536 \r
537                 // application title\r
538                 szEntry.LoadString(IDS_REG_ENTRY_APPLICATOIN_TITLE);\r
539                 if (bSaveAndValidate) { // retrieve\r
540                         szApplicationTitle = AfxGetApp()->GetProfileString(szApplicationName, szEntry);\r
541                         m_Data[nApplicationID].SetApplicationTitle(szApplicationTitle);\r
542                 } else {                                // initialize\r
543                         szApplicationTitle = m_Data[nApplicationID].GetApplicationTitle();\r
544                         while (!szApplicationTitle.IsEmpty() && szApplicationTitle.GetAt(0) == _T(' ')) {\r
545                                 szApplicationTitle.Delete(0);\r
546                         }\r
547                         AfxGetApp()->WriteProfileString(szApplicationName, szEntry, szApplicationTitle);\r
548                 }\r
549 \r
550                 // window text\r
551                 szEntry.LoadString(IDS_REG_ENTRY_WINDOW_TEXT);\r
552                 if (bSaveAndValidate) { // retrieve\r
553                         szWindowText = AfxGetApp()->GetProfileString(szApplicationName, szEntry, _T("*"));\r
554                         if (szWindowText.IsEmpty()) {\r
555                                 szWindowText = _T('*');\r
556                         }\r
557                         m_Data[nApplicationID].SetWindowText(szWindowText);\r
558                 } else {                                // initialize\r
559                         szWindowText = m_Data[nApplicationID].GetWindowText();\r
560                         AfxGetApp()->WriteProfileString(szApplicationName, szEntry, szWindowText);\r
561                 }\r
562 \r
563                 // window text type\r
564                 szEntry.LoadString(IDS_REG_ENTRY_WINDOW_TEXT_TYPE);\r
565                 if (bSaveAndValidate) { // retrieve\r
566                         szWindowTextType = AfxGetApp()->GetProfileString(szApplicationName, szEntry);\r
567 \r
568                         int nWindowTextType = IDS_WINDOW_TEXT_IGNORE;\r
569                         if (szWindowTextType == CString(MAKEINTRESOURCE(IDS_WINDOW_TEXT_MATCH))) {\r
570                                 nWindowTextType = IDS_WINDOW_TEXT_MATCH;\r
571                         } else if (szWindowTextType == CString(MAKEINTRESOURCE(IDS_WINDOW_TEXT_MATCH_FORWARD))) {\r
572                                 nWindowTextType = IDS_WINDOW_TEXT_MATCH_FORWARD;\r
573                         } else if (szWindowTextType == CString(MAKEINTRESOURCE(IDS_WINDOW_TEXT_MATCH_BACKWARD))) {\r
574                                 nWindowTextType = IDS_WINDOW_TEXT_MATCH_BACKWARD;\r
575                         } else if (szWindowTextType == CString(MAKEINTRESOURCE(IDS_WINDOW_TEXT_MATCH_FULL))) {\r
576                                 nWindowTextType = IDS_WINDOW_TEXT_MATCH_FULL;\r
577                         }\r
578 \r
579                         m_Data[nApplicationID].SetWindowTextType(nWindowTextType);\r
580                 } else {                                // initialize\r
581                         szWindowTextType.LoadString(m_Data[nApplicationID].GetWindowTextType());\r
582                         AfxGetApp()->WriteProfileString(szApplicationName, szEntry, szWindowTextType);\r
583                 }\r
584 \r
585                 CString regApp(MAKEINTRESOURCE(IDS_REGSUBKEY_DATA));\r
586                 regApp += _T("\\") + szApplicationName;\r
587                 // on/off\r
588                 if (bSaveAndValidate) { // retrieve\r
589                         for (int nCommandID = 1; nCommandID < MAX_COMMAND; ++nCommandID) {\r
590                                 szEntry = CCommands::GetCommandName(nCommandID);\r
591                                 if (szEntry.IsEmpty()) {\r
592                                         break;\r
593                                 }\r
594                                 HKEY hKey = NULL;\r
595                                 CString regKey = regApp + _T("\\") + szEntry;\r
596                                 if (RegOpenKeyEx(HKEY_CURRENT_USER, regKey, 0, KEY_READ, &hKey) == ERROR_SUCCESS) {\r
597                                         // Use registry data\r
598                                         TCHAR szKeyBind[128] = {'\0'};\r
599                                         DWORD dwKeyBind = sizeof(szKeyBind);\r
600                                         FILETIME ft = {'\0'};   // not use\r
601                                         for (DWORD dwIndex = 0;\r
602                                                  RegEnumKeyEx(hKey, dwIndex, szKeyBind, &dwKeyBind, NULL, NULL, NULL, &ft) == ERROR_SUCCESS;\r
603                                                  ++dwIndex) {\r
604                                                 int nCommandType = 0;\r
605                                                 int nKey = 0;\r
606                                                 ReadKeyBind(&nCommandType, &nKey, szKeyBind);\r
607                                                 m_Data[nApplicationID].SetCommandID(nCommandType, nKey, nCommandID);\r
608                                                 memset(szKeyBind, 0, sizeof(szKeyBind));\r
609                                                 dwKeyBind = sizeof(szKeyBind);\r
610                                         }\r
611                                         RegCloseKey(hKey);\r
612                                 } else {\r
613                                         // Use default setting\r
614                                         for (int i = 0; ; ++i) {\r
615                                                 if (CCommands::GetDefaultControlID(nCommandID, i) == IDC_CO2) {\r
616                                                         continue;\r
617                                                 }\r
618 \r
619                                                 int nCommandType = CCommands::GetDefaultCommandType(nCommandID, i);\r
620                                                 int nKey = CCommands::GetDefaultCommandKey(nCommandID, i);\r
621                                                 if (nKey == 0) {\r
622                                                         break;\r
623                                                 }\r
624                                                 m_Data[nApplicationID].SetCommandID(nCommandType, nKey, nCommandID);\r
625                                         }\r
626                                 }\r
627                         }\r
628                         for (int nFunctionID = 0; nFunctionID < CDotXkeymacs::GetFunctionNumber(); ++nFunctionID) {\r
629                                 HKEY hKey = NULL;\r
630                                 CString regKey = regApp + _T("\\") + CDotXkeymacs::GetFunctionSymbol(nFunctionID);\r
631                                 if (RegOpenKeyEx(HKEY_CURRENT_USER, regKey, 0, KEY_READ, &hKey) == ERROR_SUCCESS) {\r
632                                         // Use registry data\r
633                                         CDotXkeymacs::ClearKey(nFunctionID, nApplicationID);\r
634                                         TCHAR szKeyBind[128] = {'\0'};\r
635                                         DWORD dwKeyBind = sizeof(szKeyBind);\r
636                                         FILETIME ft = {'\0'};   // not use\r
637                                         for (DWORD dwIndex = 0; RegEnumKeyEx(hKey, dwIndex, szKeyBind, &dwKeyBind, NULL, NULL, NULL, &ft) == ERROR_SUCCESS; ++dwIndex) {\r
638                                                 int nCommandType = 0;\r
639                                                 int nKey = 0;\r
640                                                 ReadKeyBind(&nCommandType, &nKey, szKeyBind);\r
641                                                 CDotXkeymacs::SetKey(nFunctionID, nApplicationID, nCommandType, nKey);\r
642 \r
643                                                 memset(szKeyBind, 0, sizeof(szKeyBind));\r
644                                                 dwKeyBind = sizeof(szKeyBind);\r
645                                         }\r
646                                         RegCloseKey(hKey);\r
647                                 }\r
648                         }\r
649                 } else {                                // initialize\r
650                         // create all commands\r
651                         for (int nCommandID = 1; nCommandID < MAX_COMMAND; ++nCommandID) {\r
652                                 szEntry = CCommands::GetCommandName(nCommandID);\r
653                                 if (szEntry.IsEmpty()) {\r
654                                         break;\r
655                                 }\r
656 \r
657                                 SaveCommand(szApplicationName, nCommandID);\r
658                         }\r
659                         for (int nCommandType = 0; nCommandType < MAX_COMMAND_TYPE; ++nCommandType) {\r
660                                 for (int nKey = 0; nKey < MAX_KEY; ++nKey) {\r
661                                         int nCommandID = m_Data[nApplicationID].GetCommandID(nCommandType, nKey);\r
662                                         SaveKeyBind(szApplicationName, nCommandID, nCommandType, nKey);\r
663                                 }\r
664                         }\r
665                         for (int nFunctionID = 0; nFunctionID < CDotXkeymacs::GetFunctionNumber(); ++nFunctionID) {\r
666                                 for (int nKeyID = 0; nKeyID < CDotXkeymacs::GetKeyNumber(nFunctionID, nApplicationID); ++nKeyID) {\r
667                                         int nCommandType = 0;\r
668                                         int nKey = 0;\r
669                                         CDotXkeymacs::GetKey(nFunctionID, nApplicationID, nKeyID, &nCommandType, &nKey);\r
670                                         SaveKeyBind(szApplicationName, CDotXkeymacs::GetFunctionSymbol(nFunctionID), nCommandType, nKey);\r
671                                 }\r
672                         }\r
673                 }\r
674 \r
675                 // kill-ring-max\r
676                 szEntry.LoadString(IDS_REG_ENTRY_KILL_RING_MAX);\r
677                 if (bSaveAndValidate) { // retrieve\r
678                         int nKillRingMax = AfxGetApp()->GetProfileInt(szApplicationName, szEntry, 1);\r
679                         m_Data[nApplicationID].SetKillRingMax(nKillRingMax);\r
680                 } else {                                // initialize\r
681                         int nKillRingMax = m_Data[nApplicationID].GetKillRingMax();\r
682                         AfxGetApp()->WriteProfileInt(szApplicationName, szEntry, nKillRingMax);\r
683                 }\r
684 \r
685                 // Use Dialog Setting\r
686                 szEntry.LoadString(IDS_REG_ENTRY_USE_DIALOG_SETTING);\r
687                 if (bSaveAndValidate) { // retrieve\r
688                         BOOL bUseDialogSetting = AfxGetApp()->GetProfileInt(szApplicationName,szEntry, 1);\r
689                         m_Data[nApplicationID].SetUseDialogSetting(bUseDialogSetting);\r
690                 } else {                                // initialize\r
691                         BOOL bUseDialogSetting = m_Data[nApplicationID].GetUseDialogSetting();\r
692                         AfxGetApp()->WriteProfileInt(szApplicationName, szEntry, bUseDialogSetting);\r
693                 }\r
694 \r
695                 // Setting Style\r
696                 szEntry.LoadString(IDS_REG_ENTRY_DISABLE_XKEYMACS);\r
697                 if (bSaveAndValidate) { // retrieve\r
698                         int nSettingStyle = SETTING_DEFAULT;\r
699                         HKEY hKey;\r
700                         if (AfxGetApp()->GetProfileInt(szApplicationName, szEntry, 0) != 0) {\r
701                                 nSettingStyle = SETTING_DISABLE;\r
702                         } else if (RegOpenKeyEx(HKEY_CURRENT_USER, regApp, 0, KEY_READ, &hKey) == ERROR_SUCCESS) {\r
703                                 RegCloseKey(hKey);\r
704                                 nSettingStyle = SETTING_SPECIFIC;\r
705                         }\r
706                         m_Data[nApplicationID].SetSettingStyle(nSettingStyle);\r
707                 } else {                                // initialize\r
708                         BOOL bDisableXkeymacs = FALSE;\r
709                         if (m_Data[nApplicationID].GetSettingStyle() == SETTING_DISABLE) {\r
710                                 bDisableXkeymacs = TRUE;\r
711                         }\r
712                         AfxGetApp()->WriteProfileInt(szApplicationName, szEntry, bDisableXkeymacs);\r
713                 }\r
714 \r
715                 // Ignore Meta Ctrl+? when it is undefined.\r
716                 szEntry.LoadString(IDC_REG_ENTRY_IGNORE_META_CTRL);\r
717                 if (bSaveAndValidate) { // retrieve\r
718                         m_Data[nApplicationID].SetIgnoreUndefinedMetaCtrl(AfxGetApp()->GetProfileInt(szApplicationName, szEntry, 0));\r
719                 } else {                                // initialize\r
720                         AfxGetApp()->WriteProfileInt(szApplicationName, szEntry, m_Data[nApplicationID].GetIgnoreUndefinedMetaCtrl());\r
721                 }\r
722 \r
723                 // Ignore C-x ? when it is undefined.\r
724                 szEntry.LoadString(IDC_REG_ENTRY_IGNORE_C_X);\r
725                 if (bSaveAndValidate) { // retrieve\r
726                         m_Data[nApplicationID].SetIgnoreUndefinedC_x(AfxGetApp()->GetProfileInt(szApplicationName, szEntry, 0));\r
727                 } else {                                // initialize\r
728                         AfxGetApp()->WriteProfileInt(szApplicationName, szEntry, m_Data[nApplicationID].GetIgnoreUndefinedC_x());\r
729                 }\r
730 \r
731                 // Enable CUA-mode\r
732                 szEntry.LoadString(IDC_REG_ENTRY_ENABLE_CUA);\r
733                 if (bSaveAndValidate) { // retrieve\r
734                         m_Data[nApplicationID].SetEnableCUA(AfxGetApp()->GetProfileInt(szApplicationName, szEntry, 0));\r
735                 } else {                                // initialize\r
736                         AfxGetApp()->WriteProfileInt(szApplicationName, szEntry, m_Data[nApplicationID].GetEnableCUA());\r
737                 }\r
738 \r
739                 // Version 3.26 compatible mode\r
740                 szEntry.LoadString(IDS_REG_ENTRY_326_COMPATIBLE);\r
741                 if (bSaveAndValidate) { // retrieve\r
742                         m_Data[nApplicationID].Set326Compatible(AfxGetApp()->GetProfileInt(szApplicationName, szEntry, 0));\r
743                 } else {                                // initialize\r
744                         AfxGetApp()->WriteProfileInt(szApplicationName, szEntry, m_Data[nApplicationID].Get326Compatible());\r
745                 }\r
746         }\r
747 }\r
748 \r
749 void CProfile::LoadData()\r
750 {\r
751         CDotXkeymacs::Load();\r
752         LevelUp();\r
753         UpdateRegistryData(TRUE);\r
754 }\r
755 \r
756 void CProfile::SaveData()\r
757 {\r
758         DeleteAllRegistryData();\r
759         UpdateRegistryData(FALSE);\r
760         SetDllData();\r
761 }\r
762 \r
763 void CProfile::SetDllData()\r
764 {\r
765         CXkeymacsDll::ClearFunctionDefinition();\r
766         for (int nFunctionID = 0; nFunctionID < CDotXkeymacs::GetFunctionNumber(); ++nFunctionID) {\r
767                 CXkeymacsDll::SetFunctionDefinition(nFunctionID, CDotXkeymacs::GetFunctionDefinition(nFunctionID));\r
768         }\r
769 \r
770         for (int nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
771 \r
772                 CString szApplicationName = m_Data[nApplicationID].GetApplicationName();\r
773 \r
774                 if (szApplicationName.IsEmpty()) {\r
775                         CXkeymacsDll::Clear(nApplicationID);\r
776                         continue;\r
777                 }\r
778 \r
779                 CXkeymacsDll::SetApplicationName(nApplicationID, szApplicationName);\r
780                 CXkeymacsDll::SetWindowText(nApplicationID, m_Data[nApplicationID].GetWindowText());\r
781                 CXkeymacsDll::SetCommandID(nApplicationID, CONTROL, 'X', 0);\r
782 \r
783                 for (int nCommandType = 0; nCommandType < MAX_COMMAND_TYPE; ++nCommandType) {\r
784                         for (int nKey = 0; nKey < MAX_KEY; ++nKey) {\r
785                                 const int nCommandID = m_Data[nApplicationID].GetCommandID(nCommandType, nKey);\r
786                                 CXkeymacsDll::SetCommandID(nApplicationID, nCommandType, nKey, nCommandID);\r
787                                 if ((nCommandType & CONTROLX) && nCommandID) {\r
788                                         CXkeymacsDll::SetCommandID(nApplicationID, CONTROL, 'X', 1);                    // Commands[1] is C-x\r
789                                 }\r
790                         }\r
791                 }\r
792 \r
793                 for (int nFunctionID = 0; nFunctionID < CDotXkeymacs::GetFunctionNumber(); ++nFunctionID) {\r
794                         for (int nKeyID = 0; nKeyID < CDotXkeymacs::GetKeyNumber(nFunctionID, nApplicationID); ++nKeyID) {\r
795                                 int nCommandType = 0;\r
796                                 int nKey = 0;\r
797                                 CDotXkeymacs::GetKey(nFunctionID, nApplicationID, nKeyID, &nCommandType, &nKey);\r
798                                 CXkeymacsDll::SetFunctionKey(nFunctionID, nApplicationID, nCommandType, nKey);\r
799                                 if (nCommandType & CONTROLX) {\r
800                                         CXkeymacsDll::SetCommandID(nApplicationID, CONTROL, 'X', 1);                    // Commands[1] is C-x\r
801                                 }\r
802                         }\r
803                 }\r
804 \r
805                 CXkeymacsDll::SetKillRingMax(nApplicationID, m_Data[nApplicationID].GetKillRingMax());\r
806                 CXkeymacsDll::SetUseDialogSetting(nApplicationID, m_Data[nApplicationID].GetUseDialogSetting());\r
807                 CXkeymacsDll::SetSettingStyle(nApplicationID, m_Data[nApplicationID].GetSettingStyle());\r
808                 CXkeymacsDll::SetIgnoreUndefinedMetaCtrl(nApplicationID, m_Data[nApplicationID].GetIgnoreUndefinedMetaCtrl());\r
809                 CXkeymacsDll::SetIgnoreUndefinedC_x(nApplicationID, m_Data[nApplicationID].GetIgnoreUndefinedC_x());\r
810                 CXkeymacsDll::SetEnableCUA(nApplicationID, m_Data[nApplicationID].GetEnableCUA());\r
811                 CXkeymacsDll::Set326Compatible(nApplicationID, m_Data[nApplicationID].Get326Compatible());\r
812         }\r
813         CXkeymacsApp *pApp = static_cast<CXkeymacsApp *>(AfxGetApp());\r
814         if (!pApp->IsWow64())\r
815                 return;\r
816         if (!CXkeymacsDll::SaveConfig())\r
817                 return;\r
818         pApp->SendIPCMessage(XKEYMACS_RELOAD);\r
819 }\r
820 \r
821 void CProfile::ReadKeyBind(int *const pnCommandType, int *const pnKey, LPCTSTR szKeyBind)\r
822 {\r
823         *pnCommandType = KeyBind2CommandType(szKeyBind);\r
824         *pnKey = KeyBind2Key(szKeyBind + _tcslen(CommandType2String(*pnCommandType)));\r
825 }\r
826 \r
827 CString CProfile::WriteKeyBind(const int nCommandType, const int nKey)\r
828 {\r
829         CString szKeyBind;\r
830         szKeyBind.Format(_T("%s%s"), CommandType2String(nCommandType), Key2String(nKey));\r
831         return szKeyBind;\r
832 }\r
833 \r
834 int CProfile::KeyBind2CommandType(LPCTSTR szKeyBind)\r
835 {\r
836         for (int nCommandType = MAX_COMMAND_TYPE - 1; nCommandType; --nCommandType) {\r
837                 if (IsCommandType(nCommandType, szKeyBind)) {\r
838                         return nCommandType;\r
839                 }\r
840         }\r
841         return NONE;\r
842 }\r
843 \r
844 int CProfile::KeyBind2Key(LPCTSTR szKey)\r
845 {\r
846         for (int nKey = 1; nKey <= 0xff; ++nKey) {\r
847                 if (!_tcscmp(szKey, Key2String(nKey))) {\r
848                         return nKey;\r
849                 }\r
850         }\r
851         return 0;\r
852 }\r
853 \r
854 LPCTSTR CProfile::CommandType2String(int nCommandType)\r
855 {\r
856         if (nCommandType < 0 || sizeof(CommandTypes) / sizeof(CommandTypes[0]) <= nCommandType) {\r
857                 ASSERT(0);\r
858                 nCommandType = NONE;\r
859         }\r
860         return CommandTypes[nCommandType].szCommandTypeName;\r
861 }\r
862 \r
863 LPCTSTR CProfile::Key2String(int nKey)\r
864 {\r
865         if (CProfile::Is106Keyboard()) {\r
866                 switch (nKey) {\r
867                 case 0xBA:\r
868                         return _T(":");\r
869                 case 0xBB:\r
870                         return _T(";");\r
871                 case 0xC0:\r
872                         return _T("@");\r
873                 case 0xDE:\r
874                         return _T("^");\r
875                 default:\r
876                         break;\r
877                 }\r
878         }\r
879 \r
880         if (nKey < 0 || sizeof(KeyNames) / sizeof(KeyNames[0]) <= nKey) {\r
881                 ASSERT(0);\r
882                 nKey = 0;\r
883         }\r
884         return KeyNames[nKey].name;\r
885 }\r
886 \r
887 BOOL CProfile::IsCommandType(const int nCommandType, LPCTSTR szKeyBind)\r
888 {\r
889         LPCTSTR szCommandType = CommandType2String(nCommandType);\r
890 \r
891         if (!_tcsnicmp(szKeyBind, szCommandType, _tcslen(szCommandType))) {\r
892                 return TRUE;\r
893         }\r
894 \r
895         return FALSE;\r
896 }\r
897 \r
898 void CProfile::SaveKeyBind(const CString szApplicationName, const int nCommandID, const int nCommandType, const int nKey)\r
899 {\r
900         if (!nCommandID) {\r
901                 return;\r
902         }\r
903 \r
904         CString szCommandName = CCommands::GetCommandName(nCommandID);\r
905         if (szCommandName.IsEmpty()) {\r
906                 return;\r
907         }\r
908 \r
909         SaveKeyBind(szApplicationName, szCommandName, nCommandType, nKey);\r
910 }\r
911 \r
912 void CProfile::SaveKeyBind(const CString szApplicationName, const CString szCommandName, const int nCommandType, const int nKey)\r
913 {\r
914         CString szKeyBind = WriteKeyBind(nCommandType, nKey);\r
915         CString szSubKey(MAKEINTRESOURCE(IDS_REGSUBKEY_DATA));\r
916         szSubKey += _T("\\") + szApplicationName + _T("\\") + szCommandName;\r
917         if (!szKeyBind.IsEmpty()) {\r
918                 szSubKey += _T("\\") + szKeyBind;\r
919         }\r
920 \r
921         HKEY hKey = NULL;\r
922         if (RegCreateKeyEx(HKEY_CURRENT_USER, szSubKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, NULL) == ERROR_SUCCESS) {\r
923                 RegCloseKey(hKey);\r
924         }\r
925 }\r
926 \r
927 void CProfile::SaveCommand(const CString szApplicationName, const int nCommandID)\r
928 {\r
929         SaveKeyBind(szApplicationName, nCommandID, 0, 0);\r
930 }\r
931 \r
932 void CProfile::AddKeyBind2C_(const CString szApplicationName, const BYTE bVk)\r
933 {\r
934         for (int nCommandID = 0; nCommandID < MAX_COMMAND; ++nCommandID) {\r
935                 if (Commands[nCommandID].fCommand == CCommands::C_) {\r
936                         break;\r
937                 }\r
938         }\r
939 \r
940         SaveKeyBind(szApplicationName, nCommandID, NONE, bVk);\r
941 }\r
942 \r
943 void CProfile::LevelUp()\r
944 {\r
945         const int nDefalutLevel = 0;\r
946         const int nLatestLevel = 4;\r
947 \r
948         CString szSection;\r
949         CString szEntry;\r
950         szEntry.Format(_T("Level"));\r
951 \r
952         switch (AfxGetApp()->GetProfileInt(szSection, szEntry, nDefalutLevel)) {\r
953         case nDefalutLevel:\r
954                 {\r
955                         for (int nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
956                                 CString szEntry;\r
957                                 szEntry.Format(IDS_REG_ENTRY_APPLICATION, nApplicationID);\r
958 \r
959                                 CString szApplicationName;\r
960                                 szApplicationName = AfxGetApp()->GetProfileString(CString(MAKEINTRESOURCE(IDS_REG_SECTION_APPLICATION)), szEntry);\r
961                                 if (szApplicationName.IsEmpty()) {\r
962                                         continue;\r
963                                 }\r
964 \r
965                                 AddKeyBind2C_(szApplicationName, VK_LCONTROL);\r
966                                 AddKeyBind2C_(szApplicationName, VK_RCONTROL);\r
967                         }\r
968                 }\r
969                 // Do NOT write break; here.\r
970         case 1:\r
971                 {\r
972                         for (int nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
973                                 CString szEntry;\r
974                                 szEntry.Format(IDS_REG_ENTRY_APPLICATION, nApplicationID);\r
975 \r
976                                 CString szApplicationName;\r
977                                 szApplicationName = AfxGetApp()->GetProfileString(CString(MAKEINTRESOURCE(IDS_REG_SECTION_APPLICATION)), szEntry);\r
978                                 if (szApplicationName.IsEmpty()) {\r
979                                         continue;\r
980                                 }\r
981 \r
982                                 // Set kill-ring-max 1 if it is 0.\r
983                                 if (!AfxGetApp()->GetProfileInt(szApplicationName, CString(MAKEINTRESOURCE(IDS_REG_ENTRY_KILL_RING_MAX)), 0)) {\r
984                                         AfxGetApp()->WriteProfileInt(szApplicationName, CString(MAKEINTRESOURCE(IDS_REG_ENTRY_KILL_RING_MAX)), 1);\r
985                                 }\r
986                         }\r
987                 }\r
988                 // Do NOT write break; here.\r
989         case 2:\r
990                 {\r
991                         for (int nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
992                                 CString szEntry;\r
993                                 szEntry.Format(IDS_REG_ENTRY_APPLICATION, nApplicationID);\r
994 \r
995                                 CString szApplicationName;\r
996                                 szApplicationName = AfxGetApp()->GetProfileString(CString(MAKEINTRESOURCE(IDS_REG_SECTION_APPLICATION)), szEntry);\r
997                                 if (szApplicationName.IsEmpty()) {\r
998                                         continue;\r
999                                 }\r
1000 \r
1001                                 // Chaged a label from Enter to newline.\r
1002                                 CString szSrcSubKey(MAKEINTRESOURCE(IDS_REGSUBKEY_DATA));\r
1003                                 szSrcSubKey += _T("\\") + szApplicationName + _T("\\") + _T("Enter");\r
1004                                 CString szDestSubKey(MAKEINTRESOURCE(IDS_REGSUBKEY_DATA));\r
1005                                 szDestSubKey += _T("\\") + szApplicationName + _T("\\") + _T("newline");\r
1006                                 HKEY hKeyDest = NULL;\r
1007                                 if (RegCreateKeyEx(HKEY_CURRENT_USER, szDestSubKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKeyDest, NULL) == ERROR_SUCCESS) {\r
1008                                         SHCopyKey(HKEY_CURRENT_USER, szSrcSubKey, hKeyDest, NULL);\r
1009                                         SHDeleteKey(HKEY_CURRENT_USER, szSrcSubKey);\r
1010                                         RegCloseKey(hKeyDest);\r
1011                                 }\r
1012                         }\r
1013                 }\r
1014                 // Do NOT write break; here.\r
1015         case 3:\r
1016                 {\r
1017                         for (int nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
1018                                 CString szEntry;\r
1019                                 szEntry.Format(IDS_REG_ENTRY_APPLICATION, nApplicationID);\r
1020 \r
1021                                 CString szApplicationName;\r
1022                                 szApplicationName = AfxGetApp()->GetProfileString(CString(MAKEINTRESOURCE(IDS_REG_SECTION_APPLICATION)), szEntry);\r
1023                                 if (szApplicationName.IsEmpty()) {\r
1024                                         continue;\r
1025                                 }\r
1026 \r
1027                                 // rename original function to remove IDS_REG_ORIGINAL_PREFIX\r
1028                                 for (int nFunctionID = 0; nFunctionID < CDotXkeymacs::GetFunctionNumber(); ++nFunctionID) {\r
1029                                         HKEY hKey = NULL;\r
1030                                         CString szSubKey(MAKEINTRESOURCE(IDS_REGSUBKEY_DATA));\r
1031                                         szSubKey += _T("\\") + szApplicationName + _T("\\") + CString(MAKEINTRESOURCE(IDS_REG_ORIGINAL_PREFIX)) + CDotXkeymacs::GetFunctionSymbol(nFunctionID);\r
1032                                         if (RegOpenKeyEx(HKEY_CURRENT_USER, szSubKey, 0, KEY_READ, &hKey) == ERROR_SUCCESS) {\r
1033                                                 // Use registry data\r
1034                                                 TCHAR szKeyBind[128] = {'\0'};\r
1035                                                 DWORD dwKeyBind = sizeof(szKeyBind);\r
1036                                                 FILETIME ft = {'\0'};   // not use\r
1037                                                 for (DWORD dwIndex = 0; RegEnumKeyEx(hKey, dwIndex, szKeyBind, &dwKeyBind, NULL, NULL, NULL, &ft) == ERROR_SUCCESS; ++dwIndex) {\r
1038                                                         int nCommandType = 0;\r
1039                                                         int nKey = 0;\r
1040                                                         ReadKeyBind(&nCommandType, &nKey, szKeyBind);\r
1041                                                         SaveKeyBind(szApplicationName, CDotXkeymacs::GetFunctionSymbol(nFunctionID), nCommandType, nKey);\r
1042 \r
1043                                                         memset(szKeyBind, 0, sizeof(szKeyBind));\r
1044                                                         dwKeyBind = sizeof(szKeyBind);\r
1045                                                 }\r
1046                                                 RegCloseKey(hKey);\r
1047                                         }\r
1048                                 }\r
1049                         }\r
1050                 }\r
1051 //      case 4:\r
1052 //              foo();\r
1053 //      ...\r
1054 //      case nLatestLevel-1:\r
1055 //              bar();\r
1056                 AfxGetApp()->WriteProfileInt(szSection, szEntry, nLatestLevel);\r
1057                 break;\r
1058         default:\r
1059                 break;\r
1060         }\r
1061 }\r
1062 \r
1063 void CProfile::InitDllData()\r
1064 {\r
1065         LoadData();\r
1066         SetDllData();\r
1067 }\r
1068 \r
1069 void CProfile::ClearData(const CString szCurrentApplication)\r
1070 {\r
1071         for (int nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
1072                 if (m_Data[nApplicationID].GetApplicationName() == szCurrentApplication) {\r
1073                         break;\r
1074                 }\r
1075         }\r
1076         if (nApplicationID < MAX_APP) {\r
1077                 m_Data[nApplicationID].ClearAll();\r
1078         }\r
1079 }\r
1080 \r
1081 // return count of saved settings\r
1082 int CProfile::GetSavedSettingCount()\r
1083 {\r
1084         int nSavedSetting = 0;\r
1085 \r
1086         for (int nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
1087                 CString szApplicationName;\r
1088                 szApplicationName = m_Data[nApplicationID].GetApplicationName();\r
1089                 if (!szApplicationName.IsEmpty()) {\r
1090                         ++nSavedSetting;\r
1091                 }\r
1092         }\r
1093 \r
1094         return nSavedSetting;\r
1095 }\r
1096 \r
1097 void CProfile::InitApplicationList(CComboBox *const cApplicationList)\r
1098 {\r
1099         cApplicationList->ResetContent();\r
1100 \r
1101         GetTaskList();\r
1102 \r
1103         EnumWindows(EnumWindowsProc, (LPARAM)cApplicationList);\r
1104 \r
1105         for (int i = 0; i < MAX_APP; ++i) {\r
1106                 CString szApplicationName       = m_Data[i].GetApplicationName();\r
1107                 CString szApplicationTitle      = m_Data[i].GetApplicationTitle();\r
1108 \r
1109                 CString szListItem;\r
1110                 szListItem.Format(IDS_APPLICATION_LIST_ITEM, szApplicationTitle, szApplicationName);\r
1111                 if (IsNotSameString(cApplicationList, szListItem)\r
1112                  && !IsDefault(szApplicationName)\r
1113                  && !IsDialog(szApplicationName)\r
1114                  && !szApplicationName.IsEmpty()) {\r
1115                         cApplicationList->AddString(szListItem);\r
1116                 }\r
1117         }\r
1118 \r
1119         // Add IME\r
1120         HKL hKL = GetKeyboardLayout(0);\r
1121         if (ImmIsIME(hKL)) {\r
1122                 LPTSTR szIMEDescription = NULL;\r
1123                 UINT nIMEDescription = ImmGetDescription(hKL, NULL, 0);\r
1124                 if (nIMEDescription) {\r
1125                         nIMEDescription += sizeof(TCHAR);       // for NULL\r
1126                         if ((szIMEDescription = new TCHAR[nIMEDescription]) != NULL) {\r
1127                                 ImmGetDescription(hKL, szIMEDescription, nIMEDescription);\r
1128                         }\r
1129 //                      CUtils::Log(_T("nIMEDescription = %d, szIMEDescription = _%s_"), nIMEDescription, szIMEDescription);\r
1130                 }\r
1131 \r
1132                 LPTSTR szIMEFileName = NULL;\r
1133                 UINT nIMEFileName = ImmGetIMEFileName(hKL, NULL, 0);\r
1134                 if (nIMEFileName) {\r
1135                         nIMEFileName += sizeof(TCHAR);\r
1136                         if ((szIMEFileName = new TCHAR[nIMEFileName]) != NULL) {\r
1137                                 ImmGetIMEFileName(hKL, szIMEFileName, nIMEFileName);\r
1138                         }\r
1139 //                      CUtils::Log(_T("nIMEFileName = %d, szIMEFileName = _%s_"), nIMEFileName, szIMEFileName);\r
1140                 }\r
1141 \r
1142                 CString szIMETitle;\r
1143                 CString szIME(MAKEINTRESOURCE(IDS_IME_FILE_NAME));\r
1144                 szIMETitle.Format(IDS_APPLICATION_LIST_ITEM, szIMEDescription ? szIMEDescription : szIME, szIMEFileName ? szIMEFileName : szIME);\r
1145 //              CUtils::Log(_T("szIMETitle = _%s_, szIMEDescription = _%s_, szIMEFileName = _%s_"), szIMETitle, szIMEDescription, szIMEFileName);\r
1146                 if (IsNotSameString(cApplicationList, szIMETitle)) {\r
1147                         cApplicationList->AddString(szIMETitle);\r
1148                 }\r
1149 \r
1150                 delete[] szIMEDescription;\r
1151                 szIMEDescription = NULL;\r
1152                 delete[] szIMEFileName;\r
1153                 szIMEFileName = NULL;\r
1154         }\r
1155 \r
1156         // Add Dialog\r
1157         cApplicationList->InsertString(0, CString(MAKEINTRESOURCE(IDS_DIALOG_TITLE)));\r
1158 \r
1159         // Add Default\r
1160         cApplicationList->InsertString( 0, CString(MAKEINTRESOURCE(IDS_DEFAULT_TITLE)));\r
1161         cApplicationList->SelectString(-1, CString(MAKEINTRESOURCE(IDS_DEFAULT_TITLE)));\r
1162 }\r
1163 \r
1164 void CProfile::GetTaskList()\r
1165 {\r
1166         ZeroMemory(m_TaskList, sizeof(m_TaskList));\r
1167 \r
1168         HANDLE hProcessSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);\r
1169         if (hProcessSnap == (HANDLE)-1)\r
1170                 return;\r
1171 \r
1172         m_dwTasks = 0;\r
1173         PROCESSENTRY32 processEntry32 = {sizeof(PROCESSENTRY32)};\r
1174         if (Process32First(hProcessSnap, &processEntry32)) {\r
1175                 do {\r
1176                         lstrcpy(m_TaskList[m_dwTasks].ProcessName, processEntry32.szExeFile);\r
1177                         m_TaskList[m_dwTasks++].dwProcessId = processEntry32.th32ProcessID;\r
1178                 } while (m_dwTasks < MAX_TASKS && Process32Next(hProcessSnap, &processEntry32));\r
1179         }\r
1180 \r
1181         CloseHandle(hProcessSnap);\r
1182 }\r
1183 \r
1184 // return application index\r
1185 // and update setting style\r
1186 // if there is NOT the application in the data, this function takes care of it.\r
1187 int CProfile::GetApplicationIndex(const CString szApplicationName, const BOOL bSaveAndValidate, int *const nSettingStyle)\r
1188 {\r
1189         if (!bSaveAndValidate) {        // SetDialogData\r
1190                 *nSettingStyle = SETTING_UNDEFINED;\r
1191         }\r
1192 \r
1193         int nApplicationID = GetApplicationIndex(szApplicationName);\r
1194 \r
1195         if (nApplicationID == MAX_APP) {\r
1196                 if (bSaveAndValidate) { // GetDialogData\r
1197                         for (nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
1198                                 CString sz = m_Data[nApplicationID].GetApplicationName();\r
1199                                 if (sz.IsEmpty()) {\r
1200                                         m_Data[nApplicationID].SetApplicationName(szApplicationName);\r
1201                                         break;\r
1202                                 }\r
1203                         }\r
1204                         if (nApplicationID == MAX_APP) {\r
1205                                 return nApplicationID;\r
1206                         }\r
1207                 } else {                                // SetDialogData\r
1208                         for (nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
1209                                 if (IsDefault(m_Data[nApplicationID].GetApplicationName())) {\r
1210                                         *nSettingStyle = SETTING_DEFAULT;\r
1211                                         break;\r
1212                                 }\r
1213                         }\r
1214                         if (nApplicationID == MAX_APP) {\r
1215                                 return nApplicationID;\r
1216                         }\r
1217                 }\r
1218         }\r
1219 \r
1220         if (bSaveAndValidate) { // GetDialogData\r
1221                 m_Data[nApplicationID].SetSettingStyle(*nSettingStyle);\r
1222         } else {                                // SetDialogData\r
1223                 if (*nSettingStyle == SETTING_UNDEFINED) {      // It means that *nSettingStyle != SETTING_DEFAULT.\r
1224                         *nSettingStyle = m_Data[nApplicationID].GetSettingStyle();\r
1225                 }\r
1226         }\r
1227 \r
1228         return nApplicationID;\r
1229 }\r
1230 \r
1231 BOOL CProfile::Is106Keyboard()\r
1232 {\r
1233         static KEYBOARD_TYPE keyboard = UNKNOWN_KEYBOARD;\r
1234 \r
1235         if (keyboard == UNKNOWN_KEYBOARD) {\r
1236                 OSVERSIONINFO verInfo = {0};\r
1237                 verInfo.dwOSVersionInfoSize = sizeof (verInfo);\r
1238                 GetVersionEx(&verInfo);\r
1239 \r
1240                 DWORD subtype = 0;\r
1241                 DWORD cbData = sizeof(subtype);\r
1242 \r
1243                 if (verInfo.dwPlatformId == VER_PLATFORM_WIN32_NT) {\r
1244                         HKEY hKey = NULL;\r
1245                         CString szSubKey(_T("SYSTEM\\CurrentControlSet\\Services\\i8042prt\\Parameters"));\r
1246                         if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, szSubKey, 0, KEY_QUERY_VALUE, &hKey) == ERROR_SUCCESS) {\r
1247                                 static const CString szValueName(_T("OverrideKeyboardSubtype"));\r
1248                                 if (RegQueryValueEx(hKey, szValueName, NULL, NULL, (LPBYTE)&subtype, &cbData) != ERROR_SUCCESS) {\r
1249                                         subtype = 0;\r
1250                                 }\r
1251                                 RegCloseKey(hKey);\r
1252                         }\r
1253                 } else if (verInfo.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS) {\r
1254                         subtype = GetPrivateProfileInt(_T("keyboard"), _T("subtype"), 0, _T("system.ini"));\r
1255                 }\r
1256 \r
1257                 keyboard = (subtype & 0x02) ? JAPANESE_KEYBOARD : ENGLISH_KEYBOARD;\r
1258         }\r
1259 \r
1260         return keyboard == JAPANESE_KEYBOARD;\r
1261 }\r
1262 \r
1263 BOOL CProfile::IsTheString(const CString sz, const UINT nID)\r
1264 {\r
1265         return sz == CString(MAKEINTRESOURCE(nID));\r
1266 }\r
1267 \r
1268 // if sz is "Default", return TRUE\r
1269 BOOL CProfile::IsDefault(const CString sz)\r
1270 {\r
1271         return IsTheString(sz, IDS_DEFAULT);\r
1272 }\r
1273 \r
1274 // if sz is "Dialog", return TRUE\r
1275 BOOL CProfile::IsDialog(const CString sz)\r
1276 {\r
1277         return IsTheString(sz, IDS_DIALOG);\r
1278 }\r
1279 \r
1280 void CProfile::GetApplicationTitle(CComboBox *const cApplicationList, CString &rList, const int nIndex)\r
1281 {\r
1282         if (0 <= nIndex) {\r
1283                 cApplicationList->GetLBText(nIndex, rList);\r
1284         } else {\r
1285                 cApplicationList->GetWindowText(rList);\r
1286         }\r
1287 \r
1288         if (IsTheString(rList, IDS_DEFAULT_TITLE)) {\r
1289                 rList.LoadString(IDS_DEFAULT);\r
1290         }\r
1291 \r
1292         if (IsTheString(rList, IDS_DIALOG_TITLE)) {\r
1293                 rList.LoadString(IDS_DIALOG);\r
1294         }\r
1295 \r
1296         return;\r
1297 }\r
1298 \r
1299 void CProfile::UpdateApplicationTitle(CComboBox *const cApplicationList, const CString szCurrentApplication, const int nApplicationID, const BOOL bSaveAndValidate)\r
1300 {\r
1301         static CString szApplicationTitle;\r
1302         if (bSaveAndValidate) { // GetDialogData\r
1303                 if (!CProfile::IsDefault(szCurrentApplication)) {\r
1304                         m_Data[nApplicationID].SetApplicationTitle(szApplicationTitle);\r
1305                 }\r
1306                 szApplicationTitle.Empty();\r
1307         } else {                                // SetDialogData\r
1308                 CString szListItem;\r
1309                 CProfile::GetApplicationTitle(cApplicationList, szListItem);\r
1310                 int nEndTitle = szListItem.ReverseFind(_T('('));\r
1311                 if (nEndTitle > 0) {\r
1312                         szApplicationTitle = szListItem.Left(nEndTitle);\r
1313                 }\r
1314         }\r
1315 }\r
1316 \r
1317 void CProfile::SetCommandID(const int nApplicationID, const int nCommandType, const int nKey, int nCommandID)\r
1318 {\r
1319         if (nKey == 0xf0 && Commands[nCommandID].fCommand == CCommands::C_) {\r
1320                 // Change CommandID C_Eisu\r
1321                 for (nCommandID = 1; nCommandID < MAX_COMMAND; ++nCommandID) {\r
1322                         if (Commands[nCommandID].fCommand == CCommands::C_Eisu) {\r
1323                                 break;\r
1324                         }\r
1325                 }\r
1326         }\r
1327         m_Data[nApplicationID].SetCommandID(nCommandType, nKey, nCommandID);\r
1328 }\r
1329 \r
1330 int CProfile::GetCommandID(const int nApplicationID, const int nCommandType, const int nKey)\r
1331 {\r
1332         int nCommandID = m_Data[nApplicationID].GetCommandID(nCommandType, nKey);\r
1333         if (nKey == 0xf0 && Commands[nCommandID].fCommand == CCommands::C_Eisu) {\r
1334                 // Change CommandID C_\r
1335                 for (nCommandID = 1; nCommandID < MAX_COMMAND; ++nCommandID) {\r
1336                         if (Commands[nCommandID].fCommand == CCommands::C_) {\r
1337                                 break;\r
1338                         }\r
1339                 }\r
1340         }\r
1341         return nCommandID;\r
1342 }\r
1343 \r
1344 void CProfile::SetKillRingMax(const int nApplicationID, const int nKillRingMax)\r
1345 {\r
1346         m_Data[nApplicationID].SetKillRingMax(nKillRingMax);\r
1347 }\r
1348 \r
1349 int CProfile::GetKillRingMax(const int nApplicationID)\r
1350 {\r
1351         return m_Data[nApplicationID].GetKillRingMax();\r
1352 }\r
1353 \r
1354 void CProfile::SetUseDialogSetting(const int nApplicationID, const BOOL bUseDialogSetting)\r
1355 {\r
1356         m_Data[nApplicationID].SetUseDialogSetting(bUseDialogSetting);\r
1357 }\r
1358 \r
1359 BOOL CProfile::GetUseDialogSetting(const int nApplicationID)\r
1360 {\r
1361         return m_Data[nApplicationID].GetUseDialogSetting();\r
1362 }\r
1363 \r
1364 void CProfile::SetWindowText(const int nApplicationID, const CString szWindowText)\r
1365 {\r
1366         m_Data[nApplicationID].SetWindowText(szWindowText);\r
1367 }\r
1368 \r
1369 CString CProfile::GetWindowText(const int nApplicationID)\r
1370 {\r
1371         return m_Data[nApplicationID].GetWindowText();\r
1372 }\r
1373 \r
1374 void CProfile::DeleteAllRegistryData()\r
1375 {\r
1376         HKEY hkey = NULL;\r
1377         if (RegOpenKeyEx(HKEY_CURRENT_USER, CString(MAKEINTRESOURCE(IDS_REGSUBKEY_DATA)), 0, KEY_ALL_ACCESS, &hkey) == ERROR_SUCCESS) {\r
1378                 // I am sure that I have to do only one time, but...\r
1379                 for (int nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
1380                         DWORD dwIndex = 0;\r
1381                         TCHAR szName[SUB_KEY_NAME_LENGTH] = {'\0'};\r
1382                         DWORD dwName = sizeof(szName);\r
1383                         FILETIME filetime;\r
1384 \r
1385                         while (RegEnumKeyEx(hkey, dwIndex++, szName, &dwName, NULL, NULL, NULL, &filetime) == ERROR_SUCCESS) {\r
1386 //                              RegDeleteKey(hkey, szName);\r
1387                                 SHDeleteKey(hkey, szName);\r
1388                                 ZeroMemory(szName, sizeof(szName));\r
1389                                 dwName = sizeof(szName);\r
1390                         }\r
1391                 }\r
1392                 RegCloseKey(hkey);\r
1393         }\r
1394 }\r
1395 \r
1396 int CProfile::GetCurrentApplicationID(CComboBox *const cApplicationList, const CString szCurrentApplication)\r
1397 {\r
1398         int nCounter = cApplicationList->GetCount();\r
1399         CString szListItem;\r
1400         int nCurSel = cApplicationList->GetCurSel();\r
1401 \r
1402         for (int i = 0; i < nCounter; ++i) {\r
1403                 cApplicationList->SetCurSel(i);\r
1404                 CProfile::GetApplicationTitle(cApplicationList, szListItem);\r
1405                 if (szListItem.Find(szCurrentApplication) != -1) {\r
1406                         cApplicationList->SetCurSel(nCurSel);\r
1407                         return i;\r
1408                 }\r
1409         }\r
1410         return -1;\r
1411 }\r
1412 \r
1413 void CProfile::CopyData(const CString szDestinationApplication, const CString szSourceApplication)\r
1414 {\r
1415         int nSettingStyle = SETTING_SPECIFIC;\r
1416         int nDestinationApplication = GetApplicationIndex(szDestinationApplication, TRUE, &nSettingStyle);\r
1417         int nSourceApplication = GetApplicationIndex(szSourceApplication);\r
1418 \r
1419         CString szApplicationName = m_Data[nDestinationApplication].GetApplicationName();\r
1420         CString szApplicationTitle = m_Data[nDestinationApplication].GetApplicationTitle();\r
1421         CString szWindowText = m_Data[nDestinationApplication].GetWindowText();\r
1422         int nWindowTextType = m_Data[nDestinationApplication].GetWindowTextType();\r
1423 \r
1424         m_Data[nDestinationApplication] = m_Data[nSourceApplication];\r
1425 \r
1426         m_Data[nDestinationApplication].SetApplicationName(szApplicationName);\r
1427         m_Data[nDestinationApplication].SetApplicationTitle(szApplicationTitle);\r
1428         m_Data[nDestinationApplication].SetWindowText(szWindowText);\r
1429         m_Data[nDestinationApplication].SetWindowTextType(nWindowTextType);\r
1430 }\r
1431 \r
1432 // return application index\r
1433 // if there is NOT the application in the data, return MAX_APP\r
1434 int CProfile::GetApplicationIndex(const CString szApplicationName)\r
1435 {\r
1436         int nApplicationID = 0;\r
1437         for (nApplicationID = 0; nApplicationID < MAX_APP; ++nApplicationID) {\r
1438                 if (m_Data[nApplicationID].GetApplicationName() == szApplicationName) {\r
1439                         break;\r
1440                 }\r
1441         }\r
1442         return nApplicationID;\r
1443 }\r
1444 \r
1445 // Return True if Windows Vista or later.\r
1446 BOOL CProfile::IsVistaOrLater()\r
1447 {\r
1448         OSVERSIONINFO info = {sizeof(OSVERSIONINFO)};\r
1449         GetVersionEx(&info);\r
1450 \r
1451         if (6 <= info.dwMajorVersion) {\r
1452                 return TRUE;\r
1453         }\r
1454         return FALSE;\r
1455 }\r
1456 \r
1457 void CProfile::RestartComputer()\r
1458 {\r
1459         if (!AdjustTokenPrivileges(SE_SHUTDOWN_NAME)) {\r
1460                 return;\r
1461         }\r
1462 \r
1463         ExitWindowsEx(EWX_REBOOT, 0);\r
1464 }\r
1465 \r
1466 BOOL CProfile::AdjustTokenPrivileges(LPCTSTR lpName)\r
1467 {\r
1468         BOOL rc = TRUE;\r
1469 \r
1470         HANDLE hToken;\r
1471         if (OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES, &hToken)) {\r
1472                 LUID luid;\r
1473                 if (LookupPrivilegeValue(NULL, lpName, &luid)) {\r
1474                         TOKEN_PRIVILEGES tp;\r
1475                         tp.PrivilegeCount = 1;\r
1476                         tp.Privileges[0].Luid = luid;\r
1477                         tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;\r
1478 \r
1479                         if (!::AdjustTokenPrivileges(hToken, FALSE, &tp, sizeof(TOKEN_PRIVILEGES), NULL, NULL)) {\r
1480                                 rc = FALSE;\r
1481                         }\r
1482                 } else {\r
1483                         rc = FALSE;\r
1484                 }\r
1485                 CloseHandle(hToken);\r
1486         } else {\r
1487                 rc = FALSE;\r
1488         }\r
1489 \r
1490         return rc;\r
1491 }\r
1492 \r
1493 BOOL CProfile::DiableTokenPrivileges()\r
1494 {\r
1495         BOOL rc = TRUE;\r
1496 \r
1497         HANDLE hToken;\r
1498         if (OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES, &hToken)) {\r
1499                 if (!::AdjustTokenPrivileges(hToken, TRUE, NULL, NULL, NULL, NULL)) {\r
1500                         rc = FALSE;\r
1501                 }\r
1502                 CloseHandle(hToken);\r
1503         } else {\r
1504                 rc = FALSE;\r
1505         }\r
1506 \r
1507         return rc;\r
1508 }\r
1509 \r
1510 void CProfile::ExportProperties()\r
1511 {\r
1512         if (!AdjustTokenPrivileges(SE_BACKUP_NAME)) {\r
1513                 return;\r
1514         }\r
1515 \r
1516         CFileDialog oFileOpenDialog(FALSE, _T("reg"), _T("xkeymacs"), OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT, CString(MAKEINTRESOURCE(IDS_REGISTRATION_FILTER)));\r
1517         if (oFileOpenDialog.DoModal() == IDOK) {\r
1518                 CString szCommandLine;\r
1519                 szCommandLine.Format(_T("regedit /e \"%s\" HKEY_CURRENT_USER\\%s"), oFileOpenDialog.GetPathName(), CString(MAKEINTRESOURCE(IDS_REGSUBKEY_DATA)));\r
1520                 CUtils::Run(szCommandLine, TRUE);       // regedit /e "x:\xkeymacs.reg" HKEY_CURRENT_USER\Software\Oishi\XKeymacs2\r
1521         }\r
1522 \r
1523         DiableTokenPrivileges();\r
1524         return;\r
1525 }\r
1526 \r
1527 void CProfile::ImportProperties()\r
1528 {\r
1529         if (!AdjustTokenPrivileges(SE_RESTORE_NAME)) {\r
1530                 return;\r
1531         }\r
1532 \r
1533         CFileDialog oFileOpenDialog(TRUE, _T("reg"), _T("xkeymacs"), OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT, CString(MAKEINTRESOURCE(IDS_REGISTRATION_FILTER)));\r
1534         if (oFileOpenDialog.DoModal() == IDOK) {\r
1535                 CString szCommandLine;\r
1536                 szCommandLine.Format(_T("regedit \"%s\""), oFileOpenDialog.GetPathName());\r
1537                 CUtils::Run(szCommandLine, TRUE);       // regedit "x:\xkeymacs.reg"\r
1538         }\r
1539 \r
1540         DiableTokenPrivileges();\r
1541         return;\r
1542 }\r
1543 \r
1544 BOOL CProfile::GetEnableCUA(const int nApplicationID)\r
1545 {\r
1546         return m_Data[nApplicationID].GetEnableCUA();\r
1547 }\r
1548 \r
1549 void CProfile::SetEnableCUA(const int nApplicationID, const BOOL bEnableCUA)\r
1550 {\r
1551         m_Data[nApplicationID].SetEnableCUA(bEnableCUA);\r
1552 }\r
1553 \r
1554 int CProfile::GetKeyboardSpeed()\r
1555 {\r
1556         int nKeyboardSpeed = 31; // default value of Windows\r
1557         CString szSubKey(MAKEINTRESOURCE(IDS_REGSUBKEY_KEYBOARD));\r
1558         CString szValueName(MAKEINTRESOURCE(IDS_KEYBOARD_SPEED));\r
1559 \r
1560         HKEY hKey = NULL;\r
1561         if (RegOpenKeyEx(HKEY_CURRENT_USER, szSubKey, 0, KEY_QUERY_VALUE, &hKey) == ERROR_SUCCESS) {\r
1562                 // get data size\r
1563                 DWORD dwType = REG_SZ;\r
1564                 BYTE data[4] = {0};\r
1565                 DWORD dwcbData = sizeof(data)/sizeof(data[0]);\r
1566                 RegQueryValueEx(hKey, szValueName, NULL, &dwType, (LPBYTE)&data, &dwcbData);\r
1567                 RegCloseKey(hKey);\r
1568 \r
1569                 for (DWORD i = 0; i < dwcbData; ++i) {\r
1570                         if (data[i]) {\r
1571                                 if (i) {\r
1572                                         nKeyboardSpeed = nKeyboardSpeed * 10 + data[i] - _T('0');\r
1573                                 } else {\r
1574                                         nKeyboardSpeed = data[i] - _T('0');\r
1575                                 }\r
1576                         } else {\r
1577                                 break;\r
1578                         }\r
1579                 }\r
1580         }\r
1581 \r
1582         return nKeyboardSpeed;\r
1583 }\r