3 * (HKEY_CLASSES_ROOT - Stuff)
5 * Copyright 1998, 1999, 2000 Juergen Schmied
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #include <wine/config.h>
26 #define WIN32_NO_STATUS
33 #include <shlguid_undoc.h>
35 #include <wine/debug.h>
36 #include <wine/unicode.h>
39 #include "shell32_main.h"
42 WINE_DEFAULT_DEBUG_CHANNEL(shell
);
44 #define MAX_EXTENSION_LENGTH 20
46 BOOL
HCR_MapTypeToValueW(LPCWSTR szExtension
, LPWSTR szFileType
, LONG len
, BOOL bPrependDot
)
49 WCHAR szTemp
[MAX_EXTENSION_LENGTH
+ 2];
51 TRACE("%s %p\n", debugstr_w(szExtension
), szFileType
);
53 /* added because we do not want to have double dots */
54 if (szExtension
[0] == '.')
60 lstrcpynW(szTemp
+ (bPrependDot
?1:0), szExtension
, MAX_EXTENSION_LENGTH
);
62 if (RegOpenKeyExW(HKEY_CLASSES_ROOT
, szTemp
, 0, KEY_READ
, &hkey
))
67 if (RegQueryValueW(hkey
, NULL
, szFileType
, &len
))
75 TRACE("--UE;\n} %s\n", debugstr_w(szFileType
));
80 BOOL
HCR_MapTypeToValueA(LPCSTR szExtension
, LPSTR szFileType
, LONG len
, BOOL bPrependDot
)
83 char szTemp
[MAX_EXTENSION_LENGTH
+ 2];
85 TRACE("%s %p\n", szExtension
, szFileType
);
87 /* added because we do not want to have double dots */
88 if (szExtension
[0] == '.')
94 lstrcpynA(szTemp
+ (bPrependDot
?1:0), szExtension
, MAX_EXTENSION_LENGTH
);
96 if (RegOpenKeyExA(HKEY_CLASSES_ROOT
, szTemp
, 0, KEY_READ
, &hkey
))
102 if (!RegLoadMUIStringA(hkey
, "FriendlyTypeName", szFileType
, len
, NULL
, 0, NULL
))
109 if (RegQueryValueA(hkey
, NULL
, szFileType
, &len
))
117 TRACE("--UE;\n} %s\n", szFileType
);
122 static const WCHAR swShell
[] = {'s','h','e','l','l','\\',0};
123 static const WCHAR swOpen
[] = {'o','p','e','n',0};
124 static const WCHAR swCommand
[] = {'\\','c','o','m','m','a','n','d',0};
126 BOOL
HCR_GetDefaultVerbW( HKEY hkeyClass
, LPCWSTR szVerb
, LPWSTR szDest
, DWORD len
)
128 WCHAR sTemp
[MAX_PATH
];
132 TRACE("%p %s %p\n", hkeyClass
, debugstr_w(szVerb
), szDest
);
134 if (szVerb
&& *szVerb
)
136 lstrcpynW(szDest
, szVerb
, len
);
142 if (!RegQueryValueW(hkeyClass
, swShell
, szDest
, &size
) && *szDest
)
144 /* The MSDN says to first try the default verb */
145 lstrcpyW(sTemp
, swShell
);
146 lstrcatW(sTemp
, szDest
);
147 lstrcatW(sTemp
, swCommand
);
148 if (!RegOpenKeyExW(hkeyClass
, sTemp
, 0, KEY_READ
, &hkey
))
151 TRACE("default verb=%s\n", debugstr_w(szDest
));
156 /* then fallback to 'open' */
157 lstrcpyW(sTemp
, swShell
);
158 lstrcatW(sTemp
, swOpen
);
159 lstrcatW(sTemp
, swCommand
);
160 if (!RegOpenKeyExW(hkeyClass
, sTemp
, 0, KEY_READ
, &hkey
))
163 lstrcpynW(szDest
, swOpen
, len
);
164 TRACE("default verb=open\n");
168 /* and then just use the first verb on Windows >= 2000 */
170 if (!RegOpenKeyExW(hkeyClass
, L
"shell", 0, KEY_READ
, &hkey
))
172 if (!RegEnumKeyW(hkey
, 0, szDest
, len
) && *szDest
)
174 TRACE("default verb=first verb=%s\n", debugstr_w(szDest
));
181 if (!RegEnumKeyW(hkeyClass
, 0, szDest
, len
) && *szDest
)
183 TRACE("default verb=first verb=%s\n", debugstr_w(szDest
));
188 TRACE("no default verb!\n");
192 BOOL
HCR_GetExecuteCommandW( HKEY hkeyClass
, LPCWSTR szClass
, LPCWSTR szVerb
, LPWSTR szDest
, DWORD len
)
194 WCHAR sTempVerb
[MAX_PATH
];
197 TRACE("%p %s %s %p\n", hkeyClass
, debugstr_w(szClass
), debugstr_w(szVerb
), szDest
);
200 RegOpenKeyExW(HKEY_CLASSES_ROOT
, szClass
, 0, KEY_READ
, &hkeyClass
);
205 if (HCR_GetDefaultVerbW(hkeyClass
, szVerb
, sTempVerb
, sizeof(sTempVerb
)/sizeof(sTempVerb
[0])))
207 WCHAR sTemp
[MAX_PATH
];
208 lstrcpyW(sTemp
, swShell
);
209 lstrcatW(sTemp
, sTempVerb
);
210 lstrcatW(sTemp
, swCommand
);
211 ret
= (ERROR_SUCCESS
== SHGetValueW(hkeyClass
, sTemp
, NULL
, NULL
, szDest
, &len
));
214 RegCloseKey(hkeyClass
);
216 TRACE("-- %s\n", debugstr_w(szDest
) );
220 /***************************************************************************************
221 * HCR_GetDefaultIcon [internal]
223 * Gets the icon for a filetype
225 static BOOL
HCR_RegOpenClassIDKey(REFIID riid
, HKEY
*hkey
)
228 sprintf( xriid
, "CLSID\\{%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x}",
229 riid
->Data1
, riid
->Data2
, riid
->Data3
,
230 riid
->Data4
[0], riid
->Data4
[1], riid
->Data4
[2], riid
->Data4
[3],
231 riid
->Data4
[4], riid
->Data4
[5], riid
->Data4
[6], riid
->Data4
[7] );
233 TRACE("%s\n",xriid
);
235 return !RegOpenKeyExA(HKEY_CLASSES_ROOT
, xriid
, 0, KEY_READ
, hkey
);
238 static BOOL
HCR_RegGetIconW(HKEY hkey
, LPWSTR szDest
, LPCWSTR szName
, DWORD len
, int* picon_idx
)
241 WCHAR sTemp
[MAX_PATH
];
244 if (!RegQueryValueExW(hkey
, szName
, 0, &dwType
, (LPBYTE
)szDest
, &len
))
246 if (dwType
== REG_EXPAND_SZ
)
248 ExpandEnvironmentStringsW(szDest
, sTemp
, MAX_PATH
);
249 lstrcpynW(szDest
, sTemp
, len
);
251 if (ParseFieldW (szDest
, 2, sNum
, _countof(sNum
)))
252 *picon_idx
= atoiW(sNum
);
254 *picon_idx
=0; /* sometimes the icon number is missing */
255 ParseFieldW (szDest
, 1, szDest
, len
);
256 PathUnquoteSpacesW(szDest
);
262 static BOOL
HCR_RegGetIconA(HKEY hkey
, LPSTR szDest
, LPCSTR szName
, DWORD len
, int* picon_idx
)
265 char sTemp
[MAX_PATH
];
268 if (!RegQueryValueExA(hkey
, szName
, 0, &dwType
, (LPBYTE
)szDest
, &len
))
270 if (dwType
== REG_EXPAND_SZ
)
272 ExpandEnvironmentStringsA(szDest
, sTemp
, MAX_PATH
);
273 lstrcpynA(szDest
, sTemp
, len
);
275 if (ParseFieldA (szDest
, 2, sNum
, 5))
276 *picon_idx
=atoi(sNum
);
278 *picon_idx
=0; /* sometimes the icon number is missing */
279 ParseFieldA (szDest
, 1, szDest
, len
);
280 PathUnquoteSpacesA(szDest
);
286 BOOL
HCR_GetIconW(LPCWSTR szClass
, LPWSTR szDest
, LPCWSTR szName
, DWORD len
, int* picon_idx
)
288 static const WCHAR swDefaultIcon
[] = {'\\','D','e','f','a','u','l','t','I','c','o','n',0};
290 WCHAR sTemp
[MAX_PATH
];
293 TRACE("%s\n",debugstr_w(szClass
) );
295 lstrcpynW(sTemp
, szClass
, MAX_PATH
);
296 lstrcatW(sTemp
, swDefaultIcon
);
298 if (!RegOpenKeyExW(HKEY_CLASSES_ROOT
, sTemp
, 0, KEY_READ
, &hkey
))
300 ret
= HCR_RegGetIconW(hkey
, szDest
, szName
, len
, picon_idx
);
305 TRACE("-- %s %i\n", debugstr_w(szDest
), *picon_idx
);
307 TRACE("-- not found\n");
312 BOOL
HCR_GetIconA(LPCSTR szClass
, LPSTR szDest
, LPCSTR szName
, DWORD len
, int* picon_idx
)
315 char sTemp
[MAX_PATH
];
318 TRACE("%s\n",szClass
);
320 sprintf(sTemp
, "%s\\DefaultIcon",szClass
);
322 if (!RegOpenKeyExA(HKEY_CLASSES_ROOT
, sTemp
, 0, KEY_READ
, &hkey
))
324 ret
= HCR_RegGetIconA(hkey
, szDest
, szName
, len
, picon_idx
);
327 TRACE("-- %s %i\n", szDest
, *picon_idx
);
331 /***************************************************************************************
332 * HCR_GetClassName [internal]
334 * Gets the name of a registered class
336 static const WCHAR swEmpty
[] = {0};
338 BOOL
HCR_GetClassNameW(REFIID riid
, LPWSTR szDest
, DWORD len
)
351 if (StringFromCLSID(riid
, &pStr
) == S_OK
)
353 DWORD dwLen
= buflen
* sizeof(WCHAR
);
354 swprintf(szName
, L
"Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\CLSID\\%s", pStr
);
355 if (!RegGetValueW(HKEY_CURRENT_USER
, szName
, NULL
, RRF_RT_REG_SZ
, NULL
, (PVOID
)szDest
, &dwLen
))
361 if (!ret
&& HCR_RegOpenClassIDKey(riid
, &hkey
))
363 if (HCR_RegOpenClassIDKey(riid
, &hkey
))
366 static const WCHAR wszLocalizedString
[] =
367 { 'L','o','c','a','l','i','z','e','d','S','t','r','i','n','g', 0 };
368 if (!RegLoadMUIStringW(hkey
, wszLocalizedString
, szDest
, len
, NULL
, 0, NULL
) ||
369 !RegQueryValueExW(hkey
, swEmpty
, 0, NULL
, (LPBYTE
)szDest
, &len
))
376 if (!ret
|| !szDest
[0])
378 if(IsEqualIID(riid
, &CLSID_ShellDesktop
))
380 if (LoadStringW(shell32_hInstance
, IDS_DESKTOP
, szDest
, buflen
))
383 else if (IsEqualIID(riid
, &CLSID_MyComputer
))
385 if(LoadStringW(shell32_hInstance
, IDS_MYCOMPUTER
, szDest
, buflen
))
389 else if (IsEqualIID(riid
, &CLSID_MyDocuments
))
391 if(LoadStringW(shell32_hInstance
, IDS_PERSONAL
, szDest
, buflen
))
394 else if (IsEqualIID(riid
, &CLSID_RecycleBin
))
396 if(LoadStringW(shell32_hInstance
, IDS_RECYCLEBIN_FOLDER_NAME
, szDest
, buflen
))
399 else if (IsEqualIID(riid
, &CLSID_ControlPanel
))
401 if(LoadStringW(shell32_hInstance
, IDS_CONTROLPANEL
, szDest
, buflen
))
404 else if (IsEqualIID(riid
, &CLSID_AdminFolderShortcut
))
406 if(LoadStringW(shell32_hInstance
, IDS_ADMINISTRATIVETOOLS
, szDest
, buflen
))
411 TRACE("-- %s\n", debugstr_w(szDest
));
415 BOOL
HCR_GetClassNameA(REFIID riid
, LPSTR szDest
, DWORD len
)
421 if (HCR_RegOpenClassIDKey(riid
, &hkey
))
423 if (!RegLoadMUIStringA(hkey
,"LocalizedString",szDest
,len
,NULL
,0,NULL
) ||
424 !RegQueryValueExA(hkey
,"",0,NULL
,(LPBYTE
)szDest
,&len
))
431 if (!ret
|| !szDest
[0])
433 if(IsEqualIID(riid
, &CLSID_ShellDesktop
))
435 if (LoadStringA(shell32_hInstance
, IDS_DESKTOP
, szDest
, buflen
))
438 else if (IsEqualIID(riid
, &CLSID_MyComputer
))
440 if(LoadStringA(shell32_hInstance
, IDS_MYCOMPUTER
, szDest
, buflen
))
445 TRACE("-- (%s)\n", szDest
);
450 /******************************************************************************
451 * HCR_GetFolderAttributes [Internal]
453 * Query the registry for a shell folders' attributes
456 * pidlFolder [I] A simple pidl of type PT_GUID.
457 * pdwAttributes [IO] In: Attributes to be queried, OUT: Resulting attributes.
460 * TRUE: Found information for the attributes in the registry
461 * FALSE: No attribute information found
464 * If queried for an attribute, which is set in the CallForAttributes registry
465 * value, the function binds to the shellfolder objects and queries it.
467 BOOL
HCR_GetFolderAttributes(LPCITEMIDLIST pidlFolder
, LPDWORD pdwAttributes
)
473 static const WCHAR wszAttributes
[] = { 'A','t','t','r','i','b','u','t','e','s',0 };
474 static const WCHAR wszCallForAttributes
[] = {
475 'C','a','l','l','F','o','r','A','t','t','r','i','b','u','t','e','s',0 };
476 WCHAR wszShellFolderKey
[] = { 'C','L','S','I','D','\\','{','0','0','0','2','1','4','0','0','-',
477 '0','0','0','0','-','0','0','0','0','-','C','0','0','0','-','0','0','0','0','0','0','0',
478 '0','0','0','4','6','}','\\','S','h','e','l','l','F','o','l','d','e','r',0 };
480 TRACE("(pidlFolder=%p, pdwAttributes=%p)\n", pidlFolder
, pdwAttributes
);
482 if (!_ILIsPidlSimple(pidlFolder
)) {
483 static BOOL firstHit
= TRUE
;
485 ERR("should be called for simple PIDL's only!\n");
491 if (!_ILIsDesktop(pidlFolder
)) {
492 if (FAILED(StringFromCLSID(_ILGetGUIDPointer(pidlFolder
), &pwszCLSID
))) return FALSE
;
493 memcpy(&wszShellFolderKey
[6], pwszCLSID
, 38 * sizeof(WCHAR
));
494 CoTaskMemFree(pwszCLSID
);
497 lResult
= RegOpenKeyExW(HKEY_CLASSES_ROOT
, wszShellFolderKey
, 0, KEY_READ
, &hSFKey
);
499 if (lResult
!= ERROR_SUCCESS
)
501 ERR("Cannot open key: %ls\n", wszShellFolderKey
);
505 if (lResult
!= ERROR_SUCCESS
) return FALSE
;
508 dwLen
= sizeof(DWORD
);
509 lResult
= RegQueryValueExW(hSFKey
, wszCallForAttributes
, 0, NULL
, (LPBYTE
)&dwTemp
, &dwLen
);
510 if ((lResult
== ERROR_SUCCESS
) && (dwTemp
& *pdwAttributes
)) {
511 LPSHELLFOLDER psfDesktop
, psfFolder
;
515 hr
= SHGetDesktopFolder(&psfDesktop
);
517 hr
= IShellFolder_BindToObject(psfDesktop
, pidlFolder
, NULL
, &IID_IShellFolder
,
518 (LPVOID
*)&psfFolder
);
520 hr
= IShellFolder_GetAttributesOf(psfFolder
, 0, NULL
, pdwAttributes
);
521 IShellFolder_Release(psfFolder
);
523 IShellFolder_Release(psfDesktop
);
525 if (FAILED(hr
)) return FALSE
;
527 lResult
= RegQueryValueExW(hSFKey
, wszAttributes
, 0, NULL
, (LPBYTE
)&dwTemp
, &dwLen
);
529 if (lResult
== ERROR_SUCCESS
) {
530 *pdwAttributes
&= dwTemp
;
536 TRACE("-- *pdwAttributes == 0x%08x\n", *pdwAttributes
);