2019-10-21 19:36:05 +00:00
/*
* * i_specialpaths . cpp
* * Gets special system folders where data should be stored . ( Windows version )
* *
* * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* * Copyright 2013 - 2016 Randy Heit
* * Copyright 2016 Christoph Oelckers
* * All rights reserved .
* *
* * Redistribution and use in source and binary forms , with or without
* * modification , are permitted provided that the following conditions
* * are met :
* *
* * 1. Redistributions of source code must retain the above copyright
* * notice , this list of conditions and the following disclaimer .
* * 2. Redistributions in binary form must reproduce the above copyright
* * notice , this list of conditions and the following disclaimer in the
* * documentation and / or other materials provided with the distribution .
* * 3. The name of the author may not be used to endorse or promote products
* * derived from this software without specific prior written permission .
* *
* * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ` ` AS IS ' ' AND ANY EXPRESS OR
* * IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES
* * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED .
* * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT , INDIRECT ,
* * INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT
* * NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* * DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* * THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* * ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF
* * THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
* * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* *
*/
# include <windows.h>
# include <lmcons.h>
# include <shlobj.h>
2019-10-29 18:53:46 +00:00
# include <Shlwapi.h>
2022-11-06 10:46:26 +00:00
# include <VersionHelpers.h>
2019-10-21 19:36:05 +00:00
# include "i_specialpaths.h"
# include "printf.h"
# include "cmdlib.h"
2020-04-11 21:50:43 +00:00
# include "findfile.h"
2019-12-26 13:04:53 +00:00
# include "version.h" // for GAMENAME
2022-11-06 10:46:26 +00:00
# include "gstrings.h"
# include "i_mainwindow.h"
# include "engineerrors.h"
2019-10-21 19:36:05 +00:00
2022-11-06 10:46:26 +00:00
static int isportable = - 1 ;
2019-10-21 19:36:05 +00:00
//===========================================================================
//
// IsProgramDirectoryWritable
//
// If the program directory is writable, then dump everything in there for
// historical reasons. Otherwise, known folders get used instead.
//
//===========================================================================
2022-11-06 10:46:26 +00:00
bool IsPortable ( )
2019-10-21 19:36:05 +00:00
{
// Cache this value so the semantics don't change during a single run
// of the program. (e.g. Somebody could add write access while the
// program is running.)
HANDLE file ;
2022-11-06 10:46:26 +00:00
if ( isportable > = 0 )
2019-10-21 19:36:05 +00:00
{
2022-11-06 10:46:26 +00:00
return ! ! isportable ;
2019-10-21 19:36:05 +00:00
}
2022-11-06 10:46:26 +00:00
2021-09-18 10:20:28 +00:00
// Consider 'Program Files' read only without actually checking.
bool found = false ;
for ( auto p : { L " ProgramFiles " , L " ProgramFiles(x86) " } )
2019-10-21 19:36:05 +00:00
{
2021-09-18 10:20:28 +00:00
wchar_t buffer1 [ 256 ] ;
if ( GetEnvironmentVariable ( p , buffer1 , 256 ) )
{
FString envpath ( buffer1 ) ;
FixPathSeperator ( envpath ) ;
if ( progdir . MakeLower ( ) . IndexOf ( envpath . MakeLower ( ) ) = = 0 )
{
2022-11-06 10:46:26 +00:00
isportable = false ;
return false ;
2021-09-18 10:20:28 +00:00
}
}
}
2022-11-06 10:46:26 +00:00
// A portable INI means that this storage location should also be portable if the file can be written to.
2023-02-11 11:06:58 +00:00
FStringf path ( " %s " GAMENAMELOWERCASE " _portable.ini " , progdir . GetChars ( ) ) ;
2022-11-06 10:46:26 +00:00
if ( FileExists ( path ) )
2021-09-18 10:20:28 +00:00
{
2022-11-06 10:46:26 +00:00
file = CreateFile ( path . WideString ( ) . c_str ( ) , GENERIC_READ | GENERIC_WRITE , 0 , NULL ,
OPEN_EXISTING , FILE_ATTRIBUTE_NORMAL , NULL ) ;
2021-09-18 10:20:28 +00:00
if ( file ! = INVALID_HANDLE_VALUE )
{
CloseHandle ( file ) ;
2022-11-06 10:46:26 +00:00
if ( ! batchrun ) Printf ( " Using portable configuration \n " ) ;
isportable = true ;
return true ;
2021-09-18 10:20:28 +00:00
}
2019-10-21 19:36:05 +00:00
}
2022-11-06 10:46:26 +00:00
isportable = false ;
return false ;
2019-10-21 19:36:05 +00:00
}
//===========================================================================
//
// GetKnownFolder
//
2022-11-06 10:46:26 +00:00
// Returns the known_folder from SHGetKnownFolderPath
2019-10-21 19:36:05 +00:00
//
//===========================================================================
2022-11-06 10:46:26 +00:00
FString GetKnownFolder ( int shell_folder , REFKNOWNFOLDERID known_folder , bool create )
2019-10-21 19:36:05 +00:00
{
PWSTR wpath ;
if ( FAILED ( SHGetKnownFolderPath ( known_folder , create ? KF_FLAG_CREATE : 0 , NULL , & wpath ) ) )
{
2022-11-06 10:46:26 +00:00
// This should never be triggered unless the OS was compromised
I_FatalError ( " Unable to retrieve known folder. " ) ;
2019-10-21 19:36:05 +00:00
}
2022-11-06 10:46:26 +00:00
FString path = FString ( wpath ) ;
FixPathSeperator ( path ) ;
2019-10-21 19:36:05 +00:00
CoTaskMemFree ( wpath ) ;
2022-11-06 10:46:26 +00:00
return path ;
2019-10-21 19:36:05 +00:00
}
//===========================================================================
//
// M_GetAppDataPath Windows
//
// Returns the path for the AppData folder.
//
//===========================================================================
FString M_GetAppDataPath ( bool create )
{
2022-11-06 10:46:26 +00:00
FString path = GetKnownFolder ( CSIDL_LOCAL_APPDATA , FOLDERID_LocalAppData , create ) ;
2019-10-21 19:36:05 +00:00
path + = " / " GAMENAMELOWERCASE ;
if ( create )
{
2023-10-07 16:44:31 +00:00
CreatePath ( path . GetChars ( ) ) ;
2019-10-21 19:36:05 +00:00
}
return path ;
}
2020-04-23 19:18:40 +00:00
//===========================================================================
//
// M_GetCachePath Windows
//
// Returns the path for cache GL nodes.
//
//===========================================================================
FString M_GetCachePath ( bool create )
{
2022-11-06 10:46:26 +00:00
FString path = GetKnownFolder ( CSIDL_LOCAL_APPDATA , FOLDERID_LocalAppData , create ) ;
2020-04-23 19:18:40 +00:00
// Don't use GAME_DIR and such so that ZDoom and its child ports can
// share the node cache.
path + = " /zdoom/cache " ;
if ( create )
{
2023-10-07 16:44:31 +00:00
CreatePath ( path . GetChars ( ) ) ;
2020-04-23 19:18:40 +00:00
}
return path ;
}
2019-10-21 19:36:05 +00:00
//===========================================================================
//
// M_GetAutoexecPath Windows
//
// Returns the expected location of autoexec.cfg.
//
//===========================================================================
FString M_GetAutoexecPath ( )
{
return " $PROGDIR/autoexec.cfg " ;
}
2022-11-06 10:46:26 +00:00
//===========================================================================
//
// M_GetOldConfigPath
//
// Check if we have a config in a place that's no longer used.
//
//===========================================================================
FString M_GetOldConfigPath ( int & type )
{
FString path ;
HRESULT hr ;
// construct "$PROGDIR/-$USER.ini"
WCHAR uname [ UNLEN + 1 ] ;
DWORD unamelen = UNLEN ;
path = progdir ;
hr = GetUserNameW ( uname , & unamelen ) ;
if ( SUCCEEDED ( hr ) & & uname [ 0 ] ! = 0 )
{
// Is it valid for a user name to have slashes?
// Check for them and substitute just in case.
auto probe = uname ;
while ( * probe ! = 0 )
{
if ( * probe = = ' \\ ' | | * probe = = ' / ' )
* probe = ' _ ' ;
+ + probe ;
}
path < < GAMENAMELOWERCASE " - " < < FString ( uname ) < < " .ini " ;
type = 0 ;
if ( FileExists ( path ) )
return path ;
}
// Check in app data where this was previously stored.
// We actually prefer to store the config in a more visible place so this is no longer used.
path = GetKnownFolder ( CSIDL_APPDATA , FOLDERID_RoamingAppData , true ) ;
path + = " / " GAME_DIR " / " GAMENAMELOWERCASE " .ini " ;
type = 1 ;
if ( FileExists ( path ) )
return path ;
return " " ;
}
//===========================================================================
//
// M_MigrateOldConfig
//
// Ask the user what to do with their old config.
//
//===========================================================================
int M_MigrateOldConfig ( )
{
int selection = IDCANCEL ;
auto globalstr = L " Move to Users/ folder " ;
auto portablestr = L " Convert to portable installation " ;
auto cancelstr = L " Cancel " ;
auto titlestr = L " Migrate existing configuration " ;
auto infostr = L " " GAMENAME " found a user specific config in the game folder " ;
const TASKDIALOG_BUTTON buttons [ ] = { { IDYES , globalstr } , { IDNO , portablestr } , { IDCANCEL , cancelstr } } ;
TASKDIALOGCONFIG taskDialogConfig = { } ;
taskDialogConfig . cbSize = sizeof ( TASKDIALOGCONFIG ) ;
taskDialogConfig . pszMainIcon = TD_WARNING_ICON ;
taskDialogConfig . pButtons = buttons ;
taskDialogConfig . cButtons = countof ( buttons ) ;
taskDialogConfig . pszWindowTitle = titlestr ;
taskDialogConfig . pszContent = infostr ;
taskDialogConfig . hwndParent = mainwindow . GetHandle ( ) ;
taskDialogConfig . dwFlags = TDF_USE_COMMAND_LINKS ;
TaskDialogIndirect ( & taskDialogConfig , & selection , NULL , NULL ) ;
if ( selection = = IDYES | | selection = = IDNO ) return selection ;
throw CExitEvent ( 3 ) ;
}
2019-10-21 19:36:05 +00:00
//===========================================================================
//
// M_GetConfigPath Windows
//
// Returns the path to the config file. On Windows, this can vary for reading
2019-12-26 13:04:53 +00:00
// vs writing. i.e. If the user specific ini does not exist, it will try
// to read from a neutral version, but never write to it.
2019-10-21 19:36:05 +00:00
//
//===========================================================================
FString M_GetConfigPath ( bool for_reading )
{
2022-11-06 10:46:26 +00:00
if ( IsPortable ( ) )
2019-10-21 19:36:05 +00:00
{
2022-11-06 10:46:26 +00:00
return FStringf ( " %s " GAMENAMELOWERCASE " _portable.ini " , progdir . GetChars ( ) ) ;
2019-10-21 19:36:05 +00:00
}
// Construct a user-specific config name
2022-11-06 10:46:26 +00:00
FString path = GetKnownFolder ( CSIDL_APPDATA , FOLDERID_Documents , true ) ;
path + = " /My Games/ " GAME_DIR ;
2023-10-07 16:44:31 +00:00
CreatePath ( path . GetChars ( ) ) ;
2022-11-06 10:46:26 +00:00
path + = " / " GAMENAMELOWERCASE " .ini " ;
if ( ! for_reading | | FileExists ( path ) )
return path ;
2019-10-21 19:36:05 +00:00
2022-11-06 10:46:26 +00:00
// No config was found in the accepted locations.
// Look in previously valid places to see if we have something we can migrate
int type = 0 ;
FString oldpath = M_GetOldConfigPath ( type ) ;
if ( ! oldpath . IsEmpty ( ) )
{
if ( type = = 0 )
2019-10-21 19:36:05 +00:00
{
2022-11-06 10:46:26 +00:00
// If we find a local per-user config, ask the user what to do with it.
int action = M_MigrateOldConfig ( ) ;
if ( action = = IDNO )
2019-10-21 19:36:05 +00:00
{
2022-11-06 10:46:26 +00:00
path . Format ( " %s " GAMENAMELOWERCASE " _portable.ini " , progdir . GetChars ( ) ) ;
isportable = true ;
2019-10-21 19:36:05 +00:00
}
}
2023-10-07 16:44:31 +00:00
bool res = MoveFileExW ( WideString ( oldpath . GetChars ( ) ) . c_str ( ) , WideString ( path . GetChars ( ) ) . c_str ( ) , MOVEFILE_COPY_ALLOWED ) ;
2022-11-06 10:46:26 +00:00
if ( res ) return path ;
else return oldpath ; // if we cannot move, just use the config where it was. It won't be written back, though and never be used again if a new one gets saved.
2019-10-21 19:36:05 +00:00
}
2022-11-06 10:46:26 +00:00
// Fall back to the global template if nothing was found.
// If we are reading the config file, check if it exists. If not, fallback to base version.
2019-10-21 19:36:05 +00:00
if ( for_reading )
{
if ( ! FileExists ( path ) )
{
path = progdir ;
path < < GAMENAMELOWERCASE " .ini " ;
}
}
return path ;
}
//===========================================================================
//
// M_GetScreenshotsPath Windows
//
// Returns the path to the default screenshots directory.
//
//===========================================================================
FString M_GetScreenshotsPath ( )
{
FString path ;
2022-11-06 10:46:26 +00:00
if ( IsPortable ( ) )
2019-10-21 19:36:05 +00:00
{
2022-08-26 16:28:22 +00:00
path < < progdir < < " Screenshots/ " ;
2019-10-21 19:36:05 +00:00
}
2022-11-06 10:46:26 +00:00
else if ( IsWindows8OrGreater ( ) )
2019-10-21 19:36:05 +00:00
{
2022-11-06 10:46:26 +00:00
path = GetKnownFolder ( - 1 , FOLDERID_Screenshots , true ) ;
2019-11-01 23:38:30 +00:00
path < < " / " GAMENAME " / " ;
2019-10-21 19:36:05 +00:00
}
2022-11-06 10:46:26 +00:00
else
2019-10-21 19:36:05 +00:00
{
2022-11-06 10:46:26 +00:00
path = GetKnownFolder ( CSIDL_MYPICTURES , FOLDERID_Pictures , true ) ;
2019-11-01 23:38:30 +00:00
path < < " /Screenshots/ " GAMENAME " / " ;
2019-10-21 19:36:05 +00:00
}
2023-10-07 16:44:31 +00:00
CreatePath ( path . GetChars ( ) ) ;
2019-10-21 19:36:05 +00:00
return path ;
}
//===========================================================================
//
// M_GetSavegamesPath Windows
//
// Returns the path to the default save games directory.
//
//===========================================================================
FString M_GetSavegamesPath ( )
{
FString path ;
2022-11-06 10:46:26 +00:00
if ( IsPortable ( ) )
2019-11-01 23:38:30 +00:00
{
2020-02-05 11:56:54 +00:00
path < < progdir < < " Save/ " ;
2019-10-21 19:36:05 +00:00
}
// Try standard Saved Games folder
else
{
2022-11-06 10:46:26 +00:00
path = GetKnownFolder ( - 1 , FOLDERID_SavedGames , true ) ;
path < < " / " GAMENAME " / " ;
2019-10-21 19:36:05 +00:00
}
return path ;
}
//===========================================================================
//
// M_GetDocumentsPath Windows
//
// Returns the path to the default documents directory.
//
//===========================================================================
FString M_GetDocumentsPath ( )
{
FString path ;
2022-11-06 10:46:26 +00:00
if ( IsPortable ( ) )
2019-10-21 19:36:05 +00:00
{
return progdir ;
}
// Try defacto My Documents/My Games folder
2022-11-06 10:46:26 +00:00
else
2019-10-21 19:36:05 +00:00
{
2022-11-06 10:46:26 +00:00
// I assume since this isn't a standard folder, it doesn't have a localized name either.
path = GetKnownFolder ( CSIDL_PERSONAL , FOLDERID_Documents , true ) ;
2019-11-01 23:38:30 +00:00
path < < " /My Games/ " GAMENAME " / " ;
2023-10-07 16:44:31 +00:00
CreatePath ( path . GetChars ( ) ) ;
2019-10-21 19:36:05 +00:00
}
return path ;
}
2019-10-29 18:53:46 +00:00
2019-11-01 23:38:30 +00:00
//===========================================================================
//
2020-04-23 19:18:40 +00:00
// M_GetDemoPath Windows
2019-11-01 23:38:30 +00:00
//
2020-04-23 19:18:40 +00:00
// Returns the path to the default demp directory.
2019-11-01 23:38:30 +00:00
//
//===========================================================================
FString M_GetDemoPath ( )
{
FString path ;
// A portable INI means that this storage location should also be portable.
2022-11-06 10:46:26 +00:00
if ( IsPortable ( ) )
2019-11-01 23:38:30 +00:00
{
2020-02-05 11:57:17 +00:00
path < < progdir < < " Demos/ " ;
2019-11-01 23:38:30 +00:00
}
else
// Try defacto My Documents/My Games folder
{
2022-11-06 10:46:26 +00:00
// I assume since this isn't a standard folder, it doesn't have a localized name either.
path = GetKnownFolder ( CSIDL_PERSONAL , FOLDERID_Documents , true ) ;
2020-02-05 11:57:17 +00:00
path < < " /My Games/ " GAMENAME " / " ;
2019-11-01 23:38:30 +00:00
}
2020-02-05 11:57:17 +00:00
2019-11-01 23:38:30 +00:00
return path ;
}
2020-01-03 13:19:52 +00:00
//===========================================================================
//
// M_NormalizedPath
//
// Normalizes the given path and returns the result.
//
//===========================================================================
FString M_GetNormalizedPath ( const char * path )
{
std : : wstring wpath = WideString ( path ) ;
wchar_t buffer [ MAX_PATH ] ;
GetFullPathNameW ( wpath . c_str ( ) , MAX_PATH , buffer , nullptr ) ;
FString result ( buffer ) ;
FixPathSeperator ( result ) ;
return result ;
}