cmd: DIR command outputs free space for the path.
[wine.git] / dlls / shell32 / classes.c
bloba8e70c021c20969219c9de888a2c73b60eac3bd0
1 /*
2 * file type mapping
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 <stdlib.h>
23 #include <string.h>
24 #include <stdarg.h>
25 #include <stdio.h>
27 #define COBJMACROS
29 #include "wine/debug.h"
30 #include "winerror.h"
31 #include "windef.h"
32 #include "winbase.h"
33 #include "winreg.h"
34 #include "wingdi.h"
35 #include "winuser.h"
37 #include "shlobj.h"
38 #include "shell32_main.h"
39 #include "shlguid.h"
40 #include "shresdef.h"
41 #include "shlwapi.h"
42 #include "pidl.h"
44 WINE_DEFAULT_DEBUG_CHANNEL(shell);
46 #define MAX_EXTENSION_LENGTH 20
48 BOOL HCR_MapTypeToValueW(LPCWSTR szExtension, LPWSTR szFileType, LONG len, BOOL bPrependDot)
50 HKEY hkey;
51 WCHAR szTemp[MAX_EXTENSION_LENGTH + 2];
53 TRACE("%s %p\n", debugstr_w(szExtension), szFileType);
55 /* added because we do not want to have double dots */
56 if (szExtension[0] == '.')
57 bPrependDot = FALSE;
59 if (bPrependDot)
60 szTemp[0] = '.';
62 lstrcpynW(szTemp + (bPrependDot?1:0), szExtension, MAX_EXTENSION_LENGTH);
64 if (RegOpenKeyExW(HKEY_CLASSES_ROOT, szTemp, 0, KEY_READ, &hkey))
66 return FALSE;
69 if (RegQueryValueW(hkey, NULL, szFileType, &len))
71 RegCloseKey(hkey);
72 return FALSE;
75 RegCloseKey(hkey);
77 TRACE("--UE;\n} %s\n", debugstr_w(szFileType));
79 return TRUE;
82 BOOL HCR_MapTypeToValueA(LPCSTR szExtension, LPSTR szFileType, LONG len, BOOL bPrependDot)
84 HKEY hkey;
85 char szTemp[MAX_EXTENSION_LENGTH + 2];
87 TRACE("%s %p\n", szExtension, szFileType);
89 /* added because we do not want to have double dots */
90 if (szExtension[0] == '.')
91 bPrependDot = FALSE;
93 if (bPrependDot)
94 szTemp[0] = '.';
96 lstrcpynA(szTemp + (bPrependDot?1:0), szExtension, MAX_EXTENSION_LENGTH);
98 if (RegOpenKeyExA(HKEY_CLASSES_ROOT, szTemp, 0, KEY_READ, &hkey))
100 return FALSE;
103 if (RegQueryValueA(hkey, NULL, szFileType, &len))
105 RegCloseKey(hkey);
106 return FALSE;
109 RegCloseKey(hkey);
111 TRACE("--UE;\n} %s\n", szFileType);
113 return TRUE;
116 BOOL HCR_GetDefaultVerbW( HKEY hkeyClass, LPCWSTR szVerb, LPWSTR szDest, DWORD len )
118 WCHAR sTemp[MAX_PATH];
119 LONG size;
120 HKEY hkey;
122 TRACE("%p %s %p\n", hkeyClass, debugstr_w(szVerb), szDest);
124 if (szVerb && *szVerb)
126 lstrcpynW(szDest, szVerb, len);
127 return TRUE;
130 size=len;
131 *szDest='\0';
132 if (!RegQueryValueW(hkeyClass, L"shell\\", szDest, &size) && *szDest)
134 /* The MSDN says to first try the default verb */
135 lstrcpyW(sTemp, L"shell\\");
136 lstrcatW(sTemp, szDest);
137 lstrcatW(sTemp, L"\\command");
138 if (!RegOpenKeyExW(hkeyClass, sTemp, 0, 0, &hkey))
140 RegCloseKey(hkey);
141 TRACE("default verb=%s\n", debugstr_w(szDest));
142 return TRUE;
146 /* then fallback to 'open' */
147 lstrcpyW(sTemp, L"shell\\open\\command");
148 if (!RegOpenKeyExW(hkeyClass, sTemp, 0, 0, &hkey))
150 RegCloseKey(hkey);
151 lstrcpynW(szDest, L"open", len);
152 TRACE("default verb=open\n");
153 return TRUE;
156 /* and then just use the first verb on Windows >= 2000 */
157 if (!RegEnumKeyW(hkeyClass, 0, szDest, len) && *szDest)
159 TRACE("default verb=first verb=%s\n", debugstr_w(szDest));
160 return TRUE;
163 TRACE("no default verb!\n");
164 return FALSE;
167 BOOL HCR_GetExecuteCommandW( HKEY hkeyClass, LPCWSTR szClass, LPCWSTR szVerb, LPWSTR szDest, DWORD len )
169 WCHAR sTempVerb[MAX_PATH];
170 BOOL ret;
172 TRACE("%p %s %s %p\n", hkeyClass, debugstr_w(szClass), debugstr_w(szVerb), szDest);
174 if (szClass)
175 RegOpenKeyExW(HKEY_CLASSES_ROOT, szClass, 0, KEY_READ, &hkeyClass);
176 if (!hkeyClass)
177 return FALSE;
178 ret = FALSE;
180 if (HCR_GetDefaultVerbW(hkeyClass, szVerb, sTempVerb, ARRAY_SIZE(sTempVerb)))
182 WCHAR sTemp[MAX_PATH];
183 lstrcpyW(sTemp, L"shell\\");
184 lstrcatW(sTemp, sTempVerb);
185 lstrcatW(sTemp, L"\\command");
186 ret = (ERROR_SUCCESS == SHGetValueW(hkeyClass, sTemp, NULL, NULL, szDest, &len));
188 if (szClass)
189 RegCloseKey(hkeyClass);
191 TRACE("-- %s\n", debugstr_w(szDest) );
192 return ret;
195 /***************************************************************************************
196 * HCR_GetDefaultIcon [internal]
198 * Gets the icon for a filetype
200 static BOOL HCR_RegOpenClassIDKey(REFIID riid, HKEY *hkey)
202 char xriid[50];
203 sprintf( xriid, "CLSID\\{%08lx-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x}",
204 riid->Data1, riid->Data2, riid->Data3,
205 riid->Data4[0], riid->Data4[1], riid->Data4[2], riid->Data4[3],
206 riid->Data4[4], riid->Data4[5], riid->Data4[6], riid->Data4[7] );
208 TRACE("%s\n",xriid );
210 return !RegOpenKeyExA(HKEY_CLASSES_ROOT, xriid, 0, KEY_READ, hkey);
213 static BOOL HCR_RegGetDefaultIconW(HKEY hkey, LPWSTR szDest, DWORD len, int* picon_idx)
215 DWORD dwType, size = len * sizeof(WCHAR);
216 WCHAR sTemp[MAX_PATH];
217 WCHAR sNum[5];
219 if (!RegQueryValueExW(hkey, NULL, 0, &dwType, (LPBYTE)szDest, &size))
221 if (dwType == REG_EXPAND_SZ)
223 ExpandEnvironmentStringsW(szDest, sTemp, MAX_PATH);
224 lstrcpynW(szDest, sTemp, len);
226 if (ParseFieldW (szDest, 2, sNum, 5))
227 *picon_idx = wcstol(sNum, NULL, 10);
228 else
229 *picon_idx=0; /* sometimes the icon number is missing */
230 ParseFieldW (szDest, 1, szDest, len);
231 PathUnquoteSpacesW(szDest);
232 return TRUE;
234 return FALSE;
237 static BOOL HCR_RegGetDefaultIconA(HKEY hkey, LPSTR szDest, DWORD len, int* picon_idx)
239 DWORD dwType;
240 char sTemp[MAX_PATH];
241 char sNum[5];
243 if (!RegQueryValueExA(hkey, NULL, 0, &dwType, (LPBYTE)szDest, &len))
245 if (dwType == REG_EXPAND_SZ)
247 ExpandEnvironmentStringsA(szDest, sTemp, MAX_PATH);
248 lstrcpynA(szDest, sTemp, len);
250 if (ParseFieldA (szDest, 2, sNum, 5))
251 *picon_idx=atoi(sNum);
252 else
253 *picon_idx=0; /* sometimes the icon number is missing */
254 ParseFieldA (szDest, 1, szDest, len);
255 PathUnquoteSpacesA(szDest);
256 return TRUE;
258 return FALSE;
261 BOOL HCR_GetDefaultIconW(LPCWSTR szClass, LPWSTR szDest, DWORD len, int* picon_idx)
263 HKEY hkey;
264 WCHAR sTemp[MAX_PATH];
265 BOOL ret = FALSE;
267 TRACE("%s\n",debugstr_w(szClass) );
269 lstrcpynW(sTemp, szClass, MAX_PATH);
270 lstrcatW(sTemp, L"\\DefaultIcon");
272 if (!RegOpenKeyExW(HKEY_CLASSES_ROOT, sTemp, 0, KEY_READ, &hkey))
274 ret = HCR_RegGetDefaultIconW(hkey, szDest, len, picon_idx);
275 RegCloseKey(hkey);
278 if(ret)
279 TRACE("-- %s %i\n", debugstr_w(szDest), *picon_idx);
280 else
281 TRACE("-- not found\n");
283 return ret;
286 BOOL HCR_GetDefaultIconA(LPCSTR szClass, LPSTR szDest, DWORD len, int* picon_idx)
288 HKEY hkey;
289 char sTemp[MAX_PATH];
290 BOOL ret = FALSE;
292 TRACE("%s\n",szClass );
294 sprintf(sTemp, "%s\\DefaultIcon",szClass);
296 if (!RegOpenKeyExA(HKEY_CLASSES_ROOT, sTemp, 0, KEY_READ, &hkey))
298 ret = HCR_RegGetDefaultIconA(hkey, szDest, len, picon_idx);
299 RegCloseKey(hkey);
302 if (ret)
303 TRACE("-- %s %i\n", szDest, *picon_idx);
304 else
305 TRACE("-- not found\n");
307 return ret;
310 /***************************************************************************************
311 * HCR_GetClassName [internal]
313 * Gets the name of a registered class
315 BOOL HCR_GetClassNameW(REFIID riid, LPWSTR szDest, DWORD len)
317 HKEY hkey;
318 BOOL ret = FALSE;
319 DWORD buflen = len * sizeof(WCHAR);
321 szDest[0] = 0;
322 if (HCR_RegOpenClassIDKey(riid, &hkey))
324 if (!RegLoadMUIStringW(hkey, L"LocalizedString", szDest, buflen, NULL, 0, NULL) ||
325 !RegQueryValueExW(hkey, L"", 0, NULL, (LPBYTE)szDest, &buflen))
327 ret = TRUE;
329 RegCloseKey(hkey);
332 if (!ret || !szDest[0])
334 if(IsEqualIID(riid, &CLSID_ShellDesktop))
336 if (LoadStringW(shell32_hInstance, IDS_DESKTOP, szDest, len))
337 ret = TRUE;
339 else if (IsEqualIID(riid, &CLSID_MyComputer))
341 if(LoadStringW(shell32_hInstance, IDS_MYCOMPUTER, szDest, len))
342 ret = TRUE;
345 TRACE("-- %s\n", debugstr_w(szDest));
346 return ret;
349 BOOL HCR_GetClassNameA(REFIID riid, LPSTR szDest, DWORD len)
350 { HKEY hkey;
351 BOOL ret = FALSE;
352 DWORD buflen = len;
354 szDest[0] = 0;
355 if (HCR_RegOpenClassIDKey(riid, &hkey))
357 if (!RegLoadMUIStringA(hkey,"LocalizedString",szDest,len,NULL,0,NULL) ||
358 !RegQueryValueExA(hkey,"",0,NULL,(LPBYTE)szDest,&len))
360 ret = TRUE;
362 RegCloseKey(hkey);
365 if (!ret || !szDest[0])
367 if(IsEqualIID(riid, &CLSID_ShellDesktop))
369 if (LoadStringA(shell32_hInstance, IDS_DESKTOP, szDest, buflen))
370 ret = TRUE;
372 else if (IsEqualIID(riid, &CLSID_MyComputer))
374 if(LoadStringA(shell32_hInstance, IDS_MYCOMPUTER, szDest, buflen))
375 ret = TRUE;
379 TRACE("-- (%s)\n", szDest);
381 return ret;
384 /******************************************************************************
385 * HCR_GetFolderAttributes [Internal]
387 * Query the registry for a shell folders' attributes
389 * PARAMS
390 * pidlFolder [I] A simple pidl of type PT_GUID.
391 * pdwAttributes [IO] In: Attributes to be queried, OUT: Resulting attributes.
393 * RETURNS
394 * TRUE: Found information for the attributes in the registry
395 * FALSE: No attribute information found
397 * NOTES
398 * If queried for an attribute, which is set in the CallForAttributes registry
399 * value, the function binds to the shellfolder objects and queries it.
401 BOOL HCR_GetFolderAttributes(LPCITEMIDLIST pidlFolder, LPDWORD pdwAttributes)
403 HKEY hSFKey;
404 LPOLESTR pwszCLSID;
405 LONG lResult;
406 DWORD dwTemp, dwLen;
407 WCHAR wszShellFolderKey[] = L"CLSID\\{00021400-0000-0000-C000-000000000046}\\ShellFolder";
409 TRACE("(pidlFolder=%p, pdwAttributes=%p)\n", pidlFolder, pdwAttributes);
411 if (!_ILIsPidlSimple(pidlFolder)) {
412 static BOOL firstHit = TRUE;
413 if (firstHit) {
414 ERR("should be called for simple PIDL's only!\n");
415 firstHit = FALSE;
417 return FALSE;
420 if (!_ILIsDesktop(pidlFolder)) {
421 if (FAILED(StringFromCLSID(_ILGetGUIDPointer(pidlFolder), &pwszCLSID))) return FALSE;
422 memcpy(&wszShellFolderKey[6], pwszCLSID, 38 * sizeof(WCHAR));
423 CoTaskMemFree(pwszCLSID);
426 lResult = RegOpenKeyExW(HKEY_CLASSES_ROOT, wszShellFolderKey, 0, KEY_READ, &hSFKey);
427 if (lResult != ERROR_SUCCESS) return FALSE;
429 dwLen = sizeof(DWORD);
430 lResult = RegQueryValueExW(hSFKey, L"CallForAttributes", 0, NULL, (LPBYTE)&dwTemp, &dwLen);
431 if ((lResult == ERROR_SUCCESS) && (dwTemp & *pdwAttributes)) {
432 LPSHELLFOLDER psfDesktop, psfFolder;
433 HRESULT hr;
435 RegCloseKey(hSFKey);
436 hr = SHGetDesktopFolder(&psfDesktop);
437 if (SUCCEEDED(hr)) {
438 hr = IShellFolder_BindToObject(psfDesktop, pidlFolder, NULL, &IID_IShellFolder,
439 (LPVOID*)&psfFolder);
440 if (SUCCEEDED(hr)) {
441 hr = IShellFolder_GetAttributesOf(psfFolder, 0, NULL, pdwAttributes);
442 IShellFolder_Release(psfFolder);
444 IShellFolder_Release(psfDesktop);
446 if (FAILED(hr)) return FALSE;
447 } else {
448 lResult = RegQueryValueExW(hSFKey, L"Attributes", 0, NULL, (LPBYTE)&dwTemp, &dwLen);
449 RegCloseKey(hSFKey);
450 if (lResult == ERROR_SUCCESS) {
451 *pdwAttributes &= dwTemp;
452 } else {
453 return FALSE;
457 TRACE("-- *pdwAttributes == 0x%08lx\n", *pdwAttributes);
459 return TRUE;