You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
492 lines
12 KiB
C++
492 lines
12 KiB
C++
/*
|
|
* All or portions of this file Copyright (c) Amazon.com, Inc. or its affiliates or
|
|
* its licensors.
|
|
*
|
|
* For complete copyright and license terms please see the LICENSE at the root of this
|
|
* distribution (the "License"). All use of this software is governed by the License,
|
|
* or, if provided, by the license below or the license accompanying this file. Do not
|
|
* remove or modify any license notices. This file is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
*
|
|
*/
|
|
// Original file Copyright Crytek GMBH or its affiliates, used under license.
|
|
|
|
#ifndef CRYINCLUDE_CRYCOMMON_SETTINGSMANAGERHELPERS_H
|
|
#define CRYINCLUDE_CRYCOMMON_SETTINGSMANAGERHELPERS_H
|
|
#pragma once
|
|
|
|
#include "ProjectDefines.h"
|
|
#include <string.h> // memcpy
|
|
#include <algorithm> // std::min
|
|
|
|
|
|
namespace SettingsManagerHelpers
|
|
{
|
|
namespace Utils
|
|
{
|
|
inline size_t strlen(const char* p)
|
|
{
|
|
return p ? ::strlen(p) : 0;
|
|
}
|
|
|
|
inline size_t strcmp(const char* p0, const char* p1)
|
|
{
|
|
return ::strcmp(p0, p1);
|
|
}
|
|
|
|
|
|
inline size_t strlen(const wchar_t* p)
|
|
{
|
|
return p ? ::wcslen(p) : 0;
|
|
}
|
|
|
|
inline size_t strcmp(const wchar_t* p0, const wchar_t* p1)
|
|
{
|
|
return ::wcscmp(p0, p1);
|
|
}
|
|
}
|
|
|
|
// The function copies characters from src to dst one by one until any of
|
|
// the following conditions is met:
|
|
// 1) the end of the destination buffer (minus one character) is reached
|
|
// 2) the end of the source buffer is reached
|
|
// 3) zero character is found in the source buffer
|
|
//
|
|
// When any of 1), 2), 3) happens, the function writes the terminating zero
|
|
// character to the destination buffer and return.
|
|
//
|
|
// The function guarantees writing the terminating zero character to the
|
|
// destination buffer (if the buffer can fit at least one character).
|
|
//
|
|
// The function returns false when a null pointer is passed or when
|
|
// clamping happened (i.e. when the end of the destination buffer is
|
|
// reached but the source has some characters left).
|
|
inline bool strcpy_with_clamp(char* const dst, size_t const dst_size_in_bytes, const char* const src, size_t const src_size_in_bytes = (size_t)-1)
|
|
{
|
|
if (!dst || dst_size_in_bytes < sizeof(char))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (!src || src_size_in_bytes < sizeof(char))
|
|
{
|
|
dst[0] = 0;
|
|
return src != 0; // we return true for non-null src without characters
|
|
}
|
|
|
|
const size_t src_n = src_size_in_bytes;
|
|
const size_t n = (std::min)(dst_size_in_bytes - 1, src_n);
|
|
|
|
for (size_t i = 0; i < n; ++i)
|
|
{
|
|
dst[i] = src[i];
|
|
if (!src[i])
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
dst[n] = 0;
|
|
return n >= src_n || src[n] == 0;
|
|
}
|
|
|
|
|
|
template<class T>
|
|
class CBuffer
|
|
{
|
|
public:
|
|
typedef T element_type;
|
|
|
|
private:
|
|
element_type* m_ptr;
|
|
size_t m_sizeInBytes;
|
|
|
|
public:
|
|
CBuffer(element_type* ptr, size_t sizeInBytes)
|
|
: m_ptr(ptr)
|
|
, m_sizeInBytes(ptr ? sizeInBytes : 0)
|
|
{
|
|
}
|
|
|
|
const element_type* getPtr() const
|
|
{
|
|
return m_ptr;
|
|
}
|
|
|
|
element_type* getPtr()
|
|
{
|
|
return m_ptr;
|
|
}
|
|
|
|
size_t getSizeInElements() const
|
|
{
|
|
return m_sizeInBytes / sizeof(element_type);
|
|
}
|
|
|
|
size_t getSizeInBytes() const
|
|
{
|
|
return m_sizeInBytes;
|
|
}
|
|
|
|
const element_type& operator[](size_t pos) const
|
|
{
|
|
return m_ptr[pos];
|
|
}
|
|
|
|
element_type& operator[](size_t pos)
|
|
{
|
|
return m_ptr[pos];
|
|
}
|
|
};
|
|
|
|
typedef CBuffer<char> CCharBuffer;
|
|
typedef CBuffer<wchar_t> CWCharBuffer;
|
|
|
|
|
|
template<class T, size_t CAPACITY>
|
|
class CFixedString
|
|
{
|
|
public:
|
|
typedef T char_type;
|
|
static const size_t npos = ~size_t(0);
|
|
private:
|
|
size_t m_count; // # chars (not counting trailing zero)
|
|
char_type m_buffer[CAPACITY + 1]; // '+ 1' is for trailing zero
|
|
|
|
public:
|
|
CFixedString()
|
|
{
|
|
clear();
|
|
}
|
|
|
|
CFixedString(const char_type* p)
|
|
{
|
|
set(p);
|
|
}
|
|
|
|
CFixedString& operator=(const char_type* p)
|
|
{
|
|
set(p);
|
|
return *this;
|
|
}
|
|
|
|
CFixedString& operator=(const CFixedString& s)
|
|
{
|
|
if (&s != this)
|
|
{
|
|
set(s.m_buffer, s.m_count);
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
CBuffer<char_type> getBuffer()
|
|
{
|
|
return CBuffer<char_type>(m_buffer, (CAPACITY + 1) * sizeof(char_type));
|
|
}
|
|
|
|
char_type operator[](const size_t i) const
|
|
{
|
|
return m_buffer[i];
|
|
}
|
|
|
|
const char_type* c_str() const
|
|
{
|
|
return &m_buffer[0];
|
|
}
|
|
|
|
const size_t length() const
|
|
{
|
|
return m_count;
|
|
}
|
|
|
|
void clear()
|
|
{
|
|
m_count = 0;
|
|
m_buffer[m_count] = 0;
|
|
}
|
|
|
|
void setLength(size_t n)
|
|
{
|
|
m_count = (n <= CAPACITY) ? n : CAPACITY;
|
|
m_buffer[m_count] = 0;
|
|
}
|
|
|
|
char_type* ptr()
|
|
{
|
|
return &m_buffer[0];
|
|
}
|
|
|
|
CFixedString substr(size_t pos = 0, size_t n = npos) const
|
|
{
|
|
CFixedString s;
|
|
if (pos < m_count && n > 0)
|
|
{
|
|
if (n > m_count || pos + n > m_count)
|
|
{
|
|
n = m_count - pos;
|
|
}
|
|
s.set(&m_buffer[pos], n);
|
|
}
|
|
return s;
|
|
}
|
|
|
|
void set(const char_type* p, size_t n)
|
|
{
|
|
if (p == 0 || n <= 0)
|
|
{
|
|
m_count = 0;
|
|
}
|
|
else
|
|
{
|
|
m_count = (n > CAPACITY) ? CAPACITY : n;
|
|
// memmove() is used because p may point to m_buffer
|
|
memmove(m_buffer, p, m_count * sizeof(*p));
|
|
}
|
|
m_buffer[m_count] = 0;
|
|
}
|
|
|
|
void set(const char_type* p)
|
|
{
|
|
if (p && p[0])
|
|
{
|
|
set(p, Utils::strlen(p));
|
|
}
|
|
else
|
|
{
|
|
clear();
|
|
}
|
|
}
|
|
|
|
void append(const char_type* p, size_t n)
|
|
{
|
|
if (p && n > 0)
|
|
{
|
|
if (n > CAPACITY || m_count + n > CAPACITY)
|
|
{
|
|
// assert(0);
|
|
n = CAPACITY - m_count;
|
|
}
|
|
if (n > 0)
|
|
{
|
|
memcpy(&m_buffer[m_count], p, n * sizeof(*p));
|
|
m_count += n;
|
|
m_buffer[m_count] = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
void append(const char_type* p)
|
|
{
|
|
if (p && p[0])
|
|
{
|
|
append(p, Utils::strlen(p));
|
|
}
|
|
}
|
|
|
|
void appendAscii(const char* p, size_t n)
|
|
{
|
|
if (p && n > 0)
|
|
{
|
|
if (n > CAPACITY || m_count + n > CAPACITY)
|
|
{
|
|
// assert(0);
|
|
n = CAPACITY - m_count;
|
|
}
|
|
if (n > 0)
|
|
{
|
|
for (size_t i = 0; i < n; ++i)
|
|
{
|
|
m_buffer[m_count + i] = p[i];
|
|
}
|
|
m_count += n;
|
|
m_buffer[m_count] = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
void appendAscii(const char* p)
|
|
{
|
|
if (p && p[0])
|
|
{
|
|
appendAscii(p, Utils::strlen(p));
|
|
}
|
|
}
|
|
|
|
bool equals(const char_type* p) const
|
|
{
|
|
return (p == 0 || p[0] == 0)
|
|
? (m_count == 0)
|
|
: (Utils::strcmp(m_buffer, p) == 0);
|
|
}
|
|
|
|
void trim()
|
|
{
|
|
size_t begin = 0;
|
|
while (begin < m_count && (m_buffer[begin] == ' ' || m_buffer[begin] == '\r' || m_buffer[begin] == '\t' || m_buffer[begin] == '\n'))
|
|
{
|
|
++begin;
|
|
}
|
|
|
|
if (begin >= m_count)
|
|
{
|
|
clear();
|
|
return;
|
|
}
|
|
|
|
size_t end = m_count - 1;
|
|
while (end > begin && (m_buffer[begin] == ' ' || m_buffer[begin] == '\r' || m_buffer[begin] == '\t' || m_buffer[begin] == '\n'))
|
|
{
|
|
--end;
|
|
}
|
|
|
|
m_count = end + 1;
|
|
m_buffer[m_count] = 0;
|
|
|
|
if (begin > 0)
|
|
{
|
|
set(&m_buffer[begin], m_count - begin);
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
struct SKeyValue
|
|
{
|
|
CFixedString<char, 256> key;
|
|
CFixedString<wchar_t, 256> value;
|
|
};
|
|
|
|
|
|
template<size_t CAPACITY>
|
|
class CKeyValueArray
|
|
{
|
|
private:
|
|
size_t count;
|
|
SKeyValue data[CAPACITY];
|
|
|
|
public:
|
|
CKeyValueArray()
|
|
: count(0)
|
|
{
|
|
}
|
|
|
|
size_t size() const
|
|
{
|
|
return count;
|
|
}
|
|
|
|
const SKeyValue& operator[](size_t i) const
|
|
{
|
|
return data[i];
|
|
}
|
|
|
|
void clear()
|
|
{
|
|
count = 0;
|
|
}
|
|
|
|
const SKeyValue* find(const char* key) const
|
|
{
|
|
for (size_t i = 0; i < count; ++i)
|
|
{
|
|
if (data[i].key.equals(key))
|
|
{
|
|
return &data[i];
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
SKeyValue* find(const char* key)
|
|
{
|
|
for (size_t i = 0; i < count; ++i)
|
|
{
|
|
if (data[i].key.equals(key))
|
|
{
|
|
return &data[i];
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
SKeyValue* set(const char* key, const wchar_t* value)
|
|
{
|
|
SKeyValue* p = find(key);
|
|
if (!p)
|
|
{
|
|
if (count >= CAPACITY)
|
|
{
|
|
return 0;
|
|
}
|
|
p = &data[count++];
|
|
p->key = key;
|
|
}
|
|
p->value = value;
|
|
return p;
|
|
}
|
|
};
|
|
|
|
|
|
#if defined(CRY_ENABLE_RC_HELPER)
|
|
|
|
bool Utf16ContainsAsciiOnly(const wchar_t* wstr);
|
|
|
|
void ConvertUtf16ToUtf8(const wchar_t* src, CCharBuffer dst);
|
|
|
|
void ConvertUtf8ToUtf16(const char* src, CWCharBuffer dst);
|
|
|
|
template <size_t CAPACITY>
|
|
void AddPathSeparator(CFixedString<wchar_t, CAPACITY>& wstr)
|
|
{
|
|
if (wstr.length() <= 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (wstr[wstr.length() - 1] == L'/' || wstr[wstr.length() - 1] == L'\\')
|
|
{
|
|
return;
|
|
}
|
|
|
|
wstr.appendAscii("/");
|
|
}
|
|
|
|
void GetAsciiFilename(const wchar_t* wfilename, CCharBuffer buffer);
|
|
|
|
#endif // #if defined(CRY_ENABLE_RC_HELPER)
|
|
} // namespace SettingsManagerHelpers
|
|
|
|
|
|
#if defined(CRY_ENABLE_RC_HELPER)
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// Provides settings and functions to make calls to RC.
|
|
class CEngineSettingsManager;
|
|
class CSettingsManagerTools
|
|
{
|
|
public:
|
|
CSettingsManagerTools(const wchar_t* szModuleName = 0);
|
|
~CSettingsManagerTools();
|
|
|
|
private:
|
|
CEngineSettingsManager* m_pSettingsManager;
|
|
|
|
public:
|
|
CEngineSettingsManager* GetEngineSettingsManager()
|
|
{
|
|
return m_pSettingsManager;
|
|
}
|
|
|
|
bool GetInstalledBuildPathUtf16(const int index, SettingsManagerHelpers::CWCharBuffer name, SettingsManagerHelpers::CWCharBuffer path);
|
|
bool GetInstalledBuildPathAscii(const int index, SettingsManagerHelpers::CCharBuffer name, SettingsManagerHelpers::CCharBuffer path);
|
|
|
|
void GetEditorExecutable(SettingsManagerHelpers::CWCharBuffer wbuffer);
|
|
bool CallEditor(void** pEditorWindow, void* hParent, const char* pWndName, const char* pFlag);
|
|
|
|
static bool Is64bitWindows();
|
|
};
|
|
|
|
#endif // #if defined(CRY_ENABLE_RC_HELPER)
|
|
|
|
#endif // CRYINCLUDE_CRYCOMMON_SETTINGSMANAGERHELPERS_H
|