6#define GLFW_INCLUDE_NONE
10 #define GLFW_EXPOSE_NATIVE_WIN32
11 #include <GLFW/glfw3native.h>
23 #include <Carbon/Carbon.h>
79Renderer* Application::renderer =
nullptr;
82int64_t Application::timeLast = -1L;
83bool Application::limitFPS =
true;
85GLFWwindow* Application::glfwWindow =
nullptr;
86array<unsigned int, 10> Application::glfwMouseButtonDownFrames;
87int Application::glfwMouseButtonLast = -1;
88bool Application::glfwCapsLockEnabled =
false;
89GLFWcursor* Application::glfwHandCursor =
nullptr;
91string Application::execute(
const string& command) {
93 array<char, 128> buffer;
96 shared_ptr<FILE> pipe(_popen(command.c_str(),
"r"), _pclose);
98 shared_ptr<FILE> pipe(popen(command.c_str(),
"r"), pclose);
100 if (!pipe)
throw std::runtime_error(
"popen() failed!");
101 while (!feof(pipe.get())) {
102 if (fgets(buffer.data(), buffer.size(), pipe.get()) !=
nullptr)
103 result += buffer.data();
110 system((
"start " + command).c_str());
112 system((command +
" </dev/null &>/dev/null &").c_str());
118 execute(
"explorer \"" + url +
"\"");
119 #elif defined(__APPLE__) || defined(__HAIKU__)
120 execute(
"open \"" + url +
"\"");
122 execute(
"xdg-open \"" + url +
"\"");
127 glfwSetWindowShouldClose(
glfwWindow, GLFW_FALSE);
138 glfwSetWindowShouldClose(
glfwWindow, GLFW_TRUE);
144 LONG WINAPI windowsExceptionHandler(
struct _EXCEPTION_POINTERS* exceptionInfo) {
154 HANDLE process = GetCurrentProcess();
155 HANDLE thread = GetCurrentThread();
157 vector<string> backtraceLines;
158 auto backtraceHeaderLine =
"windowsExceptionHandler(): process " + to_string((uint64_t)process) +
" crashed: Printing stacktrace(thread " + to_string((uint64_t)thread) +
")";
159 Console::println(backtraceHeaderLine);
160 backtraceLines.push_back(backtraceHeaderLine);
162 CHAR _pathToExecutable[MAX_PATH];
163 GetModuleFileName(GetModuleHandleW(NULL), _pathToExecutable, MAX_PATH);
164 string pathToExecutable = _pathToExecutable;
166 #if defined(_MSC_VER) == false
167 auto addr2lineToolCmd = StringTools::substring(pathToExecutable, 0, StringTools::lastIndexOf(pathToExecutable,
'\\')) +
"\\addr2line.exe";
168 if (FileSystem::getInstance()->fileExists(addr2lineToolCmd) ==
false) {
169 Console::println(
"handler(): " + addr2lineToolCmd +
": not found! Please copy addr2line utility to binary location!");
179 return EXCEPTION_EXECUTE_HANDLER;
181 addr2lineToolCmd =
"\"" + StringTools::replace(addr2lineToolCmd,
"\\",
"\\\\") +
"\"";
184 auto backtraceExecutableLine =
"windowsExceptionHandler(): path to executable: " + pathToExecutable;
185 Console::println(backtraceExecutableLine);
186 backtraceLines.push_back(backtraceExecutableLine);
188 pathToExecutable = string(
"\"") + StringTools::replace(pathToExecutable,
"\\",
"\\\\") +
"\"";
193 SYMOPT_OMAP_FIND_NEAREST |
194 SYMOPT_DEFERRED_LOADS |
204 STACKFRAME64 stackFrame;
205 memset(&stackFrame, 0,
sizeof(STACKFRAME64));
208 memset(&context, 0,
sizeof(context));
209 context.ContextFlags = CONTEXT_FULL;
210 RtlCaptureContext(&context);
214 image = IMAGE_FILE_MACHINE_I386;
215 stackFrame.AddrPC.Offset = context.Eip;
216 stackFrame.AddrPC.Mode = AddrModeFlat;
217 stackFrame.AddrFrame.Offset = context.Ebp;
218 stackFrame.AddrFrame.Mode = AddrModeFlat;
219 stackFrame.AddrStack.Offset = context.Esp;
220 stackFrame.AddrStack.Mode = AddrModeFlat;
222 image = IMAGE_FILE_MACHINE_AMD64;
223 stackFrame.AddrPC.Offset = context.Rip;
224 stackFrame.AddrPC.Mode = AddrModeFlat;
225 stackFrame.AddrFrame.Offset = context.Rsp;
226 stackFrame.AddrFrame.Mode = AddrModeFlat;
227 stackFrame.AddrStack.Offset = context.Rsp;
228 stackFrame.AddrStack.Mode = AddrModeFlat;
230 image = IMAGE_FILE_MACHINE_IA64;
231 stackFrame.AddrPC.Offset = context.StIIP;
232 stackFrame.AddrPC.Mode = AddrModeFlat;
233 stackFrame.AddrFrame.Offset = context.IntSp;
234 stackFrame.AddrFrame.Mode = AddrModeFlat;
235 stackFrame.AddrBStore.Offset = context.RsBSP;
236 stackFrame.AddrBStore.Mode = AddrModeFlat;
237 stackFrame.AddrStack.Offset = context.IntSp;
238 stackFrame.AddrStack.Mode = AddrModeFlat;
240 #error "Machine not supported"
245 while (StackWalk64(image, process, thread, &stackFrame, &context, NULL, &SymFunctionTableAccess64, &SymGetModuleBase64, NULL)) {
246 if (stackFrame.AddrPC.Offset == stackFrame.AddrReturn.Offset)
break;
247 if (stackFrame.AddrPC.Offset == 0)
continue;
250 char _fileName[1024];
251 char _functionName[1024];
255 #define cnBufferSize 1024
256 unsigned char byBuffer[
sizeof(IMAGEHLP_SYMBOL64) + cnBufferSize];
257 IMAGEHLP_SYMBOL64* pSymbol = (IMAGEHLP_SYMBOL64*)byBuffer;
258 DWORD64 dwDisplacement;
259 memset(pSymbol, 0,
sizeof(IMAGEHLP_SYMBOL64) + cnBufferSize);
260 pSymbol->SizeOfStruct =
sizeof(IMAGEHLP_SYMBOL64);
261 pSymbol->MaxNameLength = cnBufferSize;
262 if (!SymGetSymFromAddr64(process, stackFrame.AddrPC.Offset, &dwDisplacement, pSymbol)) {
263 strcpy(_functionName,
"??");
265 pSymbol->Name[cnBufferSize-1] =
'\0';
266 strcpy(_functionName, pSymbol->Name);
272 IMAGEHLP_LINE64 theLine;
273 DWORD dwDisplacement;
274 memset(&theLine, 0,
sizeof(theLine));
275 theLine.SizeOfStruct =
sizeof(theLine);
276 if(!SymGetLineFromAddr64(process, stackFrame.AddrPC.Offset, &dwDisplacement, &theLine))
278 strcpy(_fileName,
"??");
282 const char* pszFile = strrchr(theLine.FileName,
'\\');
283 if (!pszFile) pszFile = theLine.FileName;
else ++pszFile;
284 strncpy(_fileName, pszFile, cnBufferSize);
285 line = theLine.LineNumber;
290 string functionName = _functionName;
291 string fileName = _fileName;
292 #if defined(_MSC_VER) == false
293 if (functionName ==
"??") {
295 Hex::encodeInt(stackFrame.AddrPC.Offset, hexAddr);
296 string addr2LineCommand =
"\"" + addr2lineToolCmd +
" -f -p -e " + string(pathToExecutable) +
" " + hexAddr +
"\"";
301 auto addr2lineFunctionName = t.
nextToken();
302 addr2LineOutput = StringTools::replace(StringTools::replace(addr2LineOutput, addr2lineFunctionName, RTTI::demangle(addr2lineFunctionName.c_str())),
"\n",
"");
304 auto backtraceLine = to_string(frameIdx) +
": " + addr2LineOutput;
305 Console::println(backtraceLine);
306 backtraceLines.push_back(backtraceLine);
308 auto backtraceLine = to_string(frameIdx) +
": " + string(RTTI::demangle(_functionName)) +
" at " + fileName +
":" + to_string(line);
309 Console::println(backtraceLine);
310 backtraceLines.push_back(backtraceLine);
313 Console::println(to_string(frameIdx) +
": " + functionName +
" at " + fileName +
":" + to_string(line));
319 FileSystem::getInstance()->setContentFromStringArray(
".",
"crash.log", backtraceLines);
331 return EXCEPTION_EXECUTE_HANDLER;
344 Engine::renderer->setVSync(vSync);
345 if (Engine::renderer->getRendererType() != Renderer::RENDERERTYPE_VULKAN) {
346 glfwSwapInterval(vSync ==
true?1:0);
351 #if defined(__FreeBSD__)
353 #elif defined(__HAIKU__)
355 #elif defined(__linux__)
357 #elif defined(__APPLE__)
359 #elif defined(__NetBSD__)
361 #elif defined(__OpenBSD__)
363 #elif defined(_MSC_VER)
364 return "Windows-MSC";
365 #elif defined(_WIN32)
366 return "Windows-MINGW";
373 #if defined(__amd64__) || defined(_M_X64)
375 #elif defined(__ia64__) || defined(_M_IA64)
377 #elif defined(__aarch64__)
379 #elif defined(__arm__) || defined(_M_ARM)
381 #elif defined(__powerpc64__)
383 #elif defined(__powerpc__)
396 return GetActiveWindow() == GetForegroundWindow();
447 auto windowMonitor = glfwGetWindowMonitor(
glfwWindow);
448 if (windowMonitor ==
nullptr &&
fullScreen ==
true) {
449 auto monitor = glfwGetPrimaryMonitor();
450 auto mode = glfwGetVideoMode(monitor);
451 glfwSetWindowMonitor(
glfwWindow, monitor, 0, 0, mode->width, mode->height, mode->refreshRate);
453 if (windowMonitor !=
nullptr &&
fullScreen ==
false) {
466 SetUnhandledExceptionFilter(windowsExceptionHandler);
473 glfwSetInputMode(
glfwWindow, GLFW_CURSOR, GLFW_CURSOR_HIDDEN);
477 glfwSetInputMode(
glfwWindow, GLFW_CURSOR, GLFW_CURSOR_NORMAL);
481 glfwSetInputMode(
glfwWindow, GLFW_CURSOR, GLFW_CURSOR_NORMAL);
486 double mouseX, mouseY;
487 glfwGetCursorPos(
glfwWindow, &mouseX, &mouseY);
488 return static_cast<int>(mouseX);
492 double mouseX, mouseY;
493 glfwGetCursorPos(
glfwWindow, &mouseX, &mouseY);
494 return static_cast<int>(mouseY);
498 #if defined(__APPLE__)
499 int windowXPos, windowYPos;
502 point.x = windowXPos + x;
503 point.y = windowYPos + y;
504 CGWarpMouseCursorPosition(point);
505 CGAssociateMouseAndMouseCursorPosition(
true);
512 if (Engine::renderer->getRendererType() != Renderer::RENDERERTYPE_VULKAN) glfwSwapBuffers(
glfwWindow);
516 return string(glfwGetClipboardString(
glfwWindow));
520 glfwSetClipboardString(
glfwWindow, content.c_str());
524 Console::println(
string(
"glfwErrorCallback(): ") + description);
528 string rendererLibrary =
"libopengl3corerenderer";
529 for (
auto i = 1; i < argc; i++) {
530 auto argValue = string(argv[i]);
532 if (argValue ==
"--gles2") rendererLibrary =
"libopengles2renderer";
else
533 if (argValue ==
"--gl2") rendererLibrary =
"libopengl2renderer";
else
534 if (argValue ==
"--gl3core") rendererLibrary =
"libopengl3corerenderer";
else
535 if (argValue ==
"--vulkan") rendererLibrary =
"libvulkanrenderer";
539 rendererLibrary = rendererLibrary +
".dll";
540 #elif defined(__APPLE__)
541 rendererLibrary = rendererLibrary +
".dylib";
543 rendererLibrary = rendererLibrary +
".so";
552 if (glfwInit() ==
false) {
553 Console::println(
"glflInit(): failed!");
566 GLFWmonitor* monitor = glfwGetPrimaryMonitor();
567 const GLFWvidmode* monitorMode = glfwGetVideoMode(monitor);
568 if (monitorMode !=
nullptr) {
571 glfwGetMonitorPos(monitor, &monitorX, &monitorY);
577 Console::println(
"Application::run(): Opening renderer library: " + rendererLibrary);
580 #if defined(_MSC_VER)
582 auto rendererLibraryHandle = LoadLibrary(rendererLibrary.c_str());
583 if (rendererLibraryHandle ==
nullptr) {
584 Console::println(
"Application::run(): Could not open renderer library");
589 Renderer* (*rendererCreateInstance)() = (
Renderer*(*)())GetProcAddress(rendererLibraryHandle,
"createInstance");
591 if (rendererCreateInstance ==
nullptr) {
592 Console::println(
"Application::run(): Could not find renderer library createInstance() entry point");
599 Console::println(
"Application::run(): Could not create renderer");
605 #if defined(__HAIKU__)
606 auto rendererLibraryHandle = dlopen((
"lib/" + rendererLibrary).c_str(), RTLD_NOW);
608 auto rendererLibraryHandle = dlopen(rendererLibrary.c_str(), RTLD_NOW);
610 if (rendererLibraryHandle ==
nullptr) {
611 Console::println(
"Application::run(): Could not open renderer library");
616 Renderer* (*rendererCreateInstance)() = (
Renderer*(*)())dlsym(rendererLibraryHandle,
"createInstance");
618 if (rendererCreateInstance ==
nullptr) {
619 Console::println(
"Application::run(): Could not find renderer library createInstance() entry point");
626 Console::println(
"Application::run(): Could not create renderer");
646 Console::println(
"glfwCreateWindow(): Could not create window");
653 Console::println(
"glfwCreateWindow(): Could not initialize window system renderer context");
677 #if defined(__APPLE__)
679 auto executablePathName = string(argv[0]);
680 if (executablePathName.find(
".app/Contents/MacOS/") != string::npos) {
681 auto appBundleName = StringTools::substring(executablePathName, 0, executablePathName.rfind(
".app") +
string(
".app").size());
682 auto workingPathName = StringTools::substring(appBundleName, 0, appBundleName.rfind(
'/'));
683 FileSystem::getStandardFileSystem()->changePath(workingPathName);
686 while (glfwWindowShouldClose(
glfwWindow) ==
false) {
688 if (Engine::renderer->getRendererType() != Renderer::RENDERERTYPE_VULKAN) glfwSwapBuffers(
glfwWindow);
693 Console::println(
"Application::run(): Shutting down application");
702 auto logoFileName = StringTools::replace(StringTools::toLowerCase(
executableFileName),
".exe",
"") +
"-icon.png";
703 if (FileSystem::getInstance()->fileExists(
"resources/platforms/icons/" + logoFileName) ==
false) logoFileName =
"default-icon.png";
704 auto texture = TextureReader::read(
"resources/platforms/icons", logoFileName,
false,
false);
705 if (texture !=
nullptr) {
706 auto textureData = texture->getTextureData();
707 auto textureWidth = texture->getTextureWidth();
708 auto textureHeight = texture->getTextureHeight();
709 auto textureBytePerPixel = texture->getDepth() == 32?4:3;
710 auto glfwPixels =
new uint8_t[textureWidth * textureHeight * 4];
711 for (
auto y = 0; y < textureHeight; y++)
712 for (
auto x = 0; x < textureWidth; x++) {
713 glfwPixels[y * textureWidth * 4 + x * 4 + 0] = textureData->get(y * textureWidth * textureBytePerPixel + x * textureBytePerPixel + 0);
714 glfwPixels[y * textureWidth * 4 + x * 4 + 1] = textureData->get(y * textureWidth * textureBytePerPixel + x * textureBytePerPixel + 1);
715 glfwPixels[y * textureWidth * 4 + x * 4 + 2] = textureData->get(y * textureWidth * textureBytePerPixel + x * textureBytePerPixel + 2);
716 glfwPixels[y * textureWidth * 4 + x * 4 + 3] = textureBytePerPixel == 3?255:textureData->get(y * textureWidth * textureBytePerPixel + x * textureBytePerPixel + 3);
719 glfwIcon.width = texture->getTextureWidth();
720 glfwIcon.height = texture->getTextureHeight();
721 glfwIcon.pixels = glfwPixels;
723 texture->releaseReference();
724 delete [] glfwPixels;
735 int64_t timeNow = Time::getCurrentMillis();
738 int64_t timePassed = timeNow -
timeLast;
739 if (
limitFPS ==
true && timePassed < timeFrame) Thread::sleep(timeFrame - timePassed);
755 double mouseX, mouseY;
756 glfwGetCursorPos(window, &mouseX, &mouseY);
762 double mouseX, mouseY;
763 glfwGetCursorPos(window, &mouseX, &mouseY);
765 if (key == GLFW_KEY_CAPS_LOCK) {
766 if (action == GLFW_PRESS) {
770 if (action == GLFW_PRESS || action == GLFW_REPEAT) {
771 auto keyName = key == GLFW_KEY_SPACE?
" ":glfwGetKeyName(key, scanCode);
773 keyName ==
nullptr?-1:((mods & GLFW_MOD_SHIFT) == 0 &&
glfwCapsLockEnabled ==
false?Character::toLowerCase(keyName[0]):keyName[0]),
779 if (action == GLFW_RELEASE) {
780 auto keyName = key == GLFW_KEY_SPACE?
" ":glfwGetKeyName(key, scanCode);
782 keyName ==
nullptr?-1:((mods & GLFW_MOD_SHIFT) == 0 &&
glfwCapsLockEnabled ==
false?Character::toLowerCase(keyName[0]):keyName[0]),
801 double mouseX, mouseY;
802 glfwGetCursorPos(window, &mouseX, &mouseY);
804 if (action == GLFW_PRESS) {
814 double mouseX, mouseY;
815 glfwGetCursorPos(window, &mouseX, &mouseY);
static void glfwErrorCallback(int error, const char *description)
#define MOUSE_CURSOR_HAND
#define MOUSE_CURSOR_ENABLED
#define MOUSE_CURSOR_DISABLED
Application base class, please make sure to allocate application on heap to have correct application ...
static void setMousePosition(int x, int y)
Set mouse position.
int getWindowYPosition() const
static STATIC_DLL_IMPEXT Renderer * renderer
static STATIC_DLL_IMPEXT int glfwMouseButtonLast
static void glfwOnMouseWheel(GLFWwindow *window, double x, double y)
GLFW on key.
static string getCPUName()
static string getOSName()
static void executeBackground(const string &command)
Execute a command in background.
static STATIC_DLL_IMPEXT Application * application
static void swapBuffers()
Swap rendering buffers.
void setIcon()
Set application icon.
virtual void reshape(int width, int height)=0
Resize.
static void glfwOnMouseMoved(GLFWwindow *window, double x, double y)
GLFW on mouse moved.
int getWindowWidth() const
void setWindowXPosition(int windowXPosition)
Set window X position when initializing.
static void glfwOnClose(GLFWwindow *window)
GLFW on close.
string executableFileName
static void glfwOnWindowResize(GLFWwindow *window, int width, int height)
GLFW on window resize.
bool isFullScreen() const
static STATIC_DLL_IMPEXT array< unsigned int, 10 > glfwMouseButtonDownFrames
void setWindowYPosition(int windowYPosition)
Set window Y position when initializing.
static string execute(const string &command)
Execute a command and wait until it finished running.
void setClipboardContent(const string &content)
Set clipboard content.
static void displayInternal()
Display function.
int getWindowXPosition() const
static void glfwOnChar(GLFWwindow *window, unsigned int key)
GLFW on char.
static void reshapeInternal(int width, int height)
Reshape function.
void setWindowWidth(int windowWidth)
Set window width.
static STATIC_DLL_IMPEXT bool limitFPS
virtual ~Application()
Destructor.
static void cancelExit()
Cancels a users requested exit (ALT-F4 or X button)
int getWindowHeight() const
virtual void onClose()
On close.
static STATIC_DLL_IMPEXT bool glfwCapsLockEnabled
string getClipboardContent()
static void installExceptionHandler()
Windows only: Install exception handler that will print a stack trace if crashing.
virtual void display()=0
Display.
static STATIC_DLL_IMPEXT int64_t timeLast
void setFullScreen(bool fullScreen)
Set full screen mode.
static constexpr int WINDOW_HINT_MAXIMIZED
static STATIC_DLL_IMPEXT InputEventHandler * inputEventHandler
static int getMousePositionY()
static int getMousePositionX()
virtual void initialize()=0
Init.
static constexpr int WINDOW_HINT_NOTDECORATED
static constexpr int WINDOW_HINT_INVISIBLE
static void exit(int exitCode)
Exits this application with given exit code.
static STATIC_DLL_IMPEXT GLFWwindow * glfwWindow
static constexpr int WINDOW_HINT_NOTRESIZEABLE
static void setVSyncEnabled(bool vSync)
Set vsync enabled.
void setWindowHeight(int windowHeight)
Set window height.
virtual void dispose()=0
Disposes.
static void glfwOnKey(GLFWwindow *window, int key, int scanCode, int action, int mods)
GLFW on key.
static void openBrowser(const string &url)
Open browser with given url.
void setInputEventHandler(InputEventHandler *inputEventHandler)
Set input event handler.
void run(int argc, char **argv, const string &title, InputEventHandler *inputEventHandler=nullptr, int windowHints=WINDOW_HINT_NONE)
Run this application.
Application()
Public constructor.
static void glfwOnMouseButton(GLFWwindow *window, int button, int action, int mods)
GLFW on key.
static void setMouseCursor(int mouseCursor)
Set mouse cursor.
static STATIC_DLL_IMPEXT GLFWcursor * glfwHandCursor
virtual bool initializeWindowSystemRendererContext(GLFWwindow *glfwWindow)=0
Initialize window system renderer context.
virtual bool prepareWindowSystemRendererContext(int tryIdx)=0
Prepare window system renderer context.
File system singleton class.
Integer to hex string conversion utility class.
Run time type information utility class.
const string & nextToken()
void tokenize(const string &str, const string &delimiters)
Tokenize.