2019-11-23 20:27:39 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2010-2012 cocos2d-x.org
|
|
|
|
Copyright (c) 2013-2016 Chukong Technologies Inc.
|
|
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
2021-10-09 13:48:56 +08:00
|
|
|
Copyright (c) 2021 Bytedance Inc.
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-10-09 13:48:56 +08:00
|
|
|
https://adxe.org
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
****************************************************************************/
|
|
|
|
#include "platform/CCApplication.h"
|
|
|
|
#include "base/CCDirector.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include "platform/CCFileUtils.h"
|
|
|
|
#include <shellapi.h>
|
|
|
|
#include <WinVer.h>
|
2021-10-09 13:48:56 +08:00
|
|
|
#include <timeapi.h>
|
2021-10-09 14:05:32 +08:00
|
|
|
|
|
|
|
#include "yasio/cxx17/string_view.hpp"
|
2021-11-26 17:19:50 +08:00
|
|
|
#include "ntcvt/ntcvt.hpp"
|
2019-11-23 20:27:39 +08:00
|
|
|
/**
|
|
|
|
@brief This function change the PVRFrame show/hide setting in register.
|
|
|
|
@param bEnable If true show the PVRFrame window, otherwise hide.
|
|
|
|
*/
|
|
|
|
static void PVRFrameEnableControlWindow(bool bEnable);
|
|
|
|
|
|
|
|
NS_CC_BEGIN
|
|
|
|
|
|
|
|
// sharedApplication pointer
|
2021-12-25 10:04:45 +08:00
|
|
|
Application* Application::sm_pSharedApplication = nullptr;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
Application::Application() : _instance(nullptr), _accelTable(nullptr)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
_instance = GetModuleHandle(nullptr);
|
2019-11-23 20:27:39 +08:00
|
|
|
_animationInterval.QuadPart = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
CC_ASSERT(!sm_pSharedApplication);
|
2019-11-23 20:27:39 +08:00
|
|
|
sm_pSharedApplication = this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Application::~Application()
|
|
|
|
{
|
|
|
|
CC_ASSERT(this == sm_pSharedApplication);
|
|
|
|
sm_pSharedApplication = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Application::run()
|
|
|
|
{
|
|
|
|
PVRFrameEnableControlWindow(false);
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
/////////////// changing timer resolution
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
2021-12-25 10:04:45 +08:00
|
|
|
UINT TARGET_RESOLUTION = 1; // 1 millisecond target resolution
|
2019-11-23 20:27:39 +08:00
|
|
|
TIMECAPS tc;
|
|
|
|
UINT wTimerRes = 0;
|
|
|
|
if (TIMERR_NOERROR == timeGetDevCaps(&tc, sizeof(TIMECAPS)))
|
|
|
|
{
|
|
|
|
wTimerRes = std::min(std::max(tc.wPeriodMin, TARGET_RESOLUTION), tc.wPeriodMax);
|
|
|
|
timeBeginPeriod(wTimerRes);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Main message loop:
|
|
|
|
LARGE_INTEGER nLast;
|
|
|
|
LARGE_INTEGER nNow;
|
|
|
|
|
|
|
|
QueryPerformanceCounter(&nLast);
|
|
|
|
|
|
|
|
initGLContextAttrs();
|
|
|
|
|
|
|
|
// Initialize instance and cocos2d.
|
|
|
|
if (!applicationDidFinishLaunching())
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto director = Director::getInstance();
|
2021-12-25 10:04:45 +08:00
|
|
|
auto glview = director->getOpenGLView();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
// Retain glview to avoid glview being released in the while loop
|
|
|
|
glview->retain();
|
|
|
|
|
|
|
|
LONGLONG interval = 0LL;
|
2021-12-25 10:04:45 +08:00
|
|
|
LONG waitMS = 0L;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
LARGE_INTEGER freq;
|
|
|
|
QueryPerformanceFrequency(&freq);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
while (!glview->windowShouldClose())
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
QueryPerformanceCounter(&nNow);
|
|
|
|
interval = nNow.QuadPart - nLast.QuadPart;
|
|
|
|
if (interval >= _animationInterval.QuadPart)
|
|
|
|
{
|
|
|
|
nLast.QuadPart = nNow.QuadPart;
|
|
|
|
director->mainLoop();
|
|
|
|
glview->pollEvents();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// The precision of timer on Windows is set to highest (1ms) by 'timeBeginPeriod' from above code,
|
|
|
|
// but it's still not precise enough. For example, if the precision of timer is 1ms,
|
|
|
|
// Sleep(3) may make a sleep of 2ms or 4ms. Therefore, we subtract 1ms here to make Sleep time shorter.
|
|
|
|
// If 'waitMS' is equal or less than 1ms, don't sleep and run into next loop to
|
|
|
|
// boost CPU to next frame accurately.
|
|
|
|
waitMS = static_cast<LONG>((_animationInterval.QuadPart - interval) * 1000LL / freq.QuadPart - 1L);
|
|
|
|
if (waitMS > 1L)
|
|
|
|
Sleep(waitMS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Director should still do a cleanup if the window was closed manually.
|
|
|
|
if (glview->isOpenGLReady())
|
|
|
|
{
|
|
|
|
director->end();
|
|
|
|
director->mainLoop();
|
|
|
|
director = nullptr;
|
|
|
|
}
|
|
|
|
glview->release();
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
/////////////// restoring timer resolution
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
if (wTimerRes != 0)
|
|
|
|
{
|
|
|
|
timeEndPeriod(wTimerRes);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Application::setAnimationInterval(float interval)
|
|
|
|
{
|
|
|
|
LARGE_INTEGER freq;
|
|
|
|
QueryPerformanceFrequency(&freq);
|
|
|
|
_animationInterval.QuadPart = (LONGLONG)(interval * freq.QuadPart);
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// static member function
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
Application* Application::getInstance()
|
|
|
|
{
|
|
|
|
CC_ASSERT(sm_pSharedApplication);
|
|
|
|
return sm_pSharedApplication;
|
|
|
|
}
|
|
|
|
|
|
|
|
LanguageType Application::getCurrentLanguage()
|
|
|
|
{
|
|
|
|
LanguageType ret = LanguageType::ENGLISH;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
|
|
LCID localeID = GetUserDefaultLCID();
|
2019-11-23 20:27:39 +08:00
|
|
|
unsigned short primaryLanguageID = localeID & 0xFF;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
switch (primaryLanguageID)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
case LANG_CHINESE:
|
|
|
|
ret = LanguageType::CHINESE;
|
|
|
|
break;
|
|
|
|
case LANG_ENGLISH:
|
|
|
|
ret = LanguageType::ENGLISH;
|
|
|
|
break;
|
|
|
|
case LANG_FRENCH:
|
|
|
|
ret = LanguageType::FRENCH;
|
|
|
|
break;
|
|
|
|
case LANG_ITALIAN:
|
|
|
|
ret = LanguageType::ITALIAN;
|
|
|
|
break;
|
|
|
|
case LANG_GERMAN:
|
|
|
|
ret = LanguageType::GERMAN;
|
|
|
|
break;
|
|
|
|
case LANG_SPANISH:
|
|
|
|
ret = LanguageType::SPANISH;
|
|
|
|
break;
|
|
|
|
case LANG_DUTCH:
|
|
|
|
ret = LanguageType::DUTCH;
|
|
|
|
break;
|
|
|
|
case LANG_RUSSIAN:
|
|
|
|
ret = LanguageType::RUSSIAN;
|
|
|
|
break;
|
|
|
|
case LANG_KOREAN:
|
|
|
|
ret = LanguageType::KOREAN;
|
|
|
|
break;
|
|
|
|
case LANG_JAPANESE:
|
|
|
|
ret = LanguageType::JAPANESE;
|
|
|
|
break;
|
|
|
|
case LANG_HUNGARIAN:
|
|
|
|
ret = LanguageType::HUNGARIAN;
|
|
|
|
break;
|
|
|
|
case LANG_PORTUGUESE:
|
|
|
|
ret = LanguageType::PORTUGUESE;
|
|
|
|
break;
|
|
|
|
case LANG_ARABIC:
|
|
|
|
ret = LanguageType::ARABIC;
|
|
|
|
break;
|
|
|
|
case LANG_NORWEGIAN:
|
|
|
|
ret = LanguageType::NORWEGIAN;
|
|
|
|
break;
|
|
|
|
case LANG_POLISH:
|
|
|
|
ret = LanguageType::POLISH;
|
|
|
|
break;
|
|
|
|
case LANG_TURKISH:
|
|
|
|
ret = LanguageType::TURKISH;
|
|
|
|
break;
|
|
|
|
case LANG_UKRAINIAN:
|
|
|
|
ret = LanguageType::UKRAINIAN;
|
|
|
|
break;
|
|
|
|
case LANG_ROMANIAN:
|
|
|
|
ret = LanguageType::ROMANIAN;
|
|
|
|
break;
|
|
|
|
case LANG_BULGARIAN:
|
|
|
|
ret = LanguageType::BULGARIAN;
|
|
|
|
break;
|
|
|
|
case LANG_BELARUSIAN:
|
|
|
|
ret = LanguageType::BELARUSIAN;
|
|
|
|
break;
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
const char* Application::getCurrentLanguageCode()
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
LANGID lid = GetUserDefaultUILanguage();
|
2019-11-23 20:27:39 +08:00
|
|
|
const LCID locale_id = MAKELCID(lid, SORT_DEFAULT);
|
2021-12-25 10:04:45 +08:00
|
|
|
static char code[3] = {0};
|
2019-11-23 20:27:39 +08:00
|
|
|
GetLocaleInfoA(locale_id, LOCALE_SISO639LANGNAME, code, sizeof(code));
|
|
|
|
code[2] = '\0';
|
|
|
|
return code;
|
|
|
|
}
|
|
|
|
|
|
|
|
Application::Platform Application::getTargetPlatform()
|
|
|
|
{
|
|
|
|
return Platform::OS_WINDOWS;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string Application::getVersion()
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
char verString[256] = {0};
|
2019-11-23 20:27:39 +08:00
|
|
|
TCHAR szVersionFile[MAX_PATH];
|
|
|
|
GetModuleFileName(NULL, szVersionFile, MAX_PATH);
|
2021-12-25 10:04:45 +08:00
|
|
|
DWORD verHandle = NULL;
|
|
|
|
UINT size = 0;
|
2019-11-23 20:27:39 +08:00
|
|
|
LPBYTE lpBuffer = NULL;
|
2021-12-25 10:04:45 +08:00
|
|
|
DWORD verSize = GetFileVersionInfoSize(szVersionFile, &verHandle);
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
if (verSize != NULL)
|
|
|
|
{
|
|
|
|
LPSTR verData = new char[verSize];
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
if (GetFileVersionInfo(szVersionFile, verHandle, verSize, verData))
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
if (VerQueryValue(verData, L"\\", (VOID FAR * FAR*)&lpBuffer, &size))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
if (size)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
VS_FIXEDFILEINFO* verInfo = (VS_FIXEDFILEINFO*)lpBuffer;
|
2019-11-23 20:27:39 +08:00
|
|
|
if (verInfo->dwSignature == 0xfeef04bd)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
// Doesn't matter if you are on 32 bit or 64 bit,
|
|
|
|
// DWORD is always 32 bits, so first two revision numbers
|
|
|
|
// come from dwFileVersionMS, last two come from dwFileVersionLS
|
|
|
|
sprintf(verString, "%d.%d.%d.%d", (verInfo->dwFileVersionMS >> 16) & 0xffff,
|
2021-12-25 10:04:45 +08:00
|
|
|
(verInfo->dwFileVersionMS >> 0) & 0xffff, (verInfo->dwFileVersionLS >> 16) & 0xffff,
|
|
|
|
(verInfo->dwFileVersionLS >> 0) & 0xffff);
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete[] verData;
|
|
|
|
}
|
|
|
|
return verString;
|
|
|
|
}
|
|
|
|
|
2021-12-31 12:12:40 +08:00
|
|
|
bool Application::openURL(std::string_view url)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-10-09 14:05:32 +08:00
|
|
|
std::wstring wURL = ntcvt::from_chars(url, CP_UTF8);
|
|
|
|
HINSTANCE r = ShellExecuteW(NULL, L"open", wURL.c_str(), NULL, NULL, SW_SHOWNORMAL);
|
2021-12-25 10:04:45 +08:00
|
|
|
return (size_t)r > 32;
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
2021-12-31 12:12:40 +08:00
|
|
|
void Application::setResourceRootPath(std::string_view rootResDir)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
_resourceRootPath = rootResDir;
|
|
|
|
std::replace(_resourceRootPath.begin(), _resourceRootPath.end(), '\\', '/');
|
|
|
|
if (_resourceRootPath[_resourceRootPath.length() - 1] != '/')
|
|
|
|
{
|
|
|
|
_resourceRootPath += '/';
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
FileUtils* pFileUtils = FileUtils::getInstance();
|
2019-11-23 20:27:39 +08:00
|
|
|
std::vector<std::string> searchPaths = pFileUtils->getSearchPaths();
|
|
|
|
searchPaths.insert(searchPaths.begin(), _resourceRootPath);
|
|
|
|
pFileUtils->setSearchPaths(searchPaths);
|
|
|
|
}
|
|
|
|
|
2021-12-31 12:12:40 +08:00
|
|
|
std::string_view Application::getResourceRootPath(void)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
return _resourceRootPath;
|
|
|
|
}
|
|
|
|
|
2021-12-31 12:12:40 +08:00
|
|
|
void Application::setStartupScriptFilename(std::string_view startupScriptFile)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
_startupScriptFilename = startupScriptFile;
|
|
|
|
std::replace(_startupScriptFilename.begin(), _startupScriptFilename.end(), '\\', '/');
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_CC_END
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// Local function
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
static void PVRFrameEnableControlWindow(bool bEnable)
|
|
|
|
{
|
|
|
|
HKEY hKey = 0;
|
|
|
|
|
|
|
|
// Open PVRFrame control key, if not exist create it.
|
2021-12-25 10:04:45 +08:00
|
|
|
if (ERROR_SUCCESS != RegCreateKeyExW(HKEY_CURRENT_USER, L"Software\\Imagination Technologies\\PVRVFRame\\STARTUP\\",
|
|
|
|
0, 0, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, 0, &hKey, nullptr))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-10-09 14:05:32 +08:00
|
|
|
using namespace cxx17;
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
const WCHAR* wszValue = L"hide_gui";
|
2021-12-25 10:04:45 +08:00
|
|
|
const auto svNewData = (bEnable) ? L"NO"_sv : L"YES"_sv;
|
2019-11-23 20:27:39 +08:00
|
|
|
WCHAR wszOldData[256] = {0};
|
2021-12-25 10:04:45 +08:00
|
|
|
DWORD dwSize = static_cast<DWORD>(sizeof(wszOldData));
|
|
|
|
LSTATUS status = RegQueryValueExW(hKey, wszValue, 0, nullptr, (LPBYTE)wszOldData, &dwSize);
|
|
|
|
if (ERROR_FILE_NOT_FOUND == status // the key not exist
|
|
|
|
|| (ERROR_SUCCESS == status // or the hide_gui value is exist
|
2021-10-09 14:05:32 +08:00
|
|
|
&& svNewData != wszOldData)) // but new data and old data not equal
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-10-09 14:05:32 +08:00
|
|
|
dwSize = static_cast<DWORD>(sizeof(WCHAR) * (svNewData.length() + 1));
|
|
|
|
RegSetValueEx(hKey, wszValue, 0, REG_SZ, (const BYTE*)svNewData.data(), dwSize);
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
}
|