- added beefy new ReadMapList() native
- admin menu now uses ReadMapList() - added UTIL_TrimWhitespace() to stringutils - moved GetFileTime() implementation to ILibrarySys - cleaned up sorting include a bit - removed adminmenu_maplist.ini, since it's specified by maplists.cfg - added maplists.cfg --HG-- extra : convert_revision : svn%3A39bc706e-5318-0410-9160-8a85361fbb7c/trunk%401744
This commit is contained in:
parent
da9168082c
commit
a8ecd7fea4
@ -1,5 +0,0 @@
|
||||
// adminmenu_maplist.ini
|
||||
//
|
||||
// List maps here to be added to the map and votemap sections of the admin menu
|
||||
//
|
||||
|
34
configs/maplists.cfg
Normal file
34
configs/maplists.cfg
Normal file
@ -0,0 +1,34 @@
|
||||
/**
|
||||
* Use this file to configure map lists.
|
||||
*
|
||||
* Each section is a map list that plugins can use. For example, the Admin Menu
|
||||
* requests an "admin menu" map list, and you can control which maps appear via
|
||||
* this file.
|
||||
*
|
||||
* Each section must have a property that explains where to read the maps from.
|
||||
* There are two properties:
|
||||
*
|
||||
* target - Redirect the request to another section.
|
||||
* file - Read a file of map names, in mapcycle.txt format.
|
||||
*
|
||||
* There is one section by default, called "mapcyclefile" - it is mapped to the
|
||||
* mapcycle.txt file, or whatever the contents of your mapcyclefile cvar is.
|
||||
*
|
||||
* If a plugin requests a map list file which doesn't exist, or is empty, SourceMod
|
||||
* tries the "default" section, and then the "mapcyclefile" section.
|
||||
*/
|
||||
"MapLists"
|
||||
{
|
||||
/**
|
||||
* Default requests go right to the mapcyclefile.
|
||||
*/
|
||||
"default"
|
||||
{
|
||||
"target" "mapcyclefile"
|
||||
}
|
||||
|
||||
"admin menu"
|
||||
{
|
||||
"file" "addons/sourcemod/configs/adminmenu_maplist.ini"
|
||||
}
|
||||
}
|
@ -29,7 +29,7 @@
|
||||
* Version: $Id$
|
||||
*/
|
||||
|
||||
#if defined DEBUG
|
||||
#if 0
|
||||
#include "sm_globals.h"
|
||||
#include "sourcemm_api.h"
|
||||
#include "Tlhelp32.h"
|
||||
|
@ -28,7 +28,7 @@ OBJECTS = AdminCache.cpp CDataPack.cpp ConCmdManager.cpp ConVarManager.cpp CoreC
|
||||
OBJECTS += smn_admin.cpp smn_bitbuffer.cpp smn_console.cpp smn_core.cpp \
|
||||
smn_datapacks.cpp smn_entities.cpp smn_events.cpp smn_fakenatives.cpp \
|
||||
smn_filesystem.cpp smn_float.cpp smn_functions.cpp smn_gameconfigs.cpp smn_halflife.cpp \
|
||||
smn_handles.cpp smn_keyvalues.cpp smn_banning.cpp \
|
||||
smn_handles.cpp smn_keyvalues.cpp smn_banning.cpp smn_maplist.cpp \
|
||||
smn_lang.cpp smn_player.cpp smn_string.cpp smn_sorting.cpp smn_textparse.cpp smn_timers.cpp \
|
||||
smn_usermsgs.cpp smn_menus.cpp smn_database.cpp smn_vector.cpp smn_adt_array.cpp
|
||||
OBJECTS += systems/ExtensionSys.cpp systems/ForwardSys.cpp systems/HandleSys.cpp \
|
||||
|
@ -28,7 +28,7 @@ OBJECTS = AdminCache.cpp CDataPack.cpp ConCmdManager.cpp ConVarManager.cpp CoreC
|
||||
OBJECTS += smn_admin.cpp smn_bitbuffer.cpp smn_console.cpp smn_core.cpp \
|
||||
smn_datapacks.cpp smn_entities.cpp smn_events.cpp smn_fakenatives.cpp \
|
||||
smn_filesystem.cpp smn_float.cpp smn_functions.cpp smn_gameconfigs.cpp smn_halflife.cpp \
|
||||
smn_handles.cpp smn_keyvalues.cpp smn_banning.cpp \
|
||||
smn_handles.cpp smn_keyvalues.cpp smn_banning.cpp smn_maplist.cpp \
|
||||
smn_lang.cpp smn_player.cpp smn_string.cpp smn_sorting.cpp smn_textparse.cpp smn_timers.cpp \
|
||||
smn_usermsgs.cpp smn_menus.cpp smn_database.cpp smn_vector.cpp smn_adt_array.cpp
|
||||
OBJECTS += systems/ExtensionSys.cpp systems/ForwardSys.cpp systems/HandleSys.cpp \
|
||||
|
@ -28,7 +28,7 @@ OBJECTS = AdminCache.cpp CDataPack.cpp ConCmdManager.cpp ConVarManager.cpp CoreC
|
||||
OBJECTS += smn_admin.cpp smn_bitbuffer.cpp smn_console.cpp smn_core.cpp \
|
||||
smn_datapacks.cpp smn_entities.cpp smn_events.cpp smn_fakenatives.cpp \
|
||||
smn_filesystem.cpp smn_float.cpp smn_functions.cpp smn_gameconfigs.cpp smn_halflife.cpp \
|
||||
smn_handles.cpp smn_keyvalues.cpp smn_banning.cpp \
|
||||
smn_handles.cpp smn_keyvalues.cpp smn_banning.cpp smn_maplist.cpp \
|
||||
smn_lang.cpp smn_player.cpp smn_string.cpp smn_sorting.cpp smn_textparse.cpp smn_timers.cpp \
|
||||
smn_usermsgs.cpp smn_menus.cpp smn_database.cpp smn_vector.cpp smn_adt_array.cpp
|
||||
OBJECTS += systems/ExtensionSys.cpp systems/ForwardSys.cpp systems/HandleSys.cpp \
|
||||
|
@ -1378,6 +1378,10 @@
|
||||
RelativePath="..\smn_lang.cpp"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath="..\smn_maplists.cpp"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath="..\smn_menus.cpp"
|
||||
>
|
||||
|
@ -1275,3 +1275,34 @@ char *UTIL_ReplaceEx(char *subject, size_t maxLen, const char *search, size_t se
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
char *UTIL_TrimWhitespace(char *str, size_t &len)
|
||||
{
|
||||
char *end = str + len - 1;
|
||||
|
||||
if (!len)
|
||||
{
|
||||
return str;
|
||||
}
|
||||
|
||||
/* Iterate backwards through string until we reach first non-whitespace char */
|
||||
while (end >= str && textparsers->IsWhitespace(end))
|
||||
{
|
||||
end--;
|
||||
len--;
|
||||
}
|
||||
|
||||
/* Replace first whitespace char (at the end) with null terminator.
|
||||
* If there is none, we're just replacing the null terminator.
|
||||
*/
|
||||
*(end + 1) = '\0';
|
||||
|
||||
while (*str != '\0' && textparsers->IsWhitespace(str))
|
||||
{
|
||||
str++;
|
||||
len--;
|
||||
}
|
||||
|
||||
return str;
|
||||
}
|
||||
|
||||
|
@ -52,5 +52,6 @@ char *sm_strdup(const char *str);
|
||||
size_t CorePlayerTranslate(int client, char *buffer, size_t maxlength, const char *phrase, void **params);
|
||||
unsigned int UTIL_ReplaceAll(char *subject, size_t maxlength, const char *search, const char *replace);
|
||||
char *UTIL_ReplaceEx(char *subject, size_t maxLen, const char *search, size_t searchLen, const char *replace, size_t replaceLen);
|
||||
char *UTIL_TrimWhitespace(char *str, size_t &len);
|
||||
|
||||
#endif // _INCLUDE_SOURCEMOD_STRINGUTIL_H_
|
||||
|
@ -557,13 +557,6 @@ static cell_t sm_LogError(IPluginContext *pContext, const cell_t *params)
|
||||
return 1;
|
||||
}
|
||||
|
||||
enum
|
||||
{
|
||||
FileTime_LastAccess = 0, /* Last access (not available on FAT) */
|
||||
FileTime_Created = 1, /* Creation (not available on FAT) */
|
||||
FileTime_LastChange = 2, /* Last modification */
|
||||
};
|
||||
|
||||
static cell_t sm_GetFileTime(IPluginContext *pContext, const cell_t *params)
|
||||
{
|
||||
char *name;
|
||||
@ -574,30 +567,16 @@ static cell_t sm_GetFileTime(IPluginContext *pContext, const cell_t *params)
|
||||
return 0;
|
||||
}
|
||||
|
||||
time_t time_val;
|
||||
char realpath[PLATFORM_MAX_PATH];
|
||||
g_SourceMod.BuildPath(Path_Game, realpath, sizeof(realpath), "%s", name);
|
||||
|
||||
#ifdef PLATFORM_WINDOWS
|
||||
struct _stat s;
|
||||
if (_stat(realpath, &s) != 0)
|
||||
#elif defined PLATFORM_POSIX
|
||||
struct stat s;
|
||||
if (stat(realpath, &s) != 0)
|
||||
#endif
|
||||
if (!g_LibSys.FileTime(realpath, (FileTimeType)params[2], &time_val))
|
||||
{
|
||||
return -1;
|
||||
} else {
|
||||
if (params[2] == FileTime_LastAccess)
|
||||
{
|
||||
return (cell_t)s.st_atime;
|
||||
} else if (params[2] == FileTime_Created) {
|
||||
return (cell_t)s.st_ctime;
|
||||
} else if (params[2] == FileTime_LastChange) {
|
||||
return (cell_t)s.st_mtime;
|
||||
}
|
||||
}
|
||||
|
||||
return -1;
|
||||
return (cell_t)time_val;
|
||||
}
|
||||
|
||||
static cell_t sm_LogToOpenFile(IPluginContext *pContext, const cell_t *params)
|
||||
|
659
core/smn_maplists.cpp
Normal file
659
core/smn_maplists.cpp
Normal file
@ -0,0 +1,659 @@
|
||||
/**
|
||||
* vim: set ts=4 :
|
||||
* =============================================================================
|
||||
* SourceMod
|
||||
* Copyright (C) 2004-2007 AlliedModders LLC. All rights reserved.
|
||||
* =============================================================================
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it under
|
||||
* the terms of the GNU General Public License, version 3.0, as published by the
|
||||
* Free Software Foundation.
|
||||
*
|
||||
* 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. If not, see <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* As a special exception, AlliedModders LLC gives you permission to link the
|
||||
* code of this program (as well as its derivative works) to "Half-Life 2," the
|
||||
* "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software
|
||||
* by the Valve Corporation. You must obey the GNU General Public License in
|
||||
* all respects for all other code used. Additionally, AlliedModders LLC grants
|
||||
* this exception to all derivative works. AlliedModders LLC defines further
|
||||
* exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007),
|
||||
* or <http://www.sourcemod.net/license.php>.
|
||||
*
|
||||
* Version: $Id$
|
||||
*/
|
||||
|
||||
#include <sh_list.h>
|
||||
#include "sm_globals.h"
|
||||
#include "sm_trie_tpl.h"
|
||||
#include "CellArray.h"
|
||||
#include "convar.h"
|
||||
#include "sourcemm_api.h"
|
||||
#include "LibrarySys.h"
|
||||
#include "TextParsers.h"
|
||||
#include "sm_stringutil.h"
|
||||
#include "sourcemod.h"
|
||||
#include "Logger.h"
|
||||
#include "HandleSys.h"
|
||||
|
||||
using namespace SourceHook;
|
||||
|
||||
struct maplist_info_t
|
||||
{
|
||||
bool bIsCompat;
|
||||
bool bIsPath;
|
||||
char name[PLATFORM_MAX_PATH];
|
||||
char path[PLATFORM_MAX_PATH];
|
||||
time_t last_modified_time;
|
||||
CellArray *pArray;
|
||||
int serial;
|
||||
};
|
||||
|
||||
#define MAPLIST_FLAG_MAPSFOLDER (1<<0) /**< On failure, use all maps in the maps folder. */
|
||||
#define MAPLIST_FLAG_CLEARARRAY (1<<1) /**< If an input array is specified, clear it before adding. */
|
||||
#define MAPLIST_FLAG_NO_DEFAULT (1<<2) /**< Do not read "default" or "mapcyclefile" on failure. */
|
||||
|
||||
class MapLists : public SMGlobalClass, public ITextListener_SMC
|
||||
{
|
||||
public:
|
||||
enum MapListState
|
||||
{
|
||||
MPS_NONE,
|
||||
MPS_GLOBAL,
|
||||
MPS_MAPLIST,
|
||||
};
|
||||
public:
|
||||
MapLists()
|
||||
{
|
||||
m_pMapCycleFile = NULL;
|
||||
m_ConfigLastChanged = 0;
|
||||
m_nSerialChange = 0;
|
||||
}
|
||||
void OnSourceModAllInitialized()
|
||||
{
|
||||
m_pMapCycleFile = icvar->FindVar("mapcyclefile");
|
||||
g_SourceMod.BuildPath(Path_SM, m_ConfigFile, sizeof(m_ConfigFile), "configs/maplists.cfg");
|
||||
}
|
||||
void OnSourceModShutdown()
|
||||
{
|
||||
DumpCache(NULL);
|
||||
}
|
||||
void AddOrUpdateDefault(const char *name, const char *file)
|
||||
{
|
||||
char path[PLATFORM_MAX_PATH];
|
||||
maplist_info_t *pMapList, **ppMapList;
|
||||
|
||||
if ((ppMapList = m_ListLookup.retrieve(name)) == NULL)
|
||||
{
|
||||
pMapList = new maplist_info_t;
|
||||
pMapList->bIsCompat = true;
|
||||
pMapList->bIsPath = true;
|
||||
pMapList->last_modified_time = 0;
|
||||
strncopy(pMapList->name, name, sizeof(pMapList->name));
|
||||
pMapList->pArray = NULL;
|
||||
g_SourceMod.BuildPath(Path_Game,
|
||||
pMapList->path,
|
||||
sizeof(pMapList->path),
|
||||
"%s",
|
||||
file);
|
||||
pMapList->serial = 0;
|
||||
m_ListLookup.insert(name, pMapList);
|
||||
m_MapLists.push_back(pMapList);
|
||||
return;
|
||||
}
|
||||
|
||||
pMapList = *ppMapList;
|
||||
|
||||
/* Don't modify if it's from the config file */
|
||||
if (!pMapList->bIsCompat)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
g_SourceMod.BuildPath(Path_Game,
|
||||
path,
|
||||
sizeof(path),
|
||||
"%s",
|
||||
file);
|
||||
|
||||
/* If the path matches, don't reset the serial/time */
|
||||
if (strcmp(path, pMapList->path) == 0)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
strncopy(pMapList->path, path, sizeof(pMapList->path));
|
||||
pMapList->bIsPath = true;
|
||||
pMapList->last_modified_time = 0;
|
||||
pMapList->serial = 0;
|
||||
}
|
||||
void UpdateCache()
|
||||
{
|
||||
bool fileFound;
|
||||
SMCError error;
|
||||
time_t fileTime;
|
||||
SMCStates states = {0, 0};
|
||||
|
||||
fileFound = g_LibSys.FileTime(m_ConfigFile, FileTime_LastChange, &fileTime);
|
||||
|
||||
/* If the file is found and hasn't changed, bail out now. */
|
||||
if (fileFound && fileTime == m_ConfigLastChanged)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
/* If the file wasn't found, and we already have entries, we bail out too.
|
||||
* This case lets us optimize when a user deletes the config file, so we
|
||||
* don't reparse every single time the function is called.
|
||||
*/
|
||||
if (!fileFound && m_MapLists.size() > 0)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
/* Dump everything we know about. */
|
||||
List<maplist_info_t *> compat;
|
||||
DumpCache(&compat);
|
||||
|
||||
/* All this is to add the default entry back in. */
|
||||
maplist_info_t *pDefList = new maplist_info_t;
|
||||
|
||||
pDefList->bIsPath = true;
|
||||
strncopy(pDefList->name, "mapcyclefile", sizeof(pDefList->name));
|
||||
g_SourceMod.BuildPath(Path_Game,
|
||||
pDefList->path,
|
||||
sizeof(pDefList->path),
|
||||
"%s",
|
||||
m_pMapCycleFile ? m_pMapCycleFile->GetString() : "mapcycle.txt");
|
||||
pDefList->last_modified_time = 0;
|
||||
pDefList->pArray = NULL;
|
||||
pDefList->serial = 0;
|
||||
|
||||
m_ListLookup.insert("mapcyclefile", pDefList);
|
||||
m_MapLists.push_back(pDefList);
|
||||
|
||||
/* Now parse the config file even if we don't know about it.
|
||||
* This will give us a nice error message.
|
||||
*/
|
||||
if ((error = g_TextParser.ParseFile_SMC(m_ConfigFile, this, &states))
|
||||
!= SMCError_Okay)
|
||||
{
|
||||
const char *errmsg = g_TextParser.GetSMCErrorString(error);
|
||||
if (errmsg == NULL)
|
||||
{
|
||||
errmsg = "Unknown error";
|
||||
}
|
||||
g_Logger.LogError("[SM] Could not parse file \"%s\"", m_ConfigFile);
|
||||
g_Logger.LogError("[SM] Error on line %d (col %d): %s",
|
||||
states.line,
|
||||
states.col,
|
||||
errmsg);
|
||||
}
|
||||
else
|
||||
{
|
||||
m_ConfigLastChanged = fileTime;
|
||||
}
|
||||
|
||||
/* Now, re-add compat stuff back in if we can. */
|
||||
List<maplist_info_t *>::iterator iter = compat.begin();
|
||||
while (iter != compat.end())
|
||||
{
|
||||
if (m_ListLookup.retrieve((*iter)->name) != NULL)
|
||||
{
|
||||
/* The compatibility shim is no longer needed. */
|
||||
delete (*iter)->pArray;
|
||||
delete (*iter);
|
||||
}
|
||||
else
|
||||
{
|
||||
m_ListLookup.insert((*iter)->name, (*iter));
|
||||
m_MapLists.push_back((*iter));
|
||||
}
|
||||
iter = compat.erase(iter);
|
||||
}
|
||||
}
|
||||
void ReadSMC_ParseStart()
|
||||
{
|
||||
m_CurState = MPS_NONE;
|
||||
m_IgnoreLevel = 0;
|
||||
m_pCurMapList = NULL;
|
||||
}
|
||||
SMCResult ReadSMC_NewSection(const SMCStates *states, const char *name)
|
||||
{
|
||||
if (m_IgnoreLevel)
|
||||
{
|
||||
m_IgnoreLevel++;
|
||||
return SMCResult_Continue;
|
||||
}
|
||||
|
||||
if (m_CurState == MPS_NONE)
|
||||
{
|
||||
if (strcmp(name, "MapLists") == 0)
|
||||
{
|
||||
m_CurState = MPS_GLOBAL;
|
||||
}
|
||||
else
|
||||
{
|
||||
m_IgnoreLevel = 1;
|
||||
}
|
||||
}
|
||||
else if (m_CurState == MPS_GLOBAL)
|
||||
{
|
||||
m_pCurMapList = new maplist_info_t;
|
||||
|
||||
memset(m_pCurMapList, 0, sizeof(maplist_info_t));
|
||||
strncopy(m_pCurMapList->name, name, sizeof(m_pCurMapList->name));
|
||||
|
||||
m_CurState = MPS_MAPLIST;
|
||||
}
|
||||
else if (m_CurState == MPS_MAPLIST)
|
||||
{
|
||||
m_IgnoreLevel++;
|
||||
}
|
||||
|
||||
return SMCResult_Continue;
|
||||
}
|
||||
SMCResult ReadSMC_KeyValue(const SMCStates *states, const char *key, const char *value)
|
||||
{
|
||||
if (m_IgnoreLevel || m_pCurMapList == NULL)
|
||||
{
|
||||
return SMCResult_Continue;
|
||||
}
|
||||
|
||||
if (strcmp(key, "file") == 0)
|
||||
{
|
||||
g_SourceMod.BuildPath(Path_Game,
|
||||
m_pCurMapList->path,
|
||||
sizeof(m_pCurMapList->path),
|
||||
"%s",
|
||||
value);
|
||||
m_pCurMapList->bIsPath = true;
|
||||
}
|
||||
else if (strcmp(key, "target") == 0)
|
||||
{
|
||||
strncopy(m_pCurMapList->path, value, sizeof(m_pCurMapList->path));
|
||||
m_pCurMapList->bIsPath = false;
|
||||
}
|
||||
|
||||
return SMCResult_Continue;
|
||||
}
|
||||
SMCResult ReadSMC_LeavingSection(const SMCStates *states)
|
||||
{
|
||||
if (m_IgnoreLevel)
|
||||
{
|
||||
m_IgnoreLevel--;
|
||||
return SMCResult_Continue;
|
||||
}
|
||||
|
||||
if (m_CurState == MPS_MAPLIST)
|
||||
{
|
||||
if (m_pCurMapList != NULL
|
||||
&& m_pCurMapList->path[0] != '\0'
|
||||
&& m_ListLookup.retrieve(m_pCurMapList->name) == NULL)
|
||||
{
|
||||
m_ListLookup.insert(m_pCurMapList->name, m_pCurMapList);
|
||||
m_MapLists.push_back(m_pCurMapList);
|
||||
m_pCurMapList = NULL;
|
||||
}
|
||||
else
|
||||
{
|
||||
delete m_pCurMapList;
|
||||
m_pCurMapList = NULL;
|
||||
}
|
||||
m_CurState = MPS_GLOBAL;
|
||||
}
|
||||
else if (m_CurState == MPS_GLOBAL)
|
||||
{
|
||||
m_CurState = MPS_NONE;
|
||||
}
|
||||
|
||||
return SMCResult_Continue;
|
||||
}
|
||||
void ReadSMC_ParseEnd(bool halted, bool failed)
|
||||
{
|
||||
delete m_pCurMapList;
|
||||
m_pCurMapList = NULL;
|
||||
}
|
||||
static int sort_maps_in_adt_array(const void *str1, const void *str2)
|
||||
{
|
||||
return strcmp((char *)str1, (char *)str2);
|
||||
}
|
||||
CellArray *UpdateMapList(CellArray *pUseArray, const char *name, int *pSerial, unsigned int flags)
|
||||
{
|
||||
int change_serial;
|
||||
CellArray *pNewArray;
|
||||
bool success, free_new_array;
|
||||
|
||||
free_new_array = false;
|
||||
|
||||
if ((success = GetMapList(&pNewArray, name, &change_serial)) == false)
|
||||
{
|
||||
if ((flags & MAPLIST_FLAG_NO_DEFAULT) != MAPLIST_FLAG_NO_DEFAULT)
|
||||
{
|
||||
/* If this list failed, and it's not the default, try the default.
|
||||
*/
|
||||
if (strcmp(name, "default") != 0)
|
||||
{
|
||||
success = GetMapList(&pNewArray, name, &change_serial);
|
||||
}
|
||||
/* If either of the last two conditions failed, try again if we can. */
|
||||
if (!success)
|
||||
{
|
||||
success = GetMapList(&pNewArray, "mapcyclefile", &change_serial);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* If there was a success, and the serial has not changed, bail out. */
|
||||
if (*pSerial == change_serial)
|
||||
{
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/**
|
||||
* If there was a success but no map list, we need to look in the maps folder.
|
||||
* If there was a failure and the flag is specified, we need to look in the maps folder.
|
||||
*/
|
||||
if ((success && pNewArray == NULL)
|
||||
|| (!success && ((flags & MAPLIST_FLAG_MAPSFOLDER) == MAPLIST_FLAG_MAPSFOLDER)))
|
||||
{
|
||||
char path[255];
|
||||
IDirectory *pDir;
|
||||
|
||||
pNewArray = new CellArray(64);
|
||||
free_new_array = true;
|
||||
g_SourceMod.BuildPath(Path_Game, path, sizeof(path), "maps");
|
||||
|
||||
if ((pDir = g_LibSys.OpenDirectory(path)) != NULL)
|
||||
{
|
||||
char *ptr;
|
||||
cell_t *blk;
|
||||
char buffer[PLATFORM_MAX_PATH];
|
||||
|
||||
while (pDir->MoreFiles())
|
||||
{
|
||||
if (!pDir->IsEntryFile()
|
||||
|| strcmp(pDir->GetEntryName(), ".") == 0
|
||||
|| strcmp(pDir->GetEntryName(), "..") == 0)
|
||||
{
|
||||
pDir->NextEntry();
|
||||
continue;
|
||||
}
|
||||
strncopy(buffer, pDir->GetEntryName(), sizeof(buffer));
|
||||
if ((ptr = strstr(buffer, ".bsp")) == NULL)
|
||||
{
|
||||
pDir->NextEntry();
|
||||
continue;
|
||||
}
|
||||
*ptr = '\0';
|
||||
if (!engine->IsMapValid(buffer))
|
||||
{
|
||||
pDir->NextEntry();
|
||||
continue;
|
||||
}
|
||||
if ((blk = pNewArray->push()) == NULL)
|
||||
{
|
||||
pDir->NextEntry();
|
||||
continue;
|
||||
}
|
||||
strncopy((char *)blk, buffer, 255);
|
||||
pDir->NextEntry();
|
||||
}
|
||||
g_LibSys.CloseDirectory(pDir);
|
||||
}
|
||||
|
||||
/* Remove the array if there were no items. */
|
||||
if (pNewArray->size() == 0)
|
||||
{
|
||||
delete pNewArray;
|
||||
pNewArray = NULL;
|
||||
}
|
||||
else
|
||||
{
|
||||
qsort(pNewArray->base(),
|
||||
pNewArray->size(),
|
||||
pNewArray->blocksize() * sizeof(cell_t),
|
||||
sort_maps_in_adt_array);
|
||||
}
|
||||
|
||||
change_serial = -1;
|
||||
}
|
||||
|
||||
/* If there is still no array by this point, bail out. */
|
||||
if (pNewArray == NULL)
|
||||
{
|
||||
*pSerial = -1;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
*pSerial = change_serial;
|
||||
|
||||
/* If there is no input array, return something temporary. */
|
||||
if (pUseArray == NULL)
|
||||
{
|
||||
if (free_new_array)
|
||||
{
|
||||
return pNewArray;
|
||||
}
|
||||
else
|
||||
{
|
||||
return pNewArray->clone();
|
||||
}
|
||||
}
|
||||
|
||||
/* Clear the input array if necessary. */
|
||||
if ((flags & MAPLIST_FLAG_CLEARARRAY) == MAPLIST_FLAG_CLEARARRAY)
|
||||
{
|
||||
pUseArray->clear();
|
||||
}
|
||||
|
||||
/* Copy. */
|
||||
cell_t *blk_dst;
|
||||
cell_t *blk_src;
|
||||
for (size_t i = 0; i < pNewArray->size(); i++)
|
||||
{
|
||||
blk_dst = pUseArray->push();
|
||||
blk_src = pNewArray->at(i);
|
||||
strncopy((char *)blk_dst, (char *)blk_src, pUseArray->blocksize() * sizeof(cell_t));
|
||||
}
|
||||
|
||||
/* Free resources if necessary. */
|
||||
if (free_new_array)
|
||||
{
|
||||
delete pNewArray;
|
||||
}
|
||||
|
||||
/* Return the array we were given. */
|
||||
return pUseArray;
|
||||
}
|
||||
private:
|
||||
bool GetMapList(CellArray **ppArray, const char *name, int *pSerial)
|
||||
{
|
||||
time_t last_time;
|
||||
maplist_info_t *pMapList, **ppMapList;
|
||||
|
||||
if ((ppMapList = m_ListLookup.retrieve(name)) == NULL)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
pMapList = *ppMapList;
|
||||
|
||||
if (!pMapList->bIsPath)
|
||||
{
|
||||
return GetMapList(ppArray, pMapList->path, pSerial);
|
||||
}
|
||||
|
||||
/* If it is a path, and the path is "*", assume all files must be used. */
|
||||
if (strcmp(pMapList->path, "*") == 0)
|
||||
{
|
||||
*ppArray = NULL;
|
||||
return true;
|
||||
}
|
||||
|
||||
if (m_pMapCycleFile != NULL && strcmp(name, "mapcyclefile") == 0)
|
||||
{
|
||||
if (strcmp(m_pMapCycleFile->GetString(), pMapList->path) != 0)
|
||||
{
|
||||
strncopy(pMapList->path, m_pMapCycleFile->GetString(), sizeof(pMapList->path));
|
||||
pMapList->last_modified_time = 0;
|
||||
}
|
||||
}
|
||||
|
||||
if (!g_LibSys.FileTime(pMapList->path, FileTime_LastChange, &last_time)
|
||||
|| last_time > pMapList->last_modified_time)
|
||||
{
|
||||
/* Reparse */
|
||||
FILE *fp;
|
||||
cell_t *blk;
|
||||
char buffer[255];
|
||||
|
||||
if ((fp = fopen(pMapList->path, "rt")) == NULL)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
delete pMapList->pArray;
|
||||
pMapList->pArray = new CellArray(64);
|
||||
|
||||
while (!feof(fp) && fgets(buffer, sizeof(buffer), fp) != NULL)
|
||||
{
|
||||
size_t len = strlen(buffer);
|
||||
char *ptr = UTIL_TrimWhitespace(buffer, len);
|
||||
if (*ptr == '\0'
|
||||
|| *ptr == ';'
|
||||
|| strncmp(ptr, "//", 2) == 0)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
if (!engine->IsMapValid(ptr))
|
||||
{
|
||||
continue;
|
||||
}
|
||||
if ((blk = pMapList->pArray->push()) != NULL)
|
||||
{
|
||||
strncopy((char *)blk, ptr, 255);
|
||||
}
|
||||
}
|
||||
|
||||
fclose(fp);
|
||||
|
||||
pMapList->last_modified_time = last_time;
|
||||
pMapList->serial = ++m_nSerialChange;
|
||||
}
|
||||
|
||||
if (pMapList->pArray == NULL || pMapList->pArray->size() == 0)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
*pSerial = pMapList->serial;
|
||||
*ppArray = pMapList->pArray;
|
||||
|
||||
return true;
|
||||
}
|
||||
void DumpCache(List<maplist_info_t *> *compat_list)
|
||||
{
|
||||
m_ListLookup.clear();
|
||||
|
||||
List<maplist_info_t *>::iterator iter = m_MapLists.begin();
|
||||
while (iter != m_MapLists.end())
|
||||
{
|
||||
if (compat_list != NULL && (*iter)->bIsCompat)
|
||||
{
|
||||
compat_list->push_back((*iter));
|
||||
}
|
||||
else
|
||||
{
|
||||
delete (*iter)->pArray;
|
||||
delete (*iter);
|
||||
}
|
||||
iter = m_MapLists.erase(iter);
|
||||
}
|
||||
}
|
||||
private:
|
||||
char m_ConfigFile[PLATFORM_MAX_PATH];
|
||||
time_t m_ConfigLastChanged;
|
||||
ConVar *m_pMapCycleFile;
|
||||
KTrie<maplist_info_t *> m_ListLookup;
|
||||
List<maplist_info_t *> m_MapLists;
|
||||
MapListState m_CurState;
|
||||
unsigned int m_IgnoreLevel;
|
||||
maplist_info_t *m_pCurMapList;
|
||||
int m_nSerialChange;
|
||||
} s_MapLists;
|
||||
|
||||
static cell_t LoadMapList(IPluginContext *pContext, const cell_t *params)
|
||||
{
|
||||
char *str;
|
||||
Handle_t hndl;
|
||||
cell_t *addr, flags;
|
||||
CellArray *pArray, *pNewArray;
|
||||
|
||||
hndl = params[1];
|
||||
pContext->LocalToPhysAddr(params[2], &addr);
|
||||
pContext->LocalToString(params[3], &str);
|
||||
flags = params[4];
|
||||
|
||||
/* Make sure the input Handle is valid */
|
||||
pArray = NULL;
|
||||
if (hndl != BAD_HANDLE)
|
||||
{
|
||||
HandleError err;
|
||||
HandleSecurity sec(pContext->GetIdentity(), g_pCoreIdent);
|
||||
|
||||
if ((err = g_HandleSys.ReadHandle(hndl, htCellArray, &sec, (void **)&pArray))
|
||||
!= HandleError_None)
|
||||
{
|
||||
return pContext->ThrowNativeError("Invalid Handle %x (error %d)", hndl, err);
|
||||
}
|
||||
}
|
||||
|
||||
/* Make sure the map list cache is up to date at the root */
|
||||
s_MapLists.UpdateCache();
|
||||
|
||||
/* Try to get the map list. */
|
||||
if ((pNewArray = s_MapLists.UpdateMapList(pArray, str, addr, flags)) == NULL)
|
||||
{
|
||||
return BAD_HANDLE;
|
||||
}
|
||||
|
||||
/* If the user wanted a new array, create it now. */
|
||||
if (hndl == BAD_HANDLE)
|
||||
{
|
||||
if ((hndl = g_HandleSys.CreateHandle(htCellArray, pNewArray, pContext->GetIdentity(), g_pCoreIdent, NULL))
|
||||
== BAD_HANDLE)
|
||||
{
|
||||
*addr = -1;
|
||||
delete pNewArray;
|
||||
return BAD_HANDLE;
|
||||
}
|
||||
}
|
||||
|
||||
return hndl;
|
||||
}
|
||||
|
||||
static cell_t SetMapListCompatBind(IPluginContext *pContext, const cell_t *params)
|
||||
{
|
||||
char *name, *file;
|
||||
|
||||
pContext->LocalToString(params[1], &name);
|
||||
pContext->LocalToString(params[2], &file);
|
||||
|
||||
s_MapLists.AddOrUpdateDefault(name, file);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
REGISTER_NATIVES(mapListNatives)
|
||||
{
|
||||
{"ReadMapList", LoadMapList},
|
||||
{"SetMapListCompatBind", SetMapListCompatBind},
|
||||
{NULL, NULL},
|
||||
};
|
@ -29,6 +29,7 @@
|
||||
* Version: $Id$
|
||||
*/
|
||||
|
||||
#include <sys/stat.h>
|
||||
#include <stdio.h>
|
||||
#include <stdarg.h>
|
||||
#include <string.h>
|
||||
@ -396,3 +397,32 @@ size_t LibrarySystem::GetFileFromPath(char *buffer, size_t maxlength, const char
|
||||
/* We scanned and found no path separator */
|
||||
return UTIL_Format(buffer, maxlength, "%s", path);
|
||||
}
|
||||
|
||||
bool LibrarySystem::FileTime(const char *path, FileTimeType type, time_t *pTime)
|
||||
{
|
||||
#ifdef PLATFORM_WINDOWS
|
||||
struct _stat s;
|
||||
if (_stat(path, &s) != 0)
|
||||
#elif defined PLATFORM_POSIX
|
||||
struct stat s;
|
||||
if (stat(path, &s) != 0)
|
||||
#endif
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
if (type == FileTime_LastAccess)
|
||||
{
|
||||
*pTime = s.st_atime;
|
||||
}
|
||||
else if (type == FileTime_Created)
|
||||
{
|
||||
*pTime = s.st_ctime;
|
||||
}
|
||||
else if (type == FileTime_LastChange)
|
||||
{
|
||||
*pTime = s.st_mtime;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
@ -94,6 +94,7 @@ public:
|
||||
const char *GetFileExtension(const char *filename);
|
||||
bool CreateFolder(const char *path);
|
||||
size_t GetFileFromPath(char *buffer, size_t maxlength, const char *path);
|
||||
bool FileTime(const char *path, FileTimeType type, time_t *pTime);
|
||||
};
|
||||
|
||||
extern LibrarySystem g_LibSys;
|
||||
|
@ -49,7 +49,6 @@ public Plugin:myinfo =
|
||||
new Handle:hTopMenu = INVALID_HANDLE;
|
||||
|
||||
new Handle:g_MapList;
|
||||
new g_mapFileTime;
|
||||
|
||||
#include "basecommands/kick.sp"
|
||||
#include "basecommands/reloadadmins.sp"
|
||||
@ -82,6 +81,10 @@ public OnPluginStart()
|
||||
g_MapList = CreateMenu(MenuHandler_ChangeMap);
|
||||
SetMenuTitle(g_MapList, "Please select a map");
|
||||
SetMenuExitBackButton(g_MapList, true);
|
||||
|
||||
decl String:mapListPath[PLATFORM_MAX_PATH];
|
||||
BuildPath(Path_SM, mapListPath, sizeof(mapListPath), "configs/adminmenu_maplist.ini");
|
||||
SetMapListCompatBind("admin menu", mapListPath);
|
||||
}
|
||||
|
||||
public OnMapStart()
|
||||
|
@ -82,109 +82,37 @@ public Action:Timer_ChangeMap(Handle:timer, Handle:dp)
|
||||
return Plugin_Stop;
|
||||
}
|
||||
|
||||
new Handle:g_map_array = INVALID_HANDLE;
|
||||
new g_map_serial = -1;
|
||||
|
||||
LoadMapList(Handle:menu)
|
||||
{
|
||||
decl String:mapPath[256];
|
||||
BuildPath(Path_SM, mapPath, sizeof(mapPath), "configs/adminmenu_maplist.ini");
|
||||
new Handle:map_array;
|
||||
|
||||
if (!FileExists(mapPath))
|
||||
{
|
||||
if (g_MapList != INVALID_HANDLE)
|
||||
{
|
||||
RemoveAllMenuItems(menu);
|
||||
}
|
||||
|
||||
return LoadMapFolder(menu);
|
||||
}
|
||||
|
||||
// If the file hasn't changed, there's no reason to reload
|
||||
// all of the maps.
|
||||
new fileTime = GetFileTime(mapPath, FileTime_LastChange);
|
||||
if (g_mapFileTime == fileTime)
|
||||
if ((map_array = ReadMapList(g_map_array,
|
||||
g_map_serial,
|
||||
"admin menu",
|
||||
MAPLIST_FLAG_CLEARARRAY|MAPLIST_FLAG_NO_DEFAULT|MAPLIST_FLAG_MAPSFOLDER))
|
||||
!= INVALID_HANDLE)
|
||||
{
|
||||
return GetMenuItemCount(menu);
|
||||
g_map_array = map_array;
|
||||
}
|
||||
|
||||
g_mapFileTime = fileTime;
|
||||
|
||||
// Reset the array
|
||||
if (g_MapList != INVALID_HANDLE)
|
||||
if (g_map_array == INVALID_HANDLE)
|
||||
{
|
||||
RemoveAllMenuItems(menu);
|
||||
}
|
||||
|
||||
new Handle:file = OpenFile(mapPath, "rt");
|
||||
if (file == INVALID_HANDLE)
|
||||
{
|
||||
LogError("[SM] Could not open file: %s, reverting to map folder", mapPath);
|
||||
return LoadMapFolder(menu);
|
||||
}
|
||||
|
||||
decl String:buffer[256], len;
|
||||
while (!IsEndOfFile(file) && ReadFileLine(file, buffer, sizeof(buffer)))
|
||||
{
|
||||
TrimString(buffer);
|
||||
|
||||
if ((len = StrContains(buffer, ".bsp", false)) != -1)
|
||||
{
|
||||
buffer[len] = '\0';
|
||||
}
|
||||
|
||||
if (buffer[0] == '\0'
|
||||
|| buffer[0] == ';'
|
||||
|| buffer[0] == '/'
|
||||
|| !IsValidConVarChar(buffer[0]))
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
if (IsMapValid(buffer))
|
||||
{
|
||||
AddMenuItem(menu, buffer, buffer);
|
||||
}
|
||||
}
|
||||
|
||||
CloseHandle(file);
|
||||
|
||||
new count = GetMenuItemCount(menu);
|
||||
|
||||
if (!count)
|
||||
return LoadMapFolder(menu);
|
||||
else
|
||||
return count;
|
||||
}
|
||||
|
||||
LoadMapFolder(Handle:menu)
|
||||
{
|
||||
LogMessage("[SM] Loading menu map list from maps folder");
|
||||
|
||||
new Handle:mapDir = OpenDirectory("maps/");
|
||||
|
||||
if (mapDir == INVALID_HANDLE)
|
||||
{
|
||||
LogError("[SM] Could not open map directory for reading");
|
||||
return 0;
|
||||
}
|
||||
|
||||
new String:mapName[64];
|
||||
new String:buffer[64];
|
||||
new FileType:fileType;
|
||||
new len;
|
||||
RemoveAllMenuItems(menu);
|
||||
|
||||
while(ReadDirEntry(mapDir, mapName, sizeof(mapName), fileType))
|
||||
decl String:map_name[64];
|
||||
new map_count = GetArraySize(g_map_array);
|
||||
|
||||
for (new i = 0; i < map_count; i++)
|
||||
{
|
||||
if(fileType == FileType_File)
|
||||
{
|
||||
len = strlen(mapName);
|
||||
|
||||
if(SplitString(mapName, ".bsp", buffer, sizeof(buffer)) == len)
|
||||
{
|
||||
AddMenuItem(menu, buffer, buffer);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
CloseHandle(mapDir);
|
||||
|
||||
return GetMenuItemCount(menu);
|
||||
GetArrayString(g_map_array, i, map_name, sizeof(map_name));
|
||||
AddMenuItem(menu, map_name, map_name);
|
||||
}
|
||||
|
||||
return map_count;
|
||||
}
|
||||
|
@ -48,8 +48,8 @@ enum SortOrder
|
||||
/**
|
||||
* Data types for ADT Array Sorts
|
||||
*/
|
||||
enum SortType
|
||||
{
|
||||
enum SortType
|
||||
{
|
||||
Sort_Integer = 0,
|
||||
Sort_Float,
|
||||
Sort_String,
|
||||
@ -147,7 +147,7 @@ native SortCustom2D(array[][], array_size, SortFunc2D:sortfunc, Handle:hndl=INVA
|
||||
* @param type Data type stored in the ADT Array
|
||||
* @noreturn
|
||||
*/
|
||||
native SortADTArray(Handle:array, SortOrder:order = Sort_Ascending, SortType:type = Sort_Integer);
|
||||
native SortADTArray(Handle:array, SortOrder:order, SortType:type);
|
||||
|
||||
/**
|
||||
* Sort comparison function for ADT Array elements. Function provides you with
|
||||
|
@ -490,6 +490,70 @@ forward OnLibraryAdded(const String:name[]);
|
||||
*/
|
||||
forward OnLibraryRemoved(const String:name[]);
|
||||
|
||||
#define MAPLIST_FLAG_MAPSFOLDER (1<<0) /**< On failure, use all maps in the maps folder. */
|
||||
#define MAPLIST_FLAG_CLEARARRAY (1<<1) /**< If an input array is specified, clear it before adding. */
|
||||
#define MAPLIST_FLAG_NO_DEFAULT (1<<2) /**< Do not read "default" or "mapcyclefile" on failure. */
|
||||
|
||||
/**
|
||||
* Loads a map list to an ADT Array.
|
||||
*
|
||||
* A map list is a list of maps from a file. SourceMod allows easy configuration of
|
||||
* maplists through addons/sourcemod/configs/maplists.cfg. Each entry is given a
|
||||
* name and a file (for example, "rtv" => "rtv.cfg"), or a name and a redirection
|
||||
* (for example, "rtv" => "default"). This native will read a map list entry,
|
||||
* cache the file, and return the list of maps it holds.
|
||||
*
|
||||
* Serial change numbers are used to identify if a map list has changed. Thus, if
|
||||
* you pass a serial change number and it's equal to what SourceMod currently knows
|
||||
* about the map list, then SourceMod won't reparse the file.
|
||||
*
|
||||
* If the maps end up being read from the maps folder (MAPLIST_FLAG_MAPSFOLDER), they
|
||||
* are automatically sorted in alphabetical, ascending order.
|
||||
*
|
||||
* @param array Array to store the map list. If INVALID_HANDLE, a new blank
|
||||
* array will be created. The blocksize should be at least 16;
|
||||
* otherwise results may be truncated. Items are added to the array
|
||||
* as strings. The array is never checked for duplicates, and it is
|
||||
* not read beforehand. Only the serial number is used to detect
|
||||
* changes.
|
||||
* @param str Config name, or "default" for the default map list. Config names
|
||||
* should be somewhat descriptive. For example, the admin menu uses
|
||||
* a config name of "admin menu". The list names can be configured
|
||||
* by users in addons/sourcemod/configs/maplists.cfg.
|
||||
* @param serial Serial number to identify last known map list change. If -1, the
|
||||
* the value will not be checked. If the map list has since changed,
|
||||
* the serial is updated (even if -1 was passed). If there is an error
|
||||
* finding a valid maplist, then the serial is set to -1.
|
||||
* @param flags MAPLIST_FLAG flags.
|
||||
* @return On failure:
|
||||
* INVALID_HANDLE is returned, the serial is set to -1, and the input
|
||||
* array (if any) is left unchanged.
|
||||
* On no change:
|
||||
INVALID_HANDLE is returned, the serial is unchanged, and the input
|
||||
array (if any) is left unchanged.
|
||||
* On success:
|
||||
* A valid array Handle is returned, containing at least one map string.
|
||||
* If an array was passed, the return value is equal to the passed Array
|
||||
* Handle. If the passed array was not cleared, it will have grown by at
|
||||
* least one item. The serial number is updated to a positive number.
|
||||
* @error Invalid array Handle that is not INVALID_HANDLE.
|
||||
*/
|
||||
native Handle:ReadMapList(Handle:array=INVALID_HANDLE,
|
||||
&serial=-1,
|
||||
const String:str[]="default",
|
||||
flags=MAPLIST_FLAG_CLEARARRAY);
|
||||
|
||||
/**
|
||||
* Makes a compatibility binding for map lists. For example, if a function previously used
|
||||
* "clam.cfg" for map lists, this function will insert a "fake" binding to "clam.cfg" that
|
||||
* will be overridden if it's in the maplists.cfg file.
|
||||
*
|
||||
* @param name Configuration name that would be used with ReadMapList().
|
||||
* @param file Default file to use.
|
||||
* @noreturn
|
||||
*/
|
||||
native SetMapListCompatBind(const String:name[], const String:file[]);
|
||||
|
||||
#include <helpers>
|
||||
#include <entity>
|
||||
|
||||
|
@ -42,7 +42,14 @@
|
||||
namespace SourceMod
|
||||
{
|
||||
#define SMINTERFACE_LIBRARYSYS_NAME "ILibrarySys"
|
||||
#define SMINTERFACE_LIBRARYSYS_VERSION 3
|
||||
#define SMINTERFACE_LIBRARYSYS_VERSION 4
|
||||
|
||||
enum FileTimeType
|
||||
{
|
||||
FileTime_LastAccess = 0, /* Last access (not available on FAT) */
|
||||
FileTime_Created = 1, /* Creation (not available on FAT) */
|
||||
FileTime_LastChange = 2, /* Last modification */
|
||||
};
|
||||
|
||||
class ILibrary
|
||||
{
|
||||
@ -202,6 +209,19 @@ namespace SourceMod
|
||||
* @return True on success, false otherwise.
|
||||
*/
|
||||
virtual bool CreateFolder(const char *path) =0;
|
||||
|
||||
/**
|
||||
* @brief Returns the requested timestamp of a file.
|
||||
*
|
||||
* NOTE: On FAT file systems, the access and creation times
|
||||
* may not be valid.
|
||||
*
|
||||
* @param path Path to file.
|
||||
* @param type FileTimeType of time value to request.
|
||||
* @param pTime Pointer to store time.
|
||||
* @return True on success, false on failure.
|
||||
*/
|
||||
virtual bool FileTime(const char *path, FileTimeType type, time_t *pTime) =0;
|
||||
};
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user