1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: msw/stdpaths.cpp
3 // Purpose: wxStandardPaths implementation for Win32
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 2004 Vadim Zeitlin <vadim@wxwindows.org>
9 // License: wxWindows license
10 ///////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // for compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
31 #include "wx/dynlib.h"
32 #include "wx/filename.h"
34 #include "wx/stdpaths.h"
36 #include "wx/msw/private.h"
37 #include "wx/msw/wrapshl.h"
39 // ----------------------------------------------------------------------------
41 // ----------------------------------------------------------------------------
43 typedef HRESULT (WINAPI
*SHGetFolderPath_t
)(HWND
, int, HANDLE
, DWORD
, LPTSTR
);
44 typedef HRESULT (WINAPI
*SHGetSpecialFolderPath_t
)(HWND
, LPTSTR
, int, BOOL
);
46 // ----------------------------------------------------------------------------
48 // ----------------------------------------------------------------------------
50 // used in our wxLogTrace messages
51 static const wxChar
*TRACE_MASK
= _T("stdpaths");
53 #ifndef CSIDL_LOCAL_APPDATA
54 #define CSIDL_LOCAL_APPDATA 0x001c
57 #ifndef CSIDL_COMMON_APPDATA
58 #define CSIDL_COMMON_APPDATA 0x0023
61 #ifndef CSIDL_PROGRAM_FILES
62 #define CSIDL_PROGRAM_FILES 0x0026
65 // ----------------------------------------------------------------------------
67 // ----------------------------------------------------------------------------
73 pSHGetFolderPath
= NULL
;
74 pSHGetSpecialFolderPath
= NULL
;
78 SHGetFolderPath_t pSHGetFolderPath
;
79 SHGetSpecialFolderPath_t pSHGetSpecialFolderPath
;
84 // in spite of using a static variable, this is MT-safe as in the worst case it
85 // results in initializing the function pointer several times -- but this is
87 static ShellFunctions gs_shellFuncs
;
89 // ----------------------------------------------------------------------------
91 // ----------------------------------------------------------------------------
93 static void ResolveShellFunctions()
95 // don't give errors if the functions are unavailable, we're ready to deal
99 // start with the newest functions, fall back to the oldest ones
101 // first check for SHGetFolderPath (shell32.dll 5.0)
102 wxDynamicLibrary
dllShell32(_T("shell32"));
103 if ( !dllShell32
.IsLoaded() )
105 wxLogTrace(TRACE_MASK
, _T("Failed to load shell32.dll"));
109 static const wchar_t UNICODE_SUFFIX
= L
'W';
111 static const char UNICODE_SUFFIX
= 'A';
112 #endif // Unicode/!Unicode
114 wxString
funcname(_T("SHGetFolderPath"));
115 gs_shellFuncs
.pSHGetFolderPath
=
116 (SHGetFolderPath_t
)dllShell32
.GetSymbol(funcname
+ UNICODE_SUFFIX
);
118 // then for SHGetSpecialFolderPath (shell32.dll 4.71)
119 if ( !gs_shellFuncs
.pSHGetFolderPath
)
121 funcname
= _T("SHGetSpecialFolderPath");
122 gs_shellFuncs
.pSHGetSpecialFolderPath
= (SHGetSpecialFolderPath_t
)
123 dllShell32
.GetSymbol(funcname
+ UNICODE_SUFFIX
);
126 // finally we fall back on SHGetSpecialFolderLocation (shell32.dll 4.0),
127 // but we don't need to test for it -- it is available even under Win95
129 // shell32.dll is going to be unloaded, but it still remains in memory
130 // because we also link to it statically, so it's ok
132 gs_shellFuncs
.initialized
= true;
135 // ============================================================================
136 // wxStandardPaths implementation
137 // ============================================================================
139 // ----------------------------------------------------------------------------
141 // ----------------------------------------------------------------------------
144 wxString
wxStandardPaths::DoGetDirectory(int csidl
)
146 if ( !gs_shellFuncs
.initialized
)
147 ResolveShellFunctions();
152 // test whether the function is available during compile-time (it must be
153 // defined as either "SHGetFolderPathA" or "SHGetFolderPathW")
154 #ifdef SHGetFolderPath
155 // and now test whether we have it during run-time
156 if ( gs_shellFuncs
.pSHGetFolderPath
)
158 hr
= gs_shellFuncs
.pSHGetFolderPath
160 NULL
, // parent window, not used
162 NULL
, // access token (current user)
163 SHGFP_TYPE_CURRENT
, // current path, not just default value
164 wxStringBuffer(dir
, MAX_PATH
)
167 // somewhat incredibly, the error code in the Unicode version is
168 // different from the one in ASCII version for this function
175 // directory doesn't exist, maybe we can get its default value?
176 hr
= gs_shellFuncs
.pSHGetFolderPath
182 wxStringBuffer(dir
, MAX_PATH
)
186 #endif // SHGetFolderPath
188 #ifdef SHGetSpecialFolderPath
189 if ( FAILED(hr
) && gs_shellFuncs
.pSHGetSpecialFolderPath
)
191 hr
= gs_shellFuncs
.pSHGetSpecialFolderPath
193 NULL
, // parent window
194 wxStringBuffer(dir
, MAX_PATH
),
196 FALSE
// don't create if doesn't exist
199 #endif // SHGetSpecialFolderPath
201 // SHGetSpecialFolderLocation should be available with all compilers and
202 // under all Win32 systems, so don't test for it (and as it doesn't exist
203 // in "A" and "W" versions anyhow, testing would be more involved, too)
207 hr
= SHGetSpecialFolderLocation(NULL
, csidl
, &pidl
);
211 // creating this temp object has (nice) side effect of freeing pidl
212 dir
= wxItemIdList(pidl
).GetPath();
220 wxString
wxStandardPaths::AppendAppName(const wxString
& dir
)
222 wxString
subdir(dir
);
224 // empty string indicates that an error has occured, don't touch it then
225 if ( !subdir
.empty() )
227 const wxString appname
= wxTheApp
->GetAppName();
228 if ( !appname
.empty() )
229 subdir
<< _T('\\') << appname
;
235 // ----------------------------------------------------------------------------
237 // ----------------------------------------------------------------------------
239 wxString
wxStandardPaths::GetConfigDir() const
241 return AppendAppName(DoGetDirectory(CSIDL_COMMON_APPDATA
));
244 wxString
wxStandardPaths::GetUserConfigDir() const
246 return DoGetDirectory(CSIDL_APPDATA
);
249 wxString
wxStandardPaths::GetDataDir() const
251 return AppendAppName(DoGetDirectory(CSIDL_PROGRAM_FILES
));
254 wxString
wxStandardPaths::GetUserDataDir() const
256 return AppendAppName(GetUserConfigDir());
259 wxString
wxStandardPaths::GetUserLocalDataDir() const
261 return AppendAppName(DoGetDirectory(CSIDL_LOCAL_APPDATA
));
264 wxString
wxStandardPaths::GetPluginsDir() const
266 return wxFileName(wxGetFullModuleName()).GetPath();
270 // ============================================================================
271 // wxStandardPathsWin16 implementation
272 // ============================================================================
274 wxString
wxStandardPathsWin16::GetConfigDir() const
276 // this is for compatibility with earlier wxFileConfig versions
277 // which used the Windows directory for the global files
279 if ( !::GetWindowsDirectory(wxStringBuffer(dir
, MAX_PATH
), MAX_PATH
) )
281 wxLogLastError(_T("GetWindowsDirectory"));
287 wxString
wxStandardPathsWin16::GetUserConfigDir() const
289 // again, for wxFileConfig which uses $HOME for its user config file
290 return wxGetHomeDir();