{* UltraStar Deluxe - Karaoke Game
*
* UltraStar Deluxe is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; see the file COPYING. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*
* $URL$
* $Id$
*}
unit UPlatformWindows;
interface
{$IFDEF FPC}
{$MODE Delphi}
{$ENDIF}
{$I switches.inc}
// turn off messages for platform specific symbols
{$WARN SYMBOL_PLATFORM OFF}
uses
Classes,
UPlatform,
UPath;
type
TPlatformWindows = class(TPlatform)
private
UseLocalDirs: boolean;
function GetSpecialPath(CSIDL: integer): IPath;
procedure DetectLocalExecution();
public
procedure Init; override;
function TerminateIfAlreadyRunning(var WndTitle: String): Boolean; override;
procedure AddLibrarySearchPath(const DLLPath: IPath); override;
function GetLogPath: IPath; override;
function GetGameSharedPath: IPath; override;
function GetGameUserPath: IPath; override;
end;
implementation
uses
SysUtils,
ShlObj,
Windows,
UPathUtils,
UConfig;
procedure TPlatformWindows.Init;
begin
inherited Init();
DetectLocalExecution();
end;
//------------------------------
//Start more than One Time Prevention
//------------------------------
function TPlatformWindows.TerminateIfAlreadyRunning(var WndTitle: String): Boolean;
var
hWnd: THandle;
I: Integer;
begin
Result := false;
hWnd:= FindWindow(nil, PChar(WndTitle));
//Programm already started
if (hWnd <> 0) then
begin
I := Messagebox(0, PChar('Another Instance of Ultrastar is already running. Continue ?'), PChar(WndTitle), MB_ICONWARNING or MB_YESNO);
if (I = IDYes) then
begin
I := 1;
repeat
Inc(I);
hWnd := FindWindow(nil, PChar(WndTitle + ' Instance ' + InttoStr(I)));
until (hWnd = 0);
WndTitle := WndTitle + ' Instance ' + InttoStr(I);
end
else
Result := true;
end;
end;
(**
* Returns the path of a special folder.
*
* Some Folder IDs:
* CSIDL_APPDATA (e.g. C:\Documents and Settings\username\Application Data)
* CSIDL_LOCAL_APPDATA (e.g. C:\Documents and Settings\username\Local Settings\Application Data)
* CSIDL_PROFILE (e.g. C:\Documents and Settings\username)
* CSIDL_PERSONAL (e.g. C:\Documents and Settings\username\My Documents)
* CSIDL_MYMUSIC (e.g. C:\Documents and Settings\username\My Documents\My Music)
*)
function TPlatformWindows.GetSpecialPath(CSIDL: integer): IPath;
var
Buffer: array [0..MAX_PATH-1] of WideChar;
begin
if (SHGetSpecialFolderPathW(0, @Buffer, CSIDL, false)) then
Result := Path(Buffer)
else
Result := PATH_NONE;
end;
{**
* Detects whether the was executed locally or globally.
* - Local mode:
* - Condition:
* - config.ini is writable or creatable in the directory of the executable.
* - Examples:
* - The USDX zip-archive has been unpacked to a directory with write.
* permissions
* - XP: USDX was installed to %ProgramFiles% and the user is an admin.
* - USDX is started from an external HD- or flash-drive
* - Behavior:
* Config files like config.ini or score db reside in the directory of the
* executable. This is useful to enable windows users to have a portable
* installation e.g. on an external hdd.
* This is also the default behaviour of usdx prior to version 1.1
* - Global mode:
* - Condition:
* - config.ini is not writable.
* - Examples:
* - Vista/7: USDX was installed to %ProgramFiles%.
* - XP: USDX was installed to %ProgramFiles% and the user is not an admin.
* - USDX is started from CD
* - Behavior:
* - The config files are in a separate folder (e.g. %APPDATA%\ultrastardx)
*
* On windows, resources (themes, language-files)
* reside in the directory of the executable in any case
*
* Sets UseLocalDirs to true if the game is executed locally, false otherwise.
*}
procedure TPlatformWindows.DetectLocalExecution();
var
LocalDir, ConfigIni: IPath;
Handle: TFileHandle;
begin
LocalDir := GetExecutionDir();
ConfigIni := LocalDir.Append('config.ini');
// check if config.ini is writable or creatable, if so use local dirs
UseLocalDirs := false;
if (ConfigIni.Exists()) then
begin
// do not use a read-only config file
if (not ConfigIni.IsReadOnly()) then
begin
// Just open the file in read-write mode to be sure that we have access
// rights for it.
// Note: Do not use IsReadOnly() as it does not check file privileges, so
// a non-read-only file might not be writable for us.
Handle := ConfigIni.Open(fmOpenReadWrite);
if (Handle <> -1) then
begin
FileClose(Handle);
UseLocalDirs := true;
end;
end;
end
else // config.ini does not exist
begin
// try to create config.ini
Handle := ConfigIni.CreateFile();
if (Handle <> -1) then
begin
FileClose(Handle);
UseLocalDirs := true;
end;
end;
end;
function TPlatformWindows.GetLogPath: IPath;
begin
Result := GetGameUserPath;
end;
function TPlatformWindows.GetGameSharedPath: IPath;
begin
Result := GetExecutionDir();
end;
function TPlatformWindows.GetGameUserPath: IPath;
begin
if UseLocalDirs then
Result := GetExecutionDir()
else
Result := GetSpecialPath(CSIDL_APPDATA).Append('ultrastardx', pdAppend);
end;
function SetDllDirectory(lpPathName: PWideChar): Bool; stdcall;
external kernel32 name 'SetDllDirectoryW';
procedure TPlatformWindows.AddLibrarySearchPath(const DLLPath: IPath);
begin
SetDllDirectory(PWideChar(DLLPath.ToWide()));
end;
end.