]> git.mxchange.org Git - simgear.git/commitdiff
WindowsXP workaround for SHGetKnowwFolder
authorJames Turner <zakalawe@mac.com>
Thu, 9 Apr 2015 14:33:56 +0000 (15:33 +0100)
committerJames Turner <zakalawe@mac.com>
Thu, 9 Apr 2015 14:33:56 +0000 (15:33 +0100)
Fix from xDraconian after a bug report from Aleesandro.

simgear/misc/sg_path.cxx

index 1cfe465c06e43cc8244a677d0e060a2d4eaaeb69..d1475b1142b0d20a3b840b6cc6f93fb1c4874cce 100644 (file)
@@ -57,7 +57,7 @@ static const char sgSearchPathSep = ':';
 
 #ifdef _WIN32
 #include <ShlObj.h>         // for CSIDL
-#include <Knownfolders.h>   // for Standard Folder GUIDs
+#include <versionhelpers.h>
 
 static SGPath pathForCSIDL(int csidl, const SGPath& def)
 {
@@ -84,23 +84,34 @@ static SGPath pathForCSIDL(int csidl, const SGPath& def)
 
 static SGPath pathForKnownFolder(REFKNOWNFOLDERID folderId, const SGPath& def)
 {
-    // system call will allocate dynamic memory... which we must release when done
-    wchar_t* localFolder = 0;
-    if (SHGetKnownFolderPath(folderId, KF_FLAG_DEFAULT_PATH, NULL, &localFolder) == S_OK) {
-        // copy into local memory
-        char path[MAX_PATH];
-        size_t len;
-        if (wcstombs_s(&len, path, localFolder, MAX_PATH) != S_OK) {
-            path[0] = '\0';
-            SG_LOG(SG_GENERAL, SG_WARN, "WCS to MBS failed");
-        }
+    typedef HRESULT (WINAPI*PSHGKFP)(REFKNOWNFOLDERID, DWORD, HANDLE, PWSTR*);
+
+    HINSTANCE shellDll = LoadLibrary(TEXT("shell32"));
+    if (shellDll != NULL) {
+        PSHGKFP pSHGetKnownFolderPath = (PSHGKFP) GetProcAddress(shellDll, "SHGetKnownFolderPath");
+        if (pSHGetKnownFolderPath != NULL) {
+            // system call will allocate dynamic memory... which we must release when done
+            wchar_t* localFolder = 0;
+
+            if (pSHGetKnownFolderPath(folderId, KF_FLAG_DEFAULT_PATH, NULL, &localFolder) == S_OK) {
+                // copy into local memory
+                char path[MAX_PATH];
+                size_t len;
+                if (wcstombs_s(&len, path, localFolder, MAX_PATH) != S_OK) {
+                    path[0] = '\0';
+                    SG_LOG(SG_GENERAL, SG_WARN, "WCS to MBS failed");
+                }
 
-        SGPath folder_path = SGPath(path, def.getPermissionChecker());
+                SGPath folder_path = SGPath(path, def.getPermissionChecker());
 
-        // release dynamic memory
-        CoTaskMemFree(static_cast<void*>(localFolder));
+                // release dynamic memory
+                CoTaskMemFree(static_cast<void*>(localFolder));
+
+                return folder_path;
+            }
+        }
 
-        return folder_path;
+        FreeLibrary(shellDll);
     }
 
     return def;
@@ -730,31 +741,35 @@ SGPath SGPath::standardLocation(StandardLocation type, const SGPath& def)
   {
     case HOME:
       return home(def);
+
 #ifdef _WIN32
-#   if (WINVER > 0x0501)
-    case DESKTOP:
-        return pathForKnownFolder(FOLDERID_Desktop, def);
-    case DOWNLOADS:
-        return pathForKnownFolder(FOLDERID_Downloads, def);
-    case DOCUMENTS:
-        return pathForKnownFolder(FOLDERID_Documents, def);
-    case PICTURES:
-        return pathForKnownFolder(FOLDERID_Pictures, def);
-#   else
     case DESKTOP:
-      return pathForCSIDL(CSIDL_DESKTOPDIRECTORY, def);
+        if (IsWindowsVistaOrGreater())
+            return pathForKnownFolder(FOLDERID_Desktop, def);
+
+        return pathForCSIDL(CSIDL_DESKTOPDIRECTORY, def);
+
     case DOWNLOADS:
-      // TODO use KnownFolders
-      // http://msdn.microsoft.com/en-us/library/bb776911%28v=vs.85%29.aspx
-      if( !def.isNull() )
-        return def;
+        if (IsWindowsVistaOrGreater())
+            return pathForKnownFolder(FOLDERID_Downloads, def);
+
+        if (!def.isNull())
+            return def;
+
+        return pathForCSIDL(CSIDL_DESKTOPDIRECTORY, def);
 
-      return pathForCSIDL(CSIDL_DESKTOPDIRECTORY, def);
     case DOCUMENTS:
-      return pathForCSIDL(CSIDL_MYDOCUMENTS, def);
+        if (IsWindowsVistaOrGreater())
+            return pathForKnownFolder(FOLDERID_Documents, def);
+
+        return pathForCSIDL(CSIDL_MYDOCUMENTS, def);
+
     case PICTURES:
-      return pathForCSIDL(CSIDL_MYPICTURES, def);
-#   endif
+        if (IsWindowsVistaOrGreater())
+            return pathForKnownFolder(FOLDERID_Pictures, def);
+
+        return pathForCSIDL(CSIDL_MYPICTURES, def);
+
 #elif __APPLE__
       // since this is C++, we can't include NSPathUtilities.h to access the enum
       // values, so hard-coding them here (they are stable, don't worry)